1 /*
2  * Copyright (c) 2020 Lemonbeat GmbH
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #define DT_DRV_COMPAT silabs_gecko_trng
8 
9  #include <zephyr/drivers/entropy.h>
10  #include <string.h>
11  #include "soc.h"
12  #include "em_cmu.h"
13 
14 #if defined(CONFIG_CRYPTO_ACC_GECKO_TRNG)
15 
16 /*
17  * Select the correct Crypto ACC FIFO memory base address.
18  *
19  * Problem: Gecko SDK doesn't provide macros that check if SL_TRUSTZONE is used or not for Crypto
20  * ACC RNGOUT FIFO memory base address, like it does for register address definitions.
21  *
22  * Solution: Check which register base address is used for the Crypto ACC peripheral and select an
23  * appropriate FIFO memory base address.
24  */
25 #if (CRYPTOACC_BASE == CRYPTOACC_S_BASE)
26 #define S2_FIFO_BASE CRYPTOACC_RNGOUT_FIFO_S_MEM_BASE
27 #else
28 #define S2_FIFO_BASE CRYPTOACC_RNGOUT_FIFO_MEM_BASE
29 #endif
30 
31 /**
32  * Series 2 SoCs have different TRNG register definitions
33  */
34 #if defined(_SILICON_LABS_32B_SERIES_2_CONFIG_2)	/* xG22 */
35 #define S2_FIFO_LEVEL	(CRYPTOACC_RNGCTRL->FIFOLEVEL)
36 #define S2_CTRL		(CRYPTOACC_RNGCTRL->RNGCTRL)
37 #define S2_CTRL_ENABLE	(CRYPTOACC_RNGCTRL_ENABLE)
38 #elif defined(_SILICON_LABS_32B_SERIES_2_CONFIG_7)	/* xG27 */
39 #define S2_FIFO_LEVEL	(CRYPTOACC->NDRNG_FIFOLEVEL)
40 #define S2_CTRL		(CRYPTOACC->NDRNG_CONTROL)
41 #define S2_CTRL_ENABLE	(CRYPTOACC_NDRNG_CONTROL_ENABLE)
42 #else	/* _SILICON_LABS_32B_SERIES_2_CONFIG_* */
43 #error "Building for unsupported Series 2 SoC"
44 #endif	/* _SILICON_LABS_32B_SERIES_2_CONFIG_* */
45 #endif	/* CONFIG_CRYPTO_ACC_GECKO_TRNG */
46 
entropy_gecko_trng_read(uint8_t * output,size_t len)47 static void entropy_gecko_trng_read(uint8_t *output, size_t len)
48 {
49 #ifndef CONFIG_CRYPTO_ACC_GECKO_TRNG
50 	uint32_t tmp;
51 	uint32_t *data = (uint32_t *) output;
52 
53 	/* Read known good available data. */
54 	while (len >= 4) {
55 		*data++ = TRNG0->FIFO;
56 		len -= 4;
57 	}
58 	if (len > 0) {
59 		/* Handle the case where len is not a multiple of 4
60 		 * and FIFO data is available.
61 		 */
62 		tmp = TRNG0->FIFO;
63 		memcpy(data, (const uint8_t *) &tmp, len);
64 	}
65 #else
66 	memcpy(output, ((const uint8_t *) S2_FIFO_BASE), len);
67 #endif
68 }
69 
entropy_gecko_trng_get_entropy(const struct device * dev,uint8_t * buffer,uint16_t length)70 static int entropy_gecko_trng_get_entropy(const struct device *dev,
71 					  uint8_t *buffer,
72 					  uint16_t length)
73 {
74 	size_t count = 0;
75 	size_t available;
76 
77 	ARG_UNUSED(dev);
78 
79 	while (length) {
80 #ifndef CONFIG_CRYPTO_ACC_GECKO_TRNG
81 		available = TRNG0->FIFOLEVEL * 4;
82 #else
83 		available = S2_FIFO_LEVEL * 4;
84 #endif
85 		if (available == 0) {
86 			return -EINVAL;
87 		}
88 
89 		count = SL_MIN(length, available);
90 		entropy_gecko_trng_read(buffer, count);
91 		buffer += count;
92 		length -= count;
93 	}
94 
95 	return 0;
96 }
97 
entropy_gecko_trng_get_entropy_isr(const struct device * dev,uint8_t * buf,uint16_t len,uint32_t flags)98 static int entropy_gecko_trng_get_entropy_isr(const struct device *dev,
99 					      uint8_t *buf,
100 					      uint16_t len, uint32_t flags)
101 {
102 
103 	if ((flags & ENTROPY_BUSYWAIT) == 0U) {
104 
105 		/* No busy wait; return whatever data is available. */
106 		size_t count;
107 #ifndef CONFIG_CRYPTO_ACC_GECKO_TRNG
108 		size_t available = TRNG0->FIFOLEVEL * 4;
109 #else
110 		size_t available = S2_FIFO_LEVEL * 4;
111 #endif
112 
113 		if (available == 0) {
114 			return -ENODATA;
115 		}
116 		count = SL_MIN(len, available);
117 		entropy_gecko_trng_read(buf, count);
118 		return count;
119 
120 	} else {
121 		/* Allowed to busy-wait */
122 		int ret = entropy_gecko_trng_get_entropy(dev, buf, len);
123 
124 		if (ret == 0) {
125 			/* Data retrieved successfully. */
126 			return len;
127 		}
128 		return ret;
129 	}
130 }
131 
entropy_gecko_trng_init(const struct device * dev)132 static int entropy_gecko_trng_init(const struct device *dev)
133 {
134 	/* Enable the TRNG0 clock. */
135 #ifndef CONFIG_CRYPTO_ACC_GECKO_TRNG
136 	CMU_ClockEnable(cmuClock_TRNG0, true);
137 
138 	/* Enable TRNG0. */
139 	TRNG0->CONTROL = TRNG_CONTROL_ENABLE;
140 #else
141 	/* Enable the CRYPTO ACC clock. */
142 	CMU_ClockEnable(cmuClock_CRYPTOACC, true);
143 
144 	/* Enable TRNG */
145 	S2_CTRL |= S2_CTRL_ENABLE;
146 #endif
147 
148 	return 0;
149 }
150 
151 static DEVICE_API(entropy, entropy_gecko_trng_api_funcs) = {
152 	.get_entropy = entropy_gecko_trng_get_entropy,
153 	.get_entropy_isr = entropy_gecko_trng_get_entropy_isr
154 };
155 
156 DEVICE_DT_INST_DEFINE(0,
157 			entropy_gecko_trng_init, NULL,
158 			NULL, NULL,
159 			PRE_KERNEL_1, CONFIG_ENTROPY_INIT_PRIORITY,
160 			&entropy_gecko_trng_api_funcs);
161