// SPDX-License-Identifier: GPL-2.0+ /* * caam - Freescale FSL CAAM support for hw_random * * Copyright 2011 Freescale Semiconductor, Inc. * Copyright 2018-2019 NXP * * Based on caamalg.c crypto API driver. * */ #include <linux/hw_random.h> #include <linux/completion.h> #include <linux/atomic.h> #include <linux/kfifo.h> #include "compat.h" #include "regs.h" #include "intern.h" #include "desc_constr.h" #include "jr.h" #include "error.h" #define CAAM_RNG_MAX_FIFO_STORE_SIZE 16 /* * Length of used descriptors, see caam_init_desc() */ #define CAAM_RNG_DESC_LEN (CAAM_CMD_SZ + \ CAAM_CMD_SZ + \ CAAM_CMD_SZ + CAAM_PTR_SZ_MAX) /* rng per-device context */ struct caam_rng_ctx { struct hwrng rng; struct device *jrdev; struct device *ctrldev; void *desc_async; void *desc_sync; struct work_struct worker; struct kfifo fifo; }; struct caam_rng_job_ctx { struct completion *done; int *err; }; static struct caam_rng_ctx *to_caam_rng_ctx(struct hwrng *r) { return (struct caam_rng_ctx *)r->priv; } static void caam_rng_done(struct device *jrdev, u32 *desc, u32 err, void *context) { struct caam_rng_job_ctx *jctx = context; if (err) *jctx->err = caam_jr_strstatus(jrdev, err); complete(jctx->done); } static u32 *caam_init_desc(u32 *desc, dma_addr_t dst_dma) { init_job_desc(desc, 0); /* + 1 cmd_sz */ /* Generate random bytes: + 1 cmd_sz */ append_operation(desc, OP_ALG_ALGSEL_RNG | OP_TYPE_CLASS1_ALG | OP_ALG_PR_ON); /* Store bytes: + 1 cmd_sz + caam_ptr_sz */ append_fifo_store(desc, dst_dma, CAAM_RNG_MAX_FIFO_STORE_SIZE, FIFOST_TYPE_RNGSTORE); print_hex_dump_debug("rng job desc@: ", DUMP_PREFIX_ADDRESS, 16, 4, desc, desc_bytes(desc), 1); return desc; } static int caam_rng_read_one(struct device *jrdev, void *dst, int len, void *desc, struct completion *done) { dma_addr_t dst_dma; int err, ret = 0; struct caam_rng_job_ctx jctx = { .done = done, .err = &ret, }; len = CAAM_RNG_MAX_FIFO_STORE_SIZE; dst_dma = dma_map_single(jrdev, dst, len, DMA_FROM_DEVICE); if (dma_mapping_error(jrdev, dst_dma)) { dev_err(jrdev, "unable to map destination memory\n"); return -ENOMEM; } init_completion(done); err = caam_jr_enqueue(jrdev, caam_init_desc(desc, dst_dma), caam_rng_done, &jctx); if (err == -EINPROGRESS) { wait_for_completion(done); err = 0; } dma_unmap_single(jrdev, dst_dma, len, DMA_FROM_DEVICE); return err ?: (ret ?: len); } static void caam_rng_fill_async(struct caam_rng_ctx *ctx) { struct scatterlist sg[1]; struct completion done; int len, nents; sg_init_table(sg, ARRAY_SIZE(sg)); nents = kfifo_dma_in_prepare(&ctx->fifo, sg, ARRAY_SIZE(sg), CAAM_RNG_MAX_FIFO_STORE_SIZE); if (!nents) return; len = caam_rng_read_one(ctx->jrdev, sg_virt(&sg[0]), sg[0].length, ctx->desc_async, &done); if (len < 0) return; kfifo_dma_in_finish(&ctx->fifo, len); } static void caam_rng_worker(struct work_struct *work) { struct caam_rng_ctx *ctx = container_of(work, struct caam_rng_ctx, worker); caam_rng_fill_async(ctx); } static int caam_read(struct hwrng *rng, void *dst, size_t max, bool wait) { struct caam_rng_ctx *ctx = to_caam_rng_ctx(rng); int out; if (wait) { struct completion done; return caam_rng_read_one(ctx->jrdev, dst, max, ctx->desc_sync, &done); } out = kfifo_out(&ctx->fifo, dst, max); if (kfifo_is_empty(&ctx->fifo)) schedule_work(&ctx->worker); return out; } static void caam_cleanup(struct hwrng *rng) { struct caam_rng_ctx *ctx = to_caam_rng_ctx(rng); flush_work(&ctx->worker); caam_jr_free(ctx->jrdev); kfifo_free(&ctx->fifo); } static int caam_init(struct hwrng *rng) { struct caam_rng_ctx *ctx = to_caam_rng_ctx(rng); int err; ctx->desc_sync = devm_kzalloc(ctx->ctrldev, CAAM_RNG_DESC_LEN, GFP_DMA | GFP_KERNEL); if (!ctx->desc_sync) return -ENOMEM; ctx->desc_async = devm_kzalloc(ctx->ctrldev, CAAM_RNG_DESC_LEN, GFP_DMA | GFP_KERNEL); if (!ctx->desc_async) return -ENOMEM; if (kfifo_alloc(&ctx->fifo, CAAM_RNG_MAX_FIFO_STORE_SIZE, GFP_DMA | GFP_KERNEL)) return -ENOMEM; INIT_WORK(&ctx->worker, caam_rng_worker); ctx->jrdev = caam_jr_alloc(); err = PTR_ERR_OR_ZERO(ctx->jrdev); if (err) { kfifo_free(&ctx->fifo); pr_err("Job Ring Device allocation for transform failed\n"); return err; } /* * Fill async buffer to have early randomness data for * hw_random */ caam_rng_fill_async(ctx); return 0; } int caam_rng_init(struct device *ctrldev); void caam_rng_exit(struct device *ctrldev) { devres_release_group(ctrldev, caam_rng_init); } int caam_rng_init(struct device *ctrldev) { struct caam_rng_ctx *ctx; u32 rng_inst; struct caam_drv_private *priv = dev_get_drvdata(ctrldev); int ret; /* Check for an instantiated RNG before registration */ if (priv->era < 10) rng_inst = (rd_reg32(&priv->ctrl->perfmon.cha_num_ls) & CHA_ID_LS_RNG_MASK) >> CHA_ID_LS_RNG_SHIFT; else rng_inst = rd_reg32(&priv->ctrl->vreg.rng) & CHA_VER_NUM_MASK; if (!rng_inst) return 0; if (!devres_open_group(ctrldev, caam_rng_init, GFP_KERNEL)) return -ENOMEM; ctx = devm_kzalloc(ctrldev, sizeof(*ctx), GFP_KERNEL); if (!ctx) return -ENOMEM; ctx->ctrldev = ctrldev; ctx->rng.name = "rng-caam"; ctx->rng.init = caam_init; ctx->rng.cleanup = caam_cleanup; ctx->rng.read = caam_read; ctx->rng.priv = (unsigned long)ctx; ctx->rng.quality = 1024; dev_info(ctrldev, "registering rng-caam\n"); ret = devm_hwrng_register(ctrldev, &ctx->rng); if (ret) { caam_rng_exit(ctrldev); return ret; } devres_close_group(ctrldev, caam_rng_init); return 0; }