// SPDX-License-Identifier: GPL-2.0 /* * Copyright (c) 2024 Broadcom */ #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt #include #include #include #include #include #include #include #include #define HOST_REV_ID 0x00 #define HOST_FIFO_DEPTH 0x04 #define HOST_FIFO_COUNT 0x08 #define HOST_FIFO_THRESHOLD 0x0c #define HOST_FIFO_DATA 0x10 #define HOST_FIFO_COUNT_MASK 0xffff /* Delay range in microseconds */ #define FIFO_DELAY_MIN_US 3 #define FIFO_DELAY_MAX_US 7 #define FIFO_DELAY_MAX_COUNT 10 struct bcm74110_priv { void __iomem *base; }; static inline int bcm74110_rng_fifo_count(void __iomem *mem) { return readl_relaxed(mem) & HOST_FIFO_COUNT_MASK; } static int bcm74110_rng_read(struct hwrng *rng, void *buf, size_t max, bool wait) { struct bcm74110_priv *priv = (struct bcm74110_priv *)rng->priv; void __iomem *fc_addr = priv->base + HOST_FIFO_COUNT; void __iomem *fd_addr = priv->base + HOST_FIFO_DATA; unsigned underrun_count = 0; u32 max_words = max / sizeof(u32); u32 num_words; unsigned i; /* * We need to check how many words are available in the RNG FIFO. If * there aren't any, we need to wait for some to become available. */ while ((num_words = bcm74110_rng_fifo_count(fc_addr)) == 0) { if (!wait) return 0; /* * As a precaution, limit how long we wait. If the FIFO doesn't * refill within the allotted time, return 0 (=no data) to the * caller. */ if (likely(underrun_count < FIFO_DELAY_MAX_COUNT)) usleep_range(FIFO_DELAY_MIN_US, FIFO_DELAY_MAX_US); else return 0; underrun_count++; } if (num_words > max_words) num_words = max_words; /* Bail early if we run out of random numbers unexpectedly */ for (i = 0; i < num_words && bcm74110_rng_fifo_count(fc_addr) > 0; i++) ((u32 *)buf)[i] = readl_relaxed(fd_addr); return i * sizeof(u32); } static struct hwrng bcm74110_hwrng = { .read = bcm74110_rng_read, }; static int bcm74110_rng_probe(struct platform_device *pdev) { struct device *dev = &pdev->dev; struct bcm74110_priv *priv; int rc; priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); if (!priv) return -ENOMEM; bcm74110_hwrng.name = pdev->name; bcm74110_hwrng.priv = (unsigned long)priv; priv->base = devm_platform_ioremap_resource(pdev, 0); if (IS_ERR(priv->base)) return PTR_ERR(priv->base); rc = devm_hwrng_register(dev, &bcm74110_hwrng); if (rc) dev_err(dev, "hwrng registration failed (%d)\n", rc); else dev_info(dev, "hwrng registered\n"); return rc; } static const struct of_device_id bcm74110_rng_match[] = { { .compatible = "brcm,bcm74110-rng", }, {}, }; MODULE_DEVICE_TABLE(of, bcm74110_rng_match); static struct platform_driver bcm74110_rng_driver = { .driver = { .name = KBUILD_MODNAME, .of_match_table = bcm74110_rng_match, }, .probe = bcm74110_rng_probe, }; module_platform_driver(bcm74110_rng_driver); MODULE_AUTHOR("Markus Mayer "); MODULE_DESCRIPTION("BCM 74110 Random Number Generator (RNG) driver"); MODULE_LICENSE("GPL v2");