furi_hal_clock.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. #include <furi_hal_clock.h>
  2. #include <furi_hal_resources.h>
  3. #include <furi.h>
  4. #include <stm32wbxx_ll_pwr.h>
  5. #include <stm32wbxx_ll_rcc.h>
  6. #include <stm32wbxx_ll_utils.h>
  7. #include <stm32wbxx_ll_cortex.h>
  8. #include <stm32wbxx_ll_bus.h>
  9. #define TAG "FuriHalClock"
  10. #define CPU_CLOCK_HZ_EARLY 4000000
  11. #define CPU_CLOCK_HZ_MAIN 64000000
  12. #define TICK_INT_PRIORITY 15U
  13. #define HS_CLOCK_IS_READY() (LL_RCC_HSE_IsReady() && LL_RCC_HSI_IsReady())
  14. #define LS_CLOCK_IS_READY() (LL_RCC_LSE_IsReady() && LL_RCC_LSI1_IsReady())
  15. void furi_hal_clock_init_early() {
  16. LL_SetSystemCoreClock(CPU_CLOCK_HZ_EARLY);
  17. LL_Init1msTick(SystemCoreClock);
  18. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOA);
  19. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOB);
  20. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOC);
  21. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOD);
  22. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOE);
  23. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOH);
  24. LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_SPI1);
  25. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_SPI2);
  26. LL_APB1_GRP2_EnableClock(LL_APB1_GRP2_PERIPH_LPTIM2);
  27. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_I2C1);
  28. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_I2C3);
  29. }
  30. void furi_hal_clock_deinit_early() {
  31. LL_APB1_GRP1_DisableClock(LL_APB1_GRP1_PERIPH_I2C1);
  32. LL_APB1_GRP1_DisableClock(LL_APB1_GRP1_PERIPH_I2C3);
  33. LL_APB2_GRP1_DisableClock(LL_APB2_GRP1_PERIPH_SPI1);
  34. LL_APB1_GRP1_DisableClock(LL_APB1_GRP1_PERIPH_SPI2);
  35. LL_AHB2_GRP1_DisableClock(LL_AHB2_GRP1_PERIPH_GPIOA);
  36. LL_AHB2_GRP1_DisableClock(LL_AHB2_GRP1_PERIPH_GPIOB);
  37. LL_AHB2_GRP1_DisableClock(LL_AHB2_GRP1_PERIPH_GPIOC);
  38. LL_AHB2_GRP1_DisableClock(LL_AHB2_GRP1_PERIPH_GPIOD);
  39. LL_AHB2_GRP1_DisableClock(LL_AHB2_GRP1_PERIPH_GPIOE);
  40. LL_AHB2_GRP1_DisableClock(LL_AHB2_GRP1_PERIPH_GPIOH);
  41. }
  42. void furi_hal_clock_init() {
  43. /* Prepare Flash memory for 64MHz system clock */
  44. LL_FLASH_SetLatency(LL_FLASH_LATENCY_3);
  45. while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_3)
  46. ;
  47. /* HSE and HSI configuration and activation */
  48. LL_RCC_HSE_SetCapacitorTuning(0x26);
  49. LL_RCC_HSE_Enable();
  50. LL_RCC_HSI_Enable();
  51. while(!HS_CLOCK_IS_READY())
  52. ;
  53. /* Select HSI as system clock source after Wake Up from Stop mode
  54. * Must be set before enabling CSS */
  55. LL_RCC_SetClkAfterWakeFromStop(LL_RCC_STOP_WAKEUPCLOCK_HSI);
  56. LL_RCC_HSE_EnableCSS();
  57. /* LSE and LSI1 configuration and activation */
  58. LL_PWR_EnableBkUpAccess();
  59. LL_RCC_LSE_SetDriveCapability(LL_RCC_LSEDRIVE_HIGH);
  60. LL_RCC_LSE_Enable();
  61. LL_RCC_LSI1_Enable();
  62. while(!LS_CLOCK_IS_READY())
  63. ;
  64. LL_EXTI_EnableIT_0_31(
  65. LL_EXTI_LINE_18); /* Why? Because that's why. See RM0434, Table 61. CPU1 vector table. */
  66. LL_EXTI_EnableRisingTrig_0_31(LL_EXTI_LINE_18);
  67. LL_RCC_EnableIT_LSECSS();
  68. /* ES0394, extended case of 2.2.2 */
  69. if(!LL_RCC_IsActiveFlag_BORRST()) {
  70. LL_RCC_LSE_EnableCSS();
  71. }
  72. /* Main PLL configuration and activation */
  73. LL_RCC_PLL_ConfigDomain_SYS(LL_RCC_PLLSOURCE_HSE, LL_RCC_PLLM_DIV_2, 8, LL_RCC_PLLR_DIV_2);
  74. LL_RCC_PLL_Enable();
  75. LL_RCC_PLL_EnableDomain_SYS();
  76. while(LL_RCC_PLL_IsReady() != 1)
  77. ;
  78. LL_RCC_PLLSAI1_ConfigDomain_48M(
  79. LL_RCC_PLLSOURCE_HSE, LL_RCC_PLLM_DIV_2, 6, LL_RCC_PLLSAI1Q_DIV_2);
  80. LL_RCC_PLLSAI1_ConfigDomain_ADC(
  81. LL_RCC_PLLSOURCE_HSE, LL_RCC_PLLM_DIV_2, 6, LL_RCC_PLLSAI1R_DIV_2);
  82. LL_RCC_PLLSAI1_Enable();
  83. LL_RCC_PLLSAI1_EnableDomain_48M();
  84. LL_RCC_PLLSAI1_EnableDomain_ADC();
  85. while(LL_RCC_PLLSAI1_IsReady() != 1)
  86. ;
  87. /* Sysclk activation on the main PLL */
  88. /* Set CPU1 prescaler*/
  89. LL_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_1);
  90. /* Set CPU2 prescaler*/
  91. LL_C2_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_2);
  92. LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_PLL);
  93. while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_PLL)
  94. ;
  95. /* Set AHB SHARED prescaler*/
  96. LL_RCC_SetAHB4Prescaler(LL_RCC_SYSCLK_DIV_1);
  97. /* Set APB1 prescaler*/
  98. LL_RCC_SetAPB1Prescaler(LL_RCC_APB1_DIV_1);
  99. /* Set APB2 prescaler*/
  100. LL_RCC_SetAPB2Prescaler(LL_RCC_APB2_DIV_1);
  101. /* Disable MSI */
  102. LL_RCC_MSI_Disable();
  103. while(LL_RCC_MSI_IsReady() != 0)
  104. ;
  105. /* Update CMSIS variable (which can be updated also through SystemCoreClockUpdate function) */
  106. LL_SetSystemCoreClock(CPU_CLOCK_HZ_MAIN);
  107. /* Update the time base */
  108. LL_Init1msTick(SystemCoreClock);
  109. LL_SYSTICK_EnableIT();
  110. NVIC_SetPriority(
  111. SysTick_IRQn, NVIC_EncodePriority(NVIC_GetPriorityGrouping(), TICK_INT_PRIORITY, 0));
  112. NVIC_EnableIRQ(SysTick_IRQn);
  113. LL_RCC_SetUSARTClockSource(LL_RCC_USART1_CLKSOURCE_PCLK2);
  114. LL_RCC_SetLPUARTClockSource(LL_RCC_LPUART1_CLKSOURCE_PCLK1);
  115. LL_RCC_SetADCClockSource(LL_RCC_ADC_CLKSOURCE_PLLSAI1);
  116. LL_RCC_SetI2CClockSource(LL_RCC_I2C1_CLKSOURCE_PCLK1);
  117. LL_RCC_SetRNGClockSource(LL_RCC_RNG_CLKSOURCE_CLK48);
  118. LL_RCC_SetUSBClockSource(LL_RCC_USB_CLKSOURCE_PLLSAI1);
  119. LL_RCC_SetCLK48ClockSource(LL_RCC_CLK48_CLKSOURCE_PLLSAI1);
  120. LL_RCC_HSI_EnableInStopMode(); // Ensure that MR is capable of work in STOP0
  121. LL_RCC_SetSMPSClockSource(LL_RCC_SMPS_CLKSOURCE_HSE);
  122. LL_RCC_SetSMPSPrescaler(LL_RCC_SMPS_DIV_1);
  123. LL_RCC_SetRFWKPClockSource(LL_RCC_RFWKP_CLKSOURCE_LSE);
  124. // AHB1 GRP1
  125. LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_DMA1);
  126. LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_DMA2);
  127. LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_DMAMUX1);
  128. LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_CRC);
  129. // LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_TSC);
  130. // AHB2 GRP1
  131. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOA);
  132. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOB);
  133. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOC);
  134. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOD);
  135. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOE);
  136. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOH);
  137. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_ADC);
  138. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_AES1);
  139. // AHB3 GRP1
  140. // LL_AHB3_GRP1_EnableClock(LL_AHB3_GRP1_PERIPH_QUADSPI);
  141. LL_AHB3_GRP1_EnableClock(LL_AHB3_GRP1_PERIPH_PKA);
  142. LL_AHB3_GRP1_EnableClock(LL_AHB3_GRP1_PERIPH_AES2);
  143. LL_AHB3_GRP1_EnableClock(LL_AHB3_GRP1_PERIPH_RNG);
  144. LL_AHB3_GRP1_EnableClock(LL_AHB3_GRP1_PERIPH_HSEM);
  145. LL_AHB3_GRP1_EnableClock(LL_AHB3_GRP1_PERIPH_IPCC);
  146. LL_AHB3_GRP1_EnableClock(LL_AHB3_GRP1_PERIPH_FLASH);
  147. // APB1 GRP1
  148. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_TIM2);
  149. // LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_LCD);
  150. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_RTCAPB);
  151. // LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_WWDG);
  152. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_SPI2);
  153. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_I2C1);
  154. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_I2C3);
  155. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_CRS);
  156. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_USB);
  157. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_LPTIM1);
  158. // APB1 GRP2
  159. LL_APB1_GRP2_EnableClock(LL_APB1_GRP2_PERIPH_LPUART1);
  160. // APB2
  161. // LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_ADC);
  162. LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_TIM1);
  163. LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_SPI1);
  164. LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_USART1);
  165. LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_TIM16);
  166. LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_TIM17);
  167. // LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_SAI1);
  168. FURI_LOG_I(TAG, "Init OK");
  169. }
  170. void furi_hal_clock_switch_to_hsi() {
  171. LL_RCC_HSI_Enable();
  172. while(!LL_RCC_HSI_IsReady())
  173. ;
  174. LL_RCC_SetSMPSClockSource(LL_RCC_SMPS_CLKSOURCE_HSI);
  175. LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_HSI);
  176. while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_HSI)
  177. ;
  178. LL_C2_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_1);
  179. LL_FLASH_SetLatency(LL_FLASH_LATENCY_0);
  180. while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_0)
  181. ;
  182. }
  183. void furi_hal_clock_switch_to_pll() {
  184. LL_RCC_HSE_Enable();
  185. LL_RCC_PLL_Enable();
  186. LL_RCC_PLLSAI1_Enable();
  187. while(!LL_RCC_HSE_IsReady())
  188. ;
  189. while(!LL_RCC_PLL_IsReady())
  190. ;
  191. while(!LL_RCC_PLLSAI1_IsReady())
  192. ;
  193. LL_C2_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_2);
  194. LL_FLASH_SetLatency(LL_FLASH_LATENCY_3);
  195. while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_3)
  196. ;
  197. LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_PLL);
  198. LL_RCC_SetSMPSClockSource(LL_RCC_SMPS_CLKSOURCE_HSE);
  199. while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_PLL)
  200. ;
  201. }
  202. void furi_hal_clock_suspend_tick() {
  203. CLEAR_BIT(SysTick->CTRL, SysTick_CTRL_ENABLE_Msk);
  204. }
  205. void furi_hal_clock_resume_tick() {
  206. SET_BIT(SysTick->CTRL, SysTick_CTRL_ENABLE_Msk);
  207. }
  208. void furi_hal_clock_mco_enable(FuriHalClockMcoSourceId source, FuriHalClockMcoDivisorId div) {
  209. if(source == FuriHalClockMcoLse) {
  210. LL_RCC_ConfigMCO(LL_RCC_MCO1SOURCE_LSE, div);
  211. } else if(source == FuriHalClockMcoSysclk) {
  212. LL_RCC_ConfigMCO(LL_RCC_MCO1SOURCE_SYSCLK, div);
  213. } else {
  214. LL_RCC_MSI_Enable();
  215. while(LL_RCC_MSI_IsReady() != 1)
  216. ;
  217. switch(source) {
  218. case FuriHalClockMcoMsi100k:
  219. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_0);
  220. break;
  221. case FuriHalClockMcoMsi200k:
  222. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_1);
  223. break;
  224. case FuriHalClockMcoMsi400k:
  225. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_2);
  226. break;
  227. case FuriHalClockMcoMsi800k:
  228. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_3);
  229. break;
  230. case FuriHalClockMcoMsi1m:
  231. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_4);
  232. break;
  233. case FuriHalClockMcoMsi2m:
  234. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_5);
  235. break;
  236. case FuriHalClockMcoMsi4m:
  237. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_6);
  238. break;
  239. case FuriHalClockMcoMsi8m:
  240. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_7);
  241. break;
  242. case FuriHalClockMcoMsi16m:
  243. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_8);
  244. break;
  245. case FuriHalClockMcoMsi24m:
  246. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_9);
  247. break;
  248. case FuriHalClockMcoMsi32m:
  249. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_10);
  250. break;
  251. case FuriHalClockMcoMsi48m:
  252. LL_RCC_MSI_SetRange(LL_RCC_MSIRANGE_11);
  253. break;
  254. default:
  255. break;
  256. }
  257. LL_RCC_ConfigMCO(LL_RCC_MCO1SOURCE_MSI, div);
  258. }
  259. }
  260. void furi_hal_clock_mco_disable() {
  261. LL_RCC_ConfigMCO(LL_RCC_MCO1SOURCE_NOCLOCK, FuriHalClockMcoDiv1);
  262. LL_RCC_MSI_Disable();
  263. while(LL_RCC_MSI_IsReady() != 0)
  264. ;
  265. }