Linux Audio

Check our new training course

Loading...
v4.6
 
  1/*
  2 * Nomadik RNG support
  3 *  Copyright 2009 Alessandro Rubini
  4 *
  5 *  This program is free software; you can redistribute it and/or modify
  6 *  it under the terms of the GNU General Public License as published by
  7 *  the Free Software Foundation; either version 2 of the License, or
  8 *  (at your option) any later version.
  9 */
 10
 11#include <linux/kernel.h>
 12#include <linux/module.h>
 13#include <linux/device.h>
 14#include <linux/amba/bus.h>
 15#include <linux/hw_random.h>
 16#include <linux/io.h>
 17#include <linux/clk.h>
 18#include <linux/err.h>
 19
 20static struct clk *rng_clk;
 21
 22static int nmk_rng_read(struct hwrng *rng, void *data, size_t max, bool wait)
 23{
 24	void __iomem *base = (void __iomem *)rng->priv;
 25
 26	/*
 27	 * The register is 32 bits and gives 16 random bits (low half).
 28	 * A subsequent read will delay the core for 400ns, so we just read
 29	 * once and accept the very unlikely very small delay, even if wait==0.
 30	 */
 31	*(u16 *)data = __raw_readl(base + 8) & 0xffff;
 32	return 2;
 33}
 34
 35/* we have at most one RNG per machine, granted */
 36static struct hwrng nmk_rng = {
 37	.name		= "nomadik",
 38	.read		= nmk_rng_read,
 39};
 40
 41static int nmk_rng_probe(struct amba_device *dev, const struct amba_id *id)
 42{
 43	void __iomem *base;
 44	int ret;
 45
 46	rng_clk = devm_clk_get(&dev->dev, NULL);
 47	if (IS_ERR(rng_clk)) {
 48		dev_err(&dev->dev, "could not get rng clock\n");
 49		ret = PTR_ERR(rng_clk);
 50		return ret;
 51	}
 52
 53	clk_prepare_enable(rng_clk);
 54
 55	ret = amba_request_regions(dev, dev->dev.init_name);
 56	if (ret)
 57		goto out_clk;
 58	ret = -ENOMEM;
 59	base = devm_ioremap(&dev->dev, dev->res.start,
 60			    resource_size(&dev->res));
 61	if (!base)
 62		goto out_release;
 63	nmk_rng.priv = (unsigned long)base;
 64	ret = hwrng_register(&nmk_rng);
 65	if (ret)
 66		goto out_release;
 67	return 0;
 68
 69out_release:
 70	amba_release_regions(dev);
 71out_clk:
 72	clk_disable(rng_clk);
 73	return ret;
 74}
 75
 76static int nmk_rng_remove(struct amba_device *dev)
 77{
 78	hwrng_unregister(&nmk_rng);
 79	amba_release_regions(dev);
 80	clk_disable(rng_clk);
 81	return 0;
 82}
 83
 84static struct amba_id nmk_rng_ids[] = {
 85	{
 86		.id	= 0x000805e1,
 87		.mask	= 0x000fffff, /* top bits are rev and cfg: accept all */
 88	},
 89	{0, 0},
 90};
 91
 92MODULE_DEVICE_TABLE(amba, nmk_rng_ids);
 93
 94static struct amba_driver nmk_rng_driver = {
 95	.drv = {
 96		.owner = THIS_MODULE,
 97		.name = "rng",
 98		},
 99	.probe = nmk_rng_probe,
100	.remove = nmk_rng_remove,
101	.id_table = nmk_rng_ids,
102};
103
104module_amba_driver(nmk_rng_driver);
105
106MODULE_LICENSE("GPL");
v6.2
  1// SPDX-License-Identifier: GPL-2.0-or-later
  2/*
  3 * Nomadik RNG support
  4 *  Copyright 2009 Alessandro Rubini
 
 
 
 
 
  5 */
  6
  7#include <linux/kernel.h>
  8#include <linux/module.h>
  9#include <linux/device.h>
 10#include <linux/amba/bus.h>
 11#include <linux/hw_random.h>
 12#include <linux/io.h>
 13#include <linux/clk.h>
 14#include <linux/err.h>
 15
 16static struct clk *rng_clk;
 17
 18static int nmk_rng_read(struct hwrng *rng, void *data, size_t max, bool wait)
 19{
 20	void __iomem *base = (void __iomem *)rng->priv;
 21
 22	/*
 23	 * The register is 32 bits and gives 16 random bits (low half).
 24	 * A subsequent read will delay the core for 400ns, so we just read
 25	 * once and accept the very unlikely very small delay, even if wait==0.
 26	 */
 27	*(u16 *)data = __raw_readl(base + 8) & 0xffff;
 28	return 2;
 29}
 30
 31/* we have at most one RNG per machine, granted */
 32static struct hwrng nmk_rng = {
 33	.name		= "nomadik",
 34	.read		= nmk_rng_read,
 35};
 36
 37static int nmk_rng_probe(struct amba_device *dev, const struct amba_id *id)
 38{
 39	void __iomem *base;
 40	int ret;
 41
 42	rng_clk = devm_clk_get(&dev->dev, NULL);
 43	if (IS_ERR(rng_clk)) {
 44		dev_err(&dev->dev, "could not get rng clock\n");
 45		ret = PTR_ERR(rng_clk);
 46		return ret;
 47	}
 48
 49	clk_prepare_enable(rng_clk);
 50
 51	ret = amba_request_regions(dev, dev->dev.init_name);
 52	if (ret)
 53		goto out_clk;
 54	ret = -ENOMEM;
 55	base = devm_ioremap(&dev->dev, dev->res.start,
 56			    resource_size(&dev->res));
 57	if (!base)
 58		goto out_release;
 59	nmk_rng.priv = (unsigned long)base;
 60	ret = devm_hwrng_register(&dev->dev, &nmk_rng);
 61	if (ret)
 62		goto out_release;
 63	return 0;
 64
 65out_release:
 66	amba_release_regions(dev);
 67out_clk:
 68	clk_disable_unprepare(rng_clk);
 69	return ret;
 70}
 71
 72static void nmk_rng_remove(struct amba_device *dev)
 73{
 
 74	amba_release_regions(dev);
 75	clk_disable_unprepare(rng_clk);
 
 76}
 77
 78static const struct amba_id nmk_rng_ids[] = {
 79	{
 80		.id	= 0x000805e1,
 81		.mask	= 0x000fffff, /* top bits are rev and cfg: accept all */
 82	},
 83	{0, 0},
 84};
 85
 86MODULE_DEVICE_TABLE(amba, nmk_rng_ids);
 87
 88static struct amba_driver nmk_rng_driver = {
 89	.drv = {
 90		.owner = THIS_MODULE,
 91		.name = "rng",
 92		},
 93	.probe = nmk_rng_probe,
 94	.remove = nmk_rng_remove,
 95	.id_table = nmk_rng_ids,
 96};
 97
 98module_amba_driver(nmk_rng_driver);
 99
100MODULE_LICENSE("GPL");