1 // SPDX-License-Identifier: GPL-2.0+
3 * Driver to expose SEC4 PRNG via crypto RNG API
9 #include <linux/completion.h>
10 #include <crypto/internal/rng.h>
14 #include "desc_constr.h"
19 * Length of used descriptors, see caam_init_desc()
21 #define CAAM_PRNG_MAX_DESC_LEN (CAAM_CMD_SZ + \
23 CAAM_CMD_SZ + CAAM_PTR_SZ_MAX)
25 /* prng per-device context */
26 struct caam_prng_ctx {
28 struct completion done;
31 struct caam_prng_alg {
36 static void caam_prng_done(struct device *jrdev, u32 *desc, u32 err,
39 struct caam_prng_ctx *jctx = context;
41 jctx->err = err ? caam_jr_strstatus(jrdev, err) : 0;
43 complete(&jctx->done);
46 static u32 *caam_init_reseed_desc(u32 *desc)
48 init_job_desc(desc, 0); /* + 1 cmd_sz */
49 /* Generate random bytes: + 1 cmd_sz */
50 append_operation(desc, OP_TYPE_CLASS1_ALG | OP_ALG_ALGSEL_RNG |
53 print_hex_dump_debug("prng reseed desc@: ", DUMP_PREFIX_ADDRESS,
54 16, 4, desc, desc_bytes(desc), 1);
59 static u32 *caam_init_prng_desc(u32 *desc, dma_addr_t dst_dma, u32 len)
61 init_job_desc(desc, 0); /* + 1 cmd_sz */
62 /* Generate random bytes: + 1 cmd_sz */
63 append_operation(desc, OP_ALG_ALGSEL_RNG | OP_TYPE_CLASS1_ALG);
64 /* Store bytes: + 1 cmd_sz + caam_ptr_sz */
65 append_fifo_store(desc, dst_dma,
66 len, FIFOST_TYPE_RNGSTORE);
68 print_hex_dump_debug("prng job desc@: ", DUMP_PREFIX_ADDRESS,
69 16, 4, desc, desc_bytes(desc), 1);
74 static int caam_prng_generate(struct crypto_rng *tfm,
75 const u8 *src, unsigned int slen,
76 u8 *dst, unsigned int dlen)
78 struct caam_prng_ctx ctx;
85 buf = kzalloc(dlen, GFP_KERNEL);
89 jrdev = caam_jr_alloc();
90 ret = PTR_ERR_OR_ZERO(jrdev);
92 pr_err("Job Ring Device allocation failed\n");
97 desc = kzalloc(CAAM_PRNG_MAX_DESC_LEN, GFP_KERNEL | GFP_DMA);
103 dst_dma = dma_map_single(jrdev, buf, dlen, DMA_FROM_DEVICE);
104 if (dma_mapping_error(jrdev, dst_dma)) {
105 dev_err(jrdev, "Failed to map destination buffer memory\n");
110 init_completion(&ctx.done);
111 ret = caam_jr_enqueue(jrdev,
112 caam_init_prng_desc(desc, dst_dma, dlen),
113 caam_prng_done, &ctx);
115 if (ret == -EINPROGRESS) {
116 wait_for_completion(&ctx.done);
120 dma_unmap_single(jrdev, dst_dma, dlen, DMA_FROM_DEVICE);
123 memcpy(dst, buf, dlen);
132 static void caam_prng_exit(struct crypto_tfm *tfm) {}
134 static int caam_prng_init(struct crypto_tfm *tfm)
139 static int caam_prng_seed(struct crypto_rng *tfm,
140 const u8 *seed, unsigned int slen)
142 struct caam_prng_ctx ctx;
143 struct device *jrdev;
148 pr_err("Seed length should be zero\n");
152 jrdev = caam_jr_alloc();
153 ret = PTR_ERR_OR_ZERO(jrdev);
155 pr_err("Job Ring Device allocation failed\n");
159 desc = kzalloc(CAAM_PRNG_MAX_DESC_LEN, GFP_KERNEL | GFP_DMA);
165 init_completion(&ctx.done);
166 ret = caam_jr_enqueue(jrdev,
167 caam_init_reseed_desc(desc),
168 caam_prng_done, &ctx);
170 if (ret == -EINPROGRESS) {
171 wait_for_completion(&ctx.done);
180 static struct caam_prng_alg caam_prng_alg = {
182 .generate = caam_prng_generate,
183 .seed = caam_prng_seed,
186 .cra_name = "stdrng",
187 .cra_driver_name = "prng-caam",
189 .cra_ctxsize = sizeof(struct caam_prng_ctx),
190 .cra_module = THIS_MODULE,
191 .cra_init = caam_prng_init,
192 .cra_exit = caam_prng_exit,
197 void caam_prng_unregister(void *data)
199 if (caam_prng_alg.registered)
200 crypto_unregister_rng(&caam_prng_alg.rng);
203 int caam_prng_register(struct device *ctrldev)
205 struct caam_drv_private *priv = dev_get_drvdata(ctrldev);
209 /* Check for available RNG blocks before registration */
211 rng_inst = (rd_reg32(&priv->jr[0]->perfmon.cha_num_ls) &
212 CHA_ID_LS_RNG_MASK) >> CHA_ID_LS_RNG_SHIFT;
214 rng_inst = rd_reg32(&priv->jr[0]->vreg.rng) & CHA_VER_NUM_MASK;
217 dev_dbg(ctrldev, "RNG block is not available... skipping registering algorithm\n");
221 ret = crypto_register_rng(&caam_prng_alg.rng);
224 "couldn't register rng crypto alg: %d\n",
229 caam_prng_alg.registered = true;
232 "rng crypto API alg registered %s\n", caam_prng_alg.rng.base.cra_driver_name);