1 /*
2  * Copyright (c) 2019 STMicroelectronics
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #include <soc.h>
8 #include <stm32_ll_bus.h>
9 #include <stm32_ll_rcc.h>
10 #include <zephyr/drivers/clock_control.h>
11 #include <zephyr/sys/util.h>
12 #include <zephyr/drivers/clock_control/stm32_clock_control.h>
13 
14 /**
15  * @brief fill in AHB/APB buses configuration structure
16  */
stm32_clock_control_on(const struct device * dev,clock_control_subsys_t sub_system)17 static inline int stm32_clock_control_on(const struct device *dev,
18 					 clock_control_subsys_t sub_system)
19 {
20 	struct stm32_pclken *pclken = (struct stm32_pclken *)(sub_system);
21 
22 	ARG_UNUSED(dev);
23 
24 	switch (pclken->bus) {
25 	case STM32_CLOCK_BUS_APB1:
26 		LL_APB1_GRP1_EnableClock(pclken->enr);
27 		break;
28 	case STM32_CLOCK_BUS_APB2:
29 		LL_APB2_GRP1_EnableClock(pclken->enr);
30 		break;
31 	case STM32_CLOCK_BUS_APB3:
32 		LL_APB3_GRP1_EnableClock(pclken->enr);
33 		break;
34 	case STM32_CLOCK_BUS_APB4:
35 		LL_APB4_GRP1_EnableClock(pclken->enr);
36 		break;
37 	case STM32_CLOCK_BUS_APB5:
38 		LL_APB5_GRP1_EnableClock(pclken->enr);
39 		break;
40 	case STM32_CLOCK_BUS_AHB2:
41 		LL_AHB2_GRP1_EnableClock(pclken->enr);
42 		break;
43 	case STM32_CLOCK_BUS_AHB3:
44 		LL_AHB3_GRP1_EnableClock(pclken->enr);
45 		break;
46 	case STM32_CLOCK_BUS_AHB4:
47 		LL_AHB4_GRP1_EnableClock(pclken->enr);
48 		break;
49 	case STM32_CLOCK_BUS_AHB5:
50 		LL_AHB5_GRP1_EnableClock(pclken->enr);
51 		break;
52 	case STM32_CLOCK_BUS_AHB6:
53 		LL_AHB6_GRP1_EnableClock(pclken->enr);
54 		break;
55 	case STM32_CLOCK_BUS_AXI:
56 		LL_AXI_GRP1_EnableClock(pclken->enr);
57 		break;
58 	case STM32_CLOCK_BUS_MLAHB:
59 		LL_MLAHB_GRP1_EnableClock(pclken->enr);
60 		break;
61 	default:
62 		return -ENOTSUP;
63 	}
64 
65 	return 0;
66 }
67 
stm32_clock_control_off(const struct device * dev,clock_control_subsys_t sub_system)68 static inline int stm32_clock_control_off(const struct device *dev,
69 					  clock_control_subsys_t sub_system)
70 {
71 	struct stm32_pclken *pclken = (struct stm32_pclken *)(sub_system);
72 
73 	ARG_UNUSED(dev);
74 
75 	switch (pclken->bus) {
76 	case STM32_CLOCK_BUS_APB1:
77 		LL_APB1_GRP1_DisableClock(pclken->enr);
78 		break;
79 	case STM32_CLOCK_BUS_APB2:
80 		LL_APB2_GRP1_DisableClock(pclken->enr);
81 		break;
82 	case STM32_CLOCK_BUS_APB3:
83 		LL_APB3_GRP1_DisableClock(pclken->enr);
84 		break;
85 	case STM32_CLOCK_BUS_APB4:
86 		LL_APB4_GRP1_DisableClock(pclken->enr);
87 		break;
88 	case STM32_CLOCK_BUS_APB5:
89 		LL_APB5_GRP1_DisableClock(pclken->enr);
90 		break;
91 	case STM32_CLOCK_BUS_AHB2:
92 		LL_AHB2_GRP1_DisableClock(pclken->enr);
93 		break;
94 	case STM32_CLOCK_BUS_AHB3:
95 		LL_AHB3_GRP1_DisableClock(pclken->enr);
96 		break;
97 	case STM32_CLOCK_BUS_AHB4:
98 		LL_AHB4_GRP1_DisableClock(pclken->enr);
99 		break;
100 	case STM32_CLOCK_BUS_AHB5:
101 		LL_AHB5_GRP1_DisableClock(pclken->enr);
102 		break;
103 	case STM32_CLOCK_BUS_AHB6:
104 		LL_AHB6_GRP1_DisableClock(pclken->enr);
105 		break;
106 	case STM32_CLOCK_BUS_AXI:
107 		LL_AXI_GRP1_DisableClock(pclken->enr);
108 		break;
109 	case STM32_CLOCK_BUS_MLAHB:
110 		LL_MLAHB_GRP1_DisableClock(pclken->enr);
111 		break;
112 	default:
113 		return -ENOTSUP;
114 	}
115 
116 	return 0;
117 }
118 
stm32_clock_control_get_subsys_rate(const struct device * clock,clock_control_subsys_t sub_system,uint32_t * rate)119 static int stm32_clock_control_get_subsys_rate(const struct device *clock,
120 					       clock_control_subsys_t sub_system,
121 					       uint32_t *rate)
122 {
123 	struct stm32_pclken *pclken = (struct stm32_pclken *)(sub_system);
124 
125 	ARG_UNUSED(clock);
126 
127 	switch (pclken->bus) {
128 	case STM32_CLOCK_BUS_APB1:
129 		switch (pclken->enr) {
130 		case LL_APB1_GRP1_PERIPH_TIM2:
131 		case LL_APB1_GRP1_PERIPH_TIM3:
132 		case LL_APB1_GRP1_PERIPH_TIM4:
133 		case LL_APB1_GRP1_PERIPH_TIM5:
134 		case LL_APB1_GRP1_PERIPH_TIM6:
135 		case LL_APB1_GRP1_PERIPH_TIM7:
136 		case LL_APB1_GRP1_PERIPH_TIM12:
137 		case LL_APB1_GRP1_PERIPH_TIM13:
138 		case LL_APB1_GRP1_PERIPH_TIM14:
139 			*rate = LL_RCC_GetTIMGClockFreq(LL_RCC_TIMG1PRES);
140 			break;
141 		case LL_APB1_GRP1_PERIPH_LPTIM1:
142 			*rate = LL_RCC_GetLPTIMClockFreq(
143 					LL_RCC_LPTIM1_CLKSOURCE);
144 			break;
145 		case LL_APB1_GRP1_PERIPH_SPI2:
146 		case LL_APB1_GRP1_PERIPH_SPI3:
147 			*rate = LL_RCC_GetSPIClockFreq(LL_RCC_SPI23_CLKSOURCE);
148 			break;
149 		case LL_APB1_GRP1_PERIPH_USART2:
150 		case LL_APB1_GRP1_PERIPH_UART4:
151 			*rate = LL_RCC_GetUARTClockFreq(
152 					LL_RCC_UART24_CLKSOURCE);
153 			break;
154 		case LL_APB1_GRP1_PERIPH_USART3:
155 		case LL_APB1_GRP1_PERIPH_UART5:
156 			*rate = LL_RCC_GetUARTClockFreq(
157 					LL_RCC_UART35_CLKSOURCE);
158 			break;
159 		case LL_APB1_GRP1_PERIPH_UART7:
160 		case LL_APB1_GRP1_PERIPH_UART8:
161 			*rate = LL_RCC_GetUARTClockFreq(
162 					LL_RCC_UART78_CLKSOURCE);
163 			break;
164 		case LL_APB1_GRP1_PERIPH_I2C1:
165 		case LL_APB1_GRP1_PERIPH_I2C2:
166 			*rate = LL_RCC_GetI2CClockFreq(LL_RCC_I2C12_CLKSOURCE);
167 			break;
168 		case LL_APB1_GRP1_PERIPH_I2C3:
169 		case LL_APB1_GRP1_PERIPH_I2C5:
170 			*rate = LL_RCC_GetI2CClockFreq(LL_RCC_I2C35_CLKSOURCE);
171 			break;
172 		case LL_APB1_GRP1_PERIPH_SPDIF:
173 			*rate = LL_RCC_GetSPDIFRXClockFreq(
174 					LL_RCC_SPDIFRX_CLKSOURCE);
175 			break;
176 		case LL_APB1_GRP1_PERIPH_CEC:
177 			*rate = LL_RCC_GetCECClockFreq(LL_RCC_CEC_CLKSOURCE);
178 			break;
179 		case LL_APB1_GRP1_PERIPH_WWDG1:
180 		case LL_APB1_GRP1_PERIPH_DAC12:
181 		case LL_APB1_GRP1_PERIPH_MDIOS:
182 		default:
183 			return -ENOTSUP;
184 		}
185 		break;
186 	case STM32_CLOCK_BUS_APB2:
187 		switch (pclken->enr) {
188 		case LL_APB2_GRP1_PERIPH_TIM1:
189 		case LL_APB2_GRP1_PERIPH_TIM8:
190 		case LL_APB2_GRP1_PERIPH_TIM15:
191 		case LL_APB2_GRP1_PERIPH_TIM16:
192 		case LL_APB2_GRP1_PERIPH_TIM17:
193 			*rate = LL_RCC_GetTIMGClockFreq(LL_RCC_TIMG2PRES);
194 			break;
195 		case LL_APB2_GRP1_PERIPH_SPI1:
196 			*rate = LL_RCC_GetSPIClockFreq(LL_RCC_SPI1_CLKSOURCE);
197 			break;
198 		case LL_APB2_GRP1_PERIPH_SPI4:
199 		case LL_APB2_GRP1_PERIPH_SPI5:
200 			*rate = LL_RCC_GetSPIClockFreq(LL_RCC_SPI45_CLKSOURCE);
201 			break;
202 		case LL_APB2_GRP1_PERIPH_USART6:
203 			*rate = LL_RCC_GetUARTClockFreq(
204 					LL_RCC_USART6_CLKSOURCE);
205 			break;
206 		case LL_APB2_GRP1_PERIPH_SAI1:
207 			*rate = LL_RCC_GetSAIClockFreq(LL_RCC_SAI1_CLKSOURCE);
208 			break;
209 		case LL_APB2_GRP1_PERIPH_SAI2:
210 			*rate = LL_RCC_GetSAIClockFreq(LL_RCC_SAI2_CLKSOURCE);
211 			break;
212 		case LL_APB2_GRP1_PERIPH_SAI3:
213 			*rate = LL_RCC_GetSAIClockFreq(LL_RCC_SAI3_CLKSOURCE);
214 			break;
215 		case LL_APB2_GRP1_PERIPH_DFSDM1:
216 			*rate = LL_RCC_GetDFSDMClockFreq(
217 					LL_RCC_DFSDM_CLKSOURCE);
218 			break;
219 		case LL_APB2_GRP1_PERIPH_FDCAN:
220 			*rate = LL_RCC_GetFDCANClockFreq(
221 					LL_RCC_FDCAN_CLKSOURCE);
222 			break;
223 		case LL_APB2_GRP1_PERIPH_ADFSDM1:
224 		default:
225 			return -ENOTSUP;
226 		}
227 		break;
228 	case STM32_CLOCK_BUS_APB3:
229 		switch (pclken->enr) {
230 		case LL_APB3_GRP1_PERIPH_LPTIM2:
231 		case LL_APB3_GRP1_PERIPH_LPTIM3:
232 			*rate = LL_RCC_GetLPTIMClockFreq(
233 					LL_RCC_LPTIM23_CLKSOURCE);
234 			break;
235 		case LL_APB3_GRP1_PERIPH_LPTIM4:
236 		case LL_APB3_GRP1_PERIPH_LPTIM5:
237 			*rate = LL_RCC_GetLPTIMClockFreq(
238 					LL_RCC_LPTIM45_CLKSOURCE);
239 			break;
240 		case LL_APB3_GRP1_PERIPH_SAI4:
241 			*rate = LL_RCC_GetSAIClockFreq(LL_RCC_SAI4_CLKSOURCE);
242 			break;
243 		case LL_APB3_GRP1_PERIPH_SYSCFG:
244 		case LL_APB3_GRP1_PERIPH_VREF:
245 		case LL_APB3_GRP1_PERIPH_TMPSENS:
246 		case LL_APB3_GRP1_PERIPH_HDP:
247 		default:
248 			return -ENOTSUP;
249 		}
250 		break;
251 	case STM32_CLOCK_BUS_APB4:
252 		switch (pclken->enr) {
253 		case LL_APB4_GRP1_PERIPH_LTDC:
254 			*rate = LL_RCC_GetLTDCClockFreq();
255 			break;
256 		case LL_APB4_GRP1_PERIPH_DSI:
257 			*rate = LL_RCC_GetDSIClockFreq(LL_RCC_DSI_CLKSOURCE);
258 			break;
259 		case LL_APB4_GRP1_PERIPH_USBPHY:
260 			*rate = LL_RCC_GetUSBPHYClockFreq(
261 					LL_RCC_USBPHY_CLKSOURCE);
262 			break;
263 		case LL_APB4_GRP1_PERIPH_DDRPERFM:
264 		case LL_APB4_GRP1_PERIPH_STGENRO:
265 		case LL_APB4_GRP1_PERIPH_STGENROSTP:
266 		default:
267 			return -ENOTSUP;
268 		}
269 		break;
270 	case STM32_CLOCK_BUS_APB5:
271 		switch (pclken->enr) {
272 		case LL_APB5_GRP1_PERIPH_SPI6:
273 			*rate = LL_RCC_GetSPIClockFreq(LL_RCC_SPI6_CLKSOURCE);
274 			break;
275 		case LL_APB5_GRP1_PERIPH_I2C4:
276 		case LL_APB5_GRP1_PERIPH_I2C6:
277 			*rate = LL_RCC_GetI2CClockFreq(LL_RCC_I2C46_CLKSOURCE);
278 			break;
279 		case LL_APB5_GRP1_PERIPH_USART1:
280 			*rate = LL_RCC_GetUARTClockFreq(
281 					LL_RCC_USART1_CLKSOURCE);
282 			break;
283 		case LL_APB5_GRP1_PERIPH_STGEN:
284 		case LL_APB5_GRP1_PERIPH_STGENSTP:
285 			*rate = LL_RCC_GetSTGENClockFreq(
286 					LL_RCC_STGEN_CLKSOURCE);
287 			break;
288 		case LL_APB5_GRP1_PERIPH_RTCAPB:
289 			*rate = LL_RCC_GetRTCClockFreq();
290 			break;
291 		case LL_APB5_GRP1_PERIPH_TZC1:
292 		case LL_APB5_GRP1_PERIPH_TZC2:
293 		case LL_APB5_GRP1_PERIPH_TZPC:
294 		case LL_APB5_GRP1_PERIPH_BSEC:
295 		default:
296 			return -ENOTSUP;
297 		}
298 		break;
299 	case STM32_CLOCK_BUS_AHB2:
300 		switch (pclken->enr) {
301 		case LL_AHB2_GRP1_PERIPH_ADC12:
302 			*rate = LL_RCC_GetADCClockFreq(LL_RCC_ADC_CLKSOURCE);
303 			break;
304 		case LL_AHB2_GRP1_PERIPH_USBO:
305 			*rate = LL_RCC_GetUSBOClockFreq(LL_RCC_USBO_CLKSOURCE);
306 			break;
307 		case LL_AHB2_GRP1_PERIPH_SDMMC3:
308 			*rate = LL_RCC_GetSDMMCClockFreq(
309 					LL_RCC_SDMMC3_CLKSOURCE);
310 			break;
311 		case LL_AHB2_GRP1_PERIPH_DMA1:
312 		case LL_AHB2_GRP1_PERIPH_DMA2:
313 		case LL_AHB2_GRP1_PERIPH_DMAMUX:
314 		default:
315 			return -ENOTSUP;
316 		}
317 		break;
318 	case STM32_CLOCK_BUS_AHB3:
319 		switch (pclken->enr) {
320 		case LL_AHB3_GRP1_PERIPH_RNG2:
321 			*rate = LL_RCC_GetRNGClockFreq(LL_RCC_RNG2_CLKSOURCE);
322 			break;
323 		case LL_AHB3_GRP1_PERIPH_DCMI:
324 		case LL_AHB3_GRP1_PERIPH_CRYP2:
325 		case LL_AHB3_GRP1_PERIPH_HASH2:
326 		case LL_AHB3_GRP1_PERIPH_CRC2:
327 		case LL_AHB3_GRP1_PERIPH_HSEM:
328 		case LL_AHB3_GRP1_PERIPH_IPCC:
329 		default:
330 			return -ENOTSUP;
331 		}
332 		break;
333 	case STM32_CLOCK_BUS_AHB4:
334 		return -ENOTSUP;
335 	case STM32_CLOCK_BUS_AHB5:
336 		switch (pclken->enr) {
337 		case LL_AHB5_GRP1_PERIPH_RNG1:
338 			*rate = LL_RCC_GetRNGClockFreq(LL_RCC_RNG1_CLKSOURCE);
339 			break;
340 		case LL_AHB5_GRP1_PERIPH_GPIOZ:
341 		case LL_AHB5_GRP1_PERIPH_CRYP1:
342 		case LL_AHB5_GRP1_PERIPH_HASH1:
343 		case LL_AHB5_GRP1_PERIPH_BKPSRAM:
344 		default:
345 			return -ENOTSUP;
346 		}
347 		break;
348 	case STM32_CLOCK_BUS_AHB6:
349 		switch (pclken->enr) {
350 		case LL_AHB6_GRP1_PERIPH_ETH1CK:
351 		case LL_AHB6_GRP1_PERIPH_ETH1TX:
352 		case LL_AHB6_GRP1_PERIPH_ETH1RX:
353 		case LL_AHB6_GRP1_PERIPH_ETH1MAC:
354 		case LL_AHB6_GRP1_PERIPH_ETH1STP:
355 			*rate = LL_RCC_GetETHClockFreq(LL_RCC_ETH_CLKSOURCE);
356 			break;
357 		case LL_AHB6_GRP1_PERIPH_FMC:
358 			*rate = LL_RCC_GetFMCClockFreq(LL_RCC_FMC_CLKSOURCE);
359 			break;
360 		case LL_AHB6_GRP1_PERIPH_QSPI:
361 			*rate = LL_RCC_GetQSPIClockFreq(LL_RCC_QSPI_CLKSOURCE);
362 			break;
363 		case LL_AHB6_GRP1_PERIPH_SDMMC1:
364 		case LL_AHB6_GRP1_PERIPH_SDMMC2:
365 			*rate = LL_RCC_GetSDMMCClockFreq(
366 					LL_RCC_SDMMC12_CLKSOURCE);
367 			break;
368 		case LL_AHB6_GRP1_PERIPH_MDMA:
369 		case LL_AHB6_GRP1_PERIPH_GPU:
370 		case LL_AHB6_GRP1_PERIPH_CRC1:
371 		case LL_AHB6_GRP1_PERIPH_USBH:
372 		default:
373 			return -ENOTSUP;
374 		}
375 		break;
376 	case STM32_CLOCK_BUS_AXI:
377 		switch (pclken->enr) {
378 		case LL_AXI_GRP1_PERIPH_SYSRAMEN:
379 		default:
380 			return -ENOTSUP;
381 		}
382 		break;
383 	case STM32_CLOCK_BUS_MLAHB:
384 		switch (pclken->enr) {
385 		case LL_MLAHB_GRP1_PERIPH_RETRAMEN:
386 		default:
387 			return -ENOTSUP;
388 		}
389 		break;
390 	default:
391 		return -ENOTSUP;
392 	}
393 
394 	if (pclken->div) {
395 		*rate /= (pclken->div + 1);
396 	}
397 
398 	return 0;
399 }
400 
401 static DEVICE_API(clock_control, stm32_clock_control_api) = {
402 	.on = stm32_clock_control_on,
403 	.off = stm32_clock_control_off,
404 	.get_rate = stm32_clock_control_get_subsys_rate,
405 };
406 
stm32_clock_control_init(const struct device * dev)407 static int stm32_clock_control_init(const struct device *dev)
408 {
409 	ARG_UNUSED(dev);
410 	return 0;
411 }
412 
413 /**
414  * @brief RCC device, note that priority is intentionally set to 1 so
415  * that the device init runs just after SOC init
416  */
417 DEVICE_DT_DEFINE(DT_NODELABEL(rcc),
418 		    stm32_clock_control_init,
419 		    NULL,
420 		    NULL, NULL,
421 		    PRE_KERNEL_1,
422 		    CONFIG_CLOCK_CONTROL_INIT_PRIORITY,
423 		    &stm32_clock_control_api);
424