2 * Copyright (c) 2010-2012 Broadcom. All rights reserved.
3 * Copyright (c) 2013 Lubomir Rintel
5 * This program is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU General Public License ("GPL")
7 * version 2, as published by the Free Software Foundation.
10 #include <linux/hw_random.h>
12 #include <linux/kernel.h>
13 #include <linux/module.h>
14 #include <linux/of_address.h>
15 #include <linux/of_platform.h>
16 #include <linux/platform_device.h>
17 #include <linux/printk.h>
18 #include <linux/clk.h>
21 #define RNG_STATUS 0x4
23 #define RNG_INT_MASK 0x10
28 /* the initial numbers generated are "less random" so will be discarded */
29 #define RNG_WARMUP_COUNT 0x40000
31 #define RNG_INT_OFF 0x1
33 struct bcm2835_rng_priv
{
40 static inline struct bcm2835_rng_priv
*to_rng_priv(struct hwrng
*rng
)
42 return container_of(rng
, struct bcm2835_rng_priv
, rng
);
45 static inline u32
rng_readl(struct bcm2835_rng_priv
*priv
, u32 offset
)
47 /* MIPS chips strapped for BE will automagically configure the
48 * peripheral registers for CPU-native byte order.
50 if (IS_ENABLED(CONFIG_MIPS
) && IS_ENABLED(CONFIG_CPU_BIG_ENDIAN
))
51 return __raw_readl(priv
->base
+ offset
);
53 return readl(priv
->base
+ offset
);
56 static inline void rng_writel(struct bcm2835_rng_priv
*priv
, u32 val
,
59 if (IS_ENABLED(CONFIG_MIPS
) && IS_ENABLED(CONFIG_CPU_BIG_ENDIAN
))
60 __raw_writel(val
, priv
->base
+ offset
);
62 writel(val
, priv
->base
+ offset
);
65 static int bcm2835_rng_read(struct hwrng
*rng
, void *buf
, size_t max
,
68 struct bcm2835_rng_priv
*priv
= to_rng_priv(rng
);
69 u32 max_words
= max
/ sizeof(u32
);
72 while ((rng_readl(priv
, RNG_STATUS
) >> 24) == 0) {
78 num_words
= rng_readl(priv
, RNG_STATUS
) >> 24;
79 if (num_words
> max_words
)
80 num_words
= max_words
;
82 for (count
= 0; count
< num_words
; count
++)
83 ((u32
*)buf
)[count
] = rng_readl(priv
, RNG_DATA
);
85 return num_words
* sizeof(u32
);
88 static int bcm2835_rng_init(struct hwrng
*rng
)
90 struct bcm2835_rng_priv
*priv
= to_rng_priv(rng
);
94 if (!IS_ERR(priv
->clk
)) {
95 ret
= clk_prepare_enable(priv
->clk
);
100 if (priv
->mask_interrupts
) {
101 /* mask the interrupt */
102 val
= rng_readl(priv
, RNG_INT_MASK
);
104 rng_writel(priv
, val
, RNG_INT_MASK
);
107 /* set warm-up count & enable */
108 rng_writel(priv
, RNG_WARMUP_COUNT
, RNG_STATUS
);
109 rng_writel(priv
, RNG_RBGEN
, RNG_CTRL
);
114 static void bcm2835_rng_cleanup(struct hwrng
*rng
)
116 struct bcm2835_rng_priv
*priv
= to_rng_priv(rng
);
118 /* disable rng hardware */
119 rng_writel(priv
, 0, RNG_CTRL
);
121 if (!IS_ERR(priv
->clk
))
122 clk_disable_unprepare(priv
->clk
);
125 struct bcm2835_rng_of_data
{
126 bool mask_interrupts
;
129 static const struct bcm2835_rng_of_data nsp_rng_of_data
= {
130 .mask_interrupts
= true,
133 static const struct of_device_id bcm2835_rng_of_match
[] = {
134 { .compatible
= "brcm,bcm2835-rng"},
135 { .compatible
= "brcm,bcm-nsp-rng", .data
= &nsp_rng_of_data
},
136 { .compatible
= "brcm,bcm5301x-rng", .data
= &nsp_rng_of_data
},
137 { .compatible
= "brcm,bcm6368-rng"},
141 static int bcm2835_rng_probe(struct platform_device
*pdev
)
143 const struct bcm2835_rng_of_data
*of_data
;
144 struct device
*dev
= &pdev
->dev
;
145 struct device_node
*np
= dev
->of_node
;
146 const struct of_device_id
*rng_id
;
147 struct bcm2835_rng_priv
*priv
;
151 priv
= devm_kzalloc(dev
, sizeof(*priv
), GFP_KERNEL
);
155 platform_set_drvdata(pdev
, priv
);
157 r
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
160 priv
->base
= devm_ioremap_resource(dev
, r
);
161 if (IS_ERR(priv
->base
))
162 return PTR_ERR(priv
->base
);
164 /* Clock is optional on most platforms */
165 priv
->clk
= devm_clk_get(dev
, NULL
);
167 priv
->rng
.name
= pdev
->name
;
168 priv
->rng
.init
= bcm2835_rng_init
;
169 priv
->rng
.read
= bcm2835_rng_read
;
170 priv
->rng
.cleanup
= bcm2835_rng_cleanup
;
172 rng_id
= of_match_node(bcm2835_rng_of_match
, np
);
176 /* Check for rng init function, execute it */
177 of_data
= rng_id
->data
;
179 priv
->mask_interrupts
= of_data
->mask_interrupts
;
181 /* register driver */
182 err
= devm_hwrng_register(dev
, &priv
->rng
);
184 dev_err(dev
, "hwrng registration failed\n");
186 dev_info(dev
, "hwrng registered\n");
191 MODULE_DEVICE_TABLE(of
, bcm2835_rng_of_match
);
193 static struct platform_device_id bcm2835_rng_devtype
[] = {
194 { .name
= "bcm2835-rng" },
195 { .name
= "bcm63xx-rng" },
198 MODULE_DEVICE_TABLE(platform
, bcm2835_rng_devtype
);
200 static struct platform_driver bcm2835_rng_driver
= {
202 .name
= "bcm2835-rng",
203 .of_match_table
= bcm2835_rng_of_match
,
205 .probe
= bcm2835_rng_probe
,
206 .id_table
= bcm2835_rng_devtype
,
208 module_platform_driver(bcm2835_rng_driver
);
210 MODULE_AUTHOR("Lubomir Rintel <lkundrak@v3.sk>");
211 MODULE_DESCRIPTION("BCM2835 Random Number Generator (RNG) driver");
212 MODULE_LICENSE("GPL v2");