1 /*
2 * Copyright (c) 2019 Brett Witherspoon
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT ti_cc13xx_cc26xx_trng
8
9 #include <zephyr/kernel.h>
10 #include <zephyr/device.h>
11 #include <zephyr/drivers/entropy.h>
12 #include <zephyr/irq.h>
13 #include <zephyr/pm/policy.h>
14 #include <zephyr/pm/device.h>
15
16 #include <zephyr/sys/ring_buffer.h>
17 #include <zephyr/sys/sys_io.h>
18
19 #include <driverlib/prcm.h>
20 #include <driverlib/trng.h>
21
22 #include <ti/drivers/Power.h>
23 #include <ti/drivers/power/PowerCC26X2.h>
24
25 #define CPU_FREQ DT_PROP(DT_PATH(cpus, cpu_0), clock_frequency)
26
27 #define US_PER_SAMPLE (1000000ULL * \
28 CONFIG_ENTROPY_CC13XX_CC26XX_SAMPLES_PER_CYCLE / CPU_FREQ + 1ULL)
29
30 struct entropy_cc13xx_cc26xx_data {
31 struct k_sem lock;
32 struct k_sem sync;
33 struct ring_buf pool;
34 uint8_t data[CONFIG_ENTROPY_CC13XX_CC26XX_POOL_SIZE];
35 #ifdef CONFIG_PM
36 Power_NotifyObj post_notify;
37 bool constrained;
38 #endif
39 };
40
start_trng(struct entropy_cc13xx_cc26xx_data * data)41 static void start_trng(struct entropy_cc13xx_cc26xx_data *data)
42 {
43 /* Initialization as described in TRM section 18.6.1.2 */
44 TRNGReset();
45 while (sys_read32(TRNG_BASE + TRNG_O_SWRESET)) {
46 continue;
47 }
48
49 /* Set samples per cycle */
50 TRNGConfigure(0, CONFIG_ENTROPY_CC13XX_CC26XX_SAMPLES_PER_CYCLE,
51 0);
52 /* De-tune FROs */
53 sys_write32(TRNG_FRODETUNE_FRO_MASK_M, TRNG_BASE +
54 TRNG_O_FRODETUNE);
55 /* Enable FROs */
56 sys_write32(TRNG_FROEN_FRO_MASK_M, TRNG_BASE + TRNG_O_FROEN);
57 /* Set shutdown and alarm thresholds */
58 sys_write32((CONFIG_ENTROPY_CC13XX_CC26XX_SHUTDOWN_THRESHOLD
59 << 16) | CONFIG_ENTROPY_CC13XX_CC26XX_ALARM_THRESHOLD,
60 TRNG_BASE + TRNG_O_ALARMCNT);
61
62 TRNGEnable();
63 TRNGIntEnable(TRNG_NUMBER_READY | TRNG_FRO_SHUTDOWN);
64 }
65
66 #ifdef CONFIG_PM_DEVICE
stop_trng(struct entropy_cc13xx_cc26xx_data * data)67 static void stop_trng(struct entropy_cc13xx_cc26xx_data *data)
68 {
69 TRNGDisable();
70
71 TRNGIntClear(TRNG_NUMBER_READY | TRNG_FRO_SHUTDOWN);
72 TRNGIntDisable(TRNG_NUMBER_READY | TRNG_FRO_SHUTDOWN);
73 }
74 #endif
75
handle_shutdown_ovf(void)76 static void handle_shutdown_ovf(void)
77 {
78 uint32_t off;
79
80 /* Clear shutdown */
81 TRNGIntClear(TRNG_FRO_SHUTDOWN);
82 /* Disabled FROs */
83 off = sys_read32(TRNG_BASE + TRNG_O_ALARMSTOP);
84 /* Clear alarms */
85 sys_write32(0, TRNG_BASE + TRNG_O_ALARMMASK);
86 sys_write32(0, TRNG_BASE + TRNG_O_ALARMSTOP);
87 /* De-tune FROs */
88 sys_write32(off, TRNG_BASE + TRNG_O_FRODETUNE);
89 /* Re-enable FROs */
90 sys_write32(off, TRNG_BASE + TRNG_O_FROEN);
91 }
92
entropy_cc13xx_cc26xx_get_entropy(const struct device * dev,uint8_t * buf,uint16_t len)93 static int entropy_cc13xx_cc26xx_get_entropy(const struct device *dev,
94 uint8_t *buf,
95 uint16_t len)
96 {
97 struct entropy_cc13xx_cc26xx_data *data = dev->data;
98 uint32_t cnt;
99
100 #ifdef CONFIG_PM
101 unsigned int key = irq_lock();
102
103 if (!data->constrained) {
104 pm_policy_state_lock_get(PM_STATE_STANDBY, PM_ALL_SUBSTATES);
105 data->constrained = true;
106 }
107 irq_unlock(key);
108 #endif
109
110 TRNGIntEnable(TRNG_NUMBER_READY);
111
112 while (len) {
113 k_sem_take(&data->lock, K_FOREVER);
114 cnt = ring_buf_get(&data->pool, buf, len);
115 k_sem_give(&data->lock);
116
117 if (cnt) {
118 buf += cnt;
119 len -= cnt;
120 } else {
121 k_sem_take(&data->sync, K_FOREVER);
122 }
123 }
124
125 return 0;
126 }
127
entropy_cc13xx_cc26xx_isr(const struct device * dev)128 static void entropy_cc13xx_cc26xx_isr(const struct device *dev)
129 {
130 struct entropy_cc13xx_cc26xx_data *data = dev->data;
131 uint32_t src = 0;
132 uint32_t cnt;
133 uint32_t num[2];
134
135 /* Interrupt service routine as described in TRM section 18.6.1.3.2 */
136 src = TRNGStatusGet();
137
138 if (src & TRNG_NUMBER_READY) {
139 /* This function acknowledges the ready status */
140 num[1] = TRNGNumberGet(TRNG_HI_WORD);
141 num[0] = TRNGNumberGet(TRNG_LOW_WORD);
142
143 cnt = ring_buf_put(&data->pool, (uint8_t *)num, sizeof(num));
144
145 /* When pool is full disable interrupt and stop reading numbers */
146 if (cnt != sizeof(num)) {
147 #ifdef CONFIG_PM
148 if (data->constrained) {
149 pm_policy_state_lock_put(
150 PM_STATE_STANDBY,
151 PM_ALL_SUBSTATES);
152 data->constrained = false;
153 }
154 #endif
155 TRNGIntDisable(TRNG_NUMBER_READY);
156 }
157
158 k_sem_give(&data->sync);
159 }
160
161 /* Change the shutdown FROs' oscillating frequency in an attempt to
162 * prevent further locking on to the sampling clock frequency.
163 */
164 if (src & TRNG_FRO_SHUTDOWN) {
165 handle_shutdown_ovf();
166 }
167 }
168
entropy_cc13xx_cc26xx_get_entropy_isr(const struct device * dev,uint8_t * buf,uint16_t len,uint32_t flags)169 static int entropy_cc13xx_cc26xx_get_entropy_isr(const struct device *dev,
170 uint8_t *buf, uint16_t len,
171 uint32_t flags)
172 {
173 struct entropy_cc13xx_cc26xx_data *data = dev->data;
174 uint16_t cnt;
175 uint16_t read = len;
176 uint32_t src;
177 uint32_t num[2];
178 unsigned int key;
179
180 key = irq_lock();
181 cnt = ring_buf_get(&data->pool, buf, len);
182 irq_unlock(key);
183
184 if ((cnt == len) || ((flags & ENTROPY_BUSYWAIT) == 0U)) {
185 read = cnt;
186 } else {
187 buf += cnt;
188 len -= cnt;
189
190 /* Allowed to busy-wait. We should use a polling approach */
191 while (len) {
192 key = irq_lock();
193
194 src = TRNGStatusGet();
195 if (src & TRNG_NUMBER_READY) {
196 /*
197 * This function acknowledges the ready
198 * status
199 */
200 num[1] = TRNGNumberGet(TRNG_HI_WORD);
201 num[0] = TRNGNumberGet(TRNG_LOW_WORD);
202
203 ring_buf_put(&data->pool, (uint8_t *)num,
204 sizeof(num));
205 }
206
207 /*
208 * If interrupts were enabled during busy wait, this
209 * would allow us to pick up anything that has been put
210 * in by the ISR as well.
211 */
212 cnt = ring_buf_get(&data->pool, buf, len);
213
214 if (src & TRNG_FRO_SHUTDOWN) {
215 handle_shutdown_ovf();
216 }
217
218 irq_unlock(key);
219
220 if (cnt) {
221 buf += cnt;
222 len -= cnt;
223 } else {
224 k_busy_wait(US_PER_SAMPLE);
225 }
226 }
227
228 }
229
230 return read;
231 }
232
233 #ifdef CONFIG_PM
234 /*
235 * ======== post_notify_fxn ========
236 * Called by Power module when waking up the CPU from Standby. The TRNG needs
237 * to be reconfigured afterwards, unless Zephyr's device PM turned it off, in
238 * which case it'd be responsible for turning it back on and reconfiguring it.
239 */
post_notify_fxn(unsigned int eventType,uintptr_t eventArg,uintptr_t clientArg)240 static int post_notify_fxn(unsigned int eventType, uintptr_t eventArg,
241 uintptr_t clientArg)
242 {
243 const struct device *dev = (const struct device *)clientArg;
244 int ret = Power_NOTIFYDONE;
245 int16_t res_id;
246
247 /* Reconfigure the hardware if returning from sleep */
248 if (eventType == PowerCC26XX_AWAKE_STANDBY) {
249 res_id = PowerCC26XX_PERIPH_TRNG;
250
251 if (Power_getDependencyCount(res_id) != 0) {
252 /* Reconfigure and enable TRNG only if powered */
253 start_trng(dev->data);
254 }
255 }
256
257 return (ret);
258 }
259 #endif
260
261 #ifdef CONFIG_PM_DEVICE
entropy_cc13xx_cc26xx_pm_action(const struct device * dev,enum pm_device_action action)262 static int entropy_cc13xx_cc26xx_pm_action(const struct device *dev,
263 enum pm_device_action action)
264 {
265 struct entropy_cc13xx_cc26xx_data *data = dev->data;
266
267 switch (action) {
268 case PM_DEVICE_ACTION_RESUME:
269 Power_setDependency(PowerCC26XX_PERIPH_TRNG);
270 start_trng(data);
271 break;
272 case PM_DEVICE_ACTION_SUSPEND:
273 stop_trng(data);
274 Power_releaseDependency(PowerCC26XX_PERIPH_TRNG);
275 break;
276 default:
277 return -ENOTSUP;
278 }
279
280 return 0;
281 }
282 #endif /* CONFIG_PM_DEVICE */
283
entropy_cc13xx_cc26xx_init(const struct device * dev)284 static int entropy_cc13xx_cc26xx_init(const struct device *dev)
285 {
286 struct entropy_cc13xx_cc26xx_data *data = dev->data;
287
288 /* Initialize driver data */
289 ring_buf_init(&data->pool, sizeof(data->data), data->data);
290
291 #if defined(CONFIG_PM)
292 Power_setDependency(PowerCC26XX_PERIPH_TRNG);
293 /* Stay out of standby until buffer is filled with entropy */
294 pm_policy_state_lock_get(PM_STATE_STANDBY, PM_ALL_SUBSTATES);
295 data->constrained = true;
296 /* Register notification function */
297 Power_registerNotify(&data->post_notify,
298 PowerCC26XX_AWAKE_STANDBY,
299 post_notify_fxn, (uintptr_t)dev);
300 #else
301 /* Power TRNG domain */
302 PRCMPowerDomainOn(PRCM_DOMAIN_PERIPH);
303
304 /* Enable TRNG peripheral clocks */
305 PRCMPeripheralRunEnable(PRCM_PERIPH_TRNG);
306 /* Enabled the TRNG while in sleep mode to keep the entropy pool full. After
307 * the pool is full the TRNG will enter idle mode when random numbers are no
308 * longer being read. */
309 PRCMPeripheralSleepEnable(PRCM_PERIPH_TRNG);
310 PRCMPeripheralDeepSleepEnable(PRCM_PERIPH_TRNG);
311
312
313 /* Load PRCM settings */
314 PRCMLoadSet();
315 while (!PRCMLoadGet()) {
316 continue;
317 }
318
319 /* Peripherals should not be accessed until power domain is on. */
320 while (PRCMPowerDomainsAllOn(PRCM_DOMAIN_PERIPH) !=
321 PRCM_DOMAIN_POWER_ON) {
322 continue;
323 }
324 #endif
325
326 start_trng(data);
327
328 IRQ_CONNECT(DT_INST_IRQN(0),
329 DT_INST_IRQ(0, priority),
330 entropy_cc13xx_cc26xx_isr,
331 DEVICE_DT_INST_GET(0), 0);
332 irq_enable(DT_INST_IRQN(0));
333
334 return 0;
335 }
336
337 static DEVICE_API(entropy, entropy_cc13xx_cc26xx_driver_api) = {
338 .get_entropy = entropy_cc13xx_cc26xx_get_entropy,
339 .get_entropy_isr = entropy_cc13xx_cc26xx_get_entropy_isr,
340 };
341
342 static struct entropy_cc13xx_cc26xx_data entropy_cc13xx_cc26xx_data = {
343 .lock = Z_SEM_INITIALIZER(entropy_cc13xx_cc26xx_data.lock, 1, 1),
344 .sync = Z_SEM_INITIALIZER(entropy_cc13xx_cc26xx_data.sync, 0, 1),
345 };
346
347 PM_DEVICE_DT_INST_DEFINE(0, entropy_cc13xx_cc26xx_pm_action);
348
349 DEVICE_DT_INST_DEFINE(0,
350 entropy_cc13xx_cc26xx_init,
351 PM_DEVICE_DT_INST_GET(0),
352 &entropy_cc13xx_cc26xx_data, NULL,
353 PRE_KERNEL_1, CONFIG_ENTROPY_INIT_PRIORITY,
354 &entropy_cc13xx_cc26xx_driver_api);
355