furi_hal_clock.c 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. #include <furi_hal_clock.h>
  2. #include <furi.h>
  3. #include <stm32wbxx_ll_pwr.h>
  4. #include <stm32wbxx_ll_rcc.h>
  5. #include <stm32wbxx_ll_utils.h>
  6. #define TAG "FuriHalClock"
  7. #define HS_CLOCK_IS_READY() (LL_RCC_HSE_IsReady() && LL_RCC_HSI_IsReady())
  8. #define LS_CLOCK_IS_READY() (LL_RCC_LSE_IsReady() && LL_RCC_LSI1_IsReady())
  9. void furi_hal_clock_init() {
  10. /* Prepare Flash memory for 64mHz system clock */
  11. LL_FLASH_SetLatency(LL_FLASH_LATENCY_3);
  12. while(LL_FLASH_GetLatency() != LL_FLASH_LATENCY_3)
  13. ;
  14. /* HSE and HSI configuration and activation */
  15. LL_RCC_HSE_SetCapacitorTuning(0x26);
  16. LL_RCC_HSE_Enable();
  17. LL_RCC_HSI_Enable();
  18. while(!HS_CLOCK_IS_READY())
  19. ;
  20. LL_RCC_HSE_EnableCSS();
  21. /* LSE and LSI1 configuration and activation */
  22. LL_PWR_EnableBkUpAccess();
  23. LL_RCC_LSE_SetDriveCapability(LL_RCC_LSEDRIVE_HIGH);
  24. LL_RCC_LSE_Enable();
  25. LL_RCC_LSI1_Enable();
  26. while(!LS_CLOCK_IS_READY())
  27. ;
  28. LL_EXTI_EnableIT_0_31(
  29. LL_EXTI_LINE_18); /* Why? Because that's why. See RM0434, Table 61. CPU1 vector table. */
  30. LL_EXTI_EnableRisingTrig_0_31(LL_EXTI_LINE_18);
  31. LL_RCC_EnableIT_LSECSS();
  32. LL_RCC_LSE_EnableCSS();
  33. /* Main PLL configuration and activation */
  34. LL_RCC_PLL_ConfigDomain_SYS(LL_RCC_PLLSOURCE_HSE, LL_RCC_PLLM_DIV_2, 8, LL_RCC_PLLR_DIV_2);
  35. LL_RCC_PLL_Enable();
  36. LL_RCC_PLL_EnableDomain_SYS();
  37. while(LL_RCC_PLL_IsReady() != 1)
  38. ;
  39. LL_RCC_PLLSAI1_ConfigDomain_48M(
  40. LL_RCC_PLLSOURCE_HSE, LL_RCC_PLLM_DIV_2, 6, LL_RCC_PLLSAI1Q_DIV_2);
  41. LL_RCC_PLLSAI1_ConfigDomain_ADC(
  42. LL_RCC_PLLSOURCE_HSE, LL_RCC_PLLM_DIV_2, 6, LL_RCC_PLLSAI1R_DIV_2);
  43. LL_RCC_PLLSAI1_Enable();
  44. LL_RCC_PLLSAI1_EnableDomain_48M();
  45. LL_RCC_PLLSAI1_EnableDomain_ADC();
  46. while(LL_RCC_PLLSAI1_IsReady() != 1)
  47. ;
  48. /* Sysclk activation on the main PLL */
  49. /* Set CPU1 prescaler*/
  50. LL_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_1);
  51. /* Set CPU2 prescaler*/
  52. LL_C2_RCC_SetAHBPrescaler(LL_RCC_SYSCLK_DIV_2);
  53. LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_PLL);
  54. while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_PLL)
  55. ;
  56. /* Set AHB SHARED prescaler*/
  57. LL_RCC_SetAHB4Prescaler(LL_RCC_SYSCLK_DIV_1);
  58. /* Set APB1 prescaler*/
  59. LL_RCC_SetAPB1Prescaler(LL_RCC_APB1_DIV_1);
  60. /* Set APB2 prescaler*/
  61. LL_RCC_SetAPB2Prescaler(LL_RCC_APB2_DIV_1);
  62. /* Disable MSI */
  63. LL_RCC_MSI_Disable();
  64. while(LL_RCC_MSI_IsReady() != 0)
  65. ;
  66. /* Update CMSIS variable (which can be updated also through SystemCoreClockUpdate function) */
  67. LL_SetSystemCoreClock(64000000);
  68. /* Update the time base */
  69. if(HAL_InitTick(TICK_INT_PRIORITY) != HAL_OK) {
  70. Error_Handler();
  71. }
  72. LL_RCC_SetUSARTClockSource(LL_RCC_USART1_CLKSOURCE_PCLK2);
  73. LL_RCC_SetLPUARTClockSource(LL_RCC_LPUART1_CLKSOURCE_PCLK1);
  74. LL_RCC_SetADCClockSource(LL_RCC_ADC_CLKSOURCE_PLLSAI1);
  75. LL_RCC_SetI2CClockSource(LL_RCC_I2C1_CLKSOURCE_PCLK1);
  76. LL_RCC_SetRNGClockSource(LL_RCC_RNG_CLKSOURCE_CLK48);
  77. LL_RCC_SetUSBClockSource(LL_RCC_USB_CLKSOURCE_PLLSAI1);
  78. LL_RCC_SetCLK48ClockSource(LL_RCC_CLK48_CLKSOURCE_PLLSAI1);
  79. LL_RCC_SetSMPSClockSource(LL_RCC_SMPS_CLKSOURCE_HSE);
  80. LL_RCC_SetSMPSPrescaler(LL_RCC_SMPS_DIV_1);
  81. LL_RCC_SetRFWKPClockSource(LL_RCC_RFWKP_CLKSOURCE_LSE);
  82. // AHB1
  83. LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_DMAMUX1);
  84. LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_DMA1);
  85. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_I2C1);
  86. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_SPI2);
  87. // AHB2
  88. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOA);
  89. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOB);
  90. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOC);
  91. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOD);
  92. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOE);
  93. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOH);
  94. LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_SPI1);
  95. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_AES1);
  96. // AHB3
  97. LL_AHB3_GRP1_EnableClock(LL_AHB3_GRP1_PERIPH_PKA);
  98. LL_AHB3_GRP1_EnableClock(LL_AHB3_GRP1_PERIPH_RNG);
  99. LL_AHB3_GRP1_EnableClock(LL_AHB3_GRP1_PERIPH_AES2);
  100. // APB1
  101. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_TIM2);
  102. LL_APB1_GRP2_EnableClock(LL_APB1_GRP2_PERIPH_LPUART1);
  103. // APB2
  104. LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_USART1);
  105. FURI_LOG_I(TAG, "Init OK");
  106. }
  107. void furi_hal_clock_switch_to_hsi() {
  108. LL_RCC_HSI_Enable();
  109. while(!LL_RCC_HSI_IsReady())
  110. ;
  111. LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_HSI);
  112. LL_RCC_SetSMPSClockSource(LL_RCC_SMPS_CLKSOURCE_HSI);
  113. while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_HSI)
  114. ;
  115. }
  116. void furi_hal_clock_switch_to_pll() {
  117. LL_RCC_HSE_Enable();
  118. LL_RCC_PLL_Enable();
  119. while(!LL_RCC_HSE_IsReady())
  120. ;
  121. while(!LL_RCC_PLL_IsReady())
  122. ;
  123. LL_RCC_SetSysClkSource(LL_RCC_SYS_CLKSOURCE_PLL);
  124. LL_RCC_SetSMPSClockSource(LL_RCC_SMPS_CLKSOURCE_HSE);
  125. while(LL_RCC_GetSysClkSource() != LL_RCC_SYS_CLKSOURCE_STATUS_PLL)
  126. ;
  127. }