stm32/uart: Generalise UART source clock calculation for H5 and H7 MCUs.

This gets the calculation working properly for H5 MCUs, and fixes the
switch statement to switch on csel&7 instead of csel&3.

Signed-off-by: Damien George <damien@micropython.org>
This commit is contained in:
Damien George 2023-09-28 15:33:56 +10:00
parent 82b4ab7576
commit 88564c1406
3 changed files with 38 additions and 27 deletions

View File

@ -404,6 +404,7 @@ HAL_SRC_C += $(addprefix $(STM32LIB_HAL_BASE)/Src/stm32$(MCU_SERIES)xx_,\
hal_tim.c \ hal_tim.c \
hal_tim_ex.c \ hal_tim_ex.c \
hal_uart.c \ hal_uart.c \
ll_rcc.c \
ll_utils.c \ ll_utils.c \
) )

View File

@ -95,6 +95,7 @@
#include "stm32h7xx_ll_adc.h" #include "stm32h7xx_ll_adc.h"
#include "stm32h7xx_ll_lpuart.h" #include "stm32h7xx_ll_lpuart.h"
#include "stm32h7xx_ll_pwr.h" #include "stm32h7xx_ll_pwr.h"
#include "stm32h7xx_ll_rcc.h"
#include "stm32h7xx_ll_rtc.h" #include "stm32h7xx_ll_rtc.h"
#include "stm32h7xx_ll_usart.h" #include "stm32h7xx_ll_usart.h"

View File

@ -837,48 +837,57 @@ uint32_t uart_get_source_freq(pyb_uart_obj_t *self) {
uart_clk = LSE_VALUE; uart_clk = LSE_VALUE;
break; break;
} }
#elif defined(STM32H7A3xx) || defined(STM32H7A3xxQ) || defined(STM32H7B3xx) || defined(STM32H7B3xxQ)
#elif defined(STM32H5) || defined(STM32H7)
uint32_t csel; uint32_t csel;
unsigned int bus_pclk;
#if defined(STM32H5)
if (1 <= self->uart_id && self->uart_id <= 10) {
csel = RCC->CCIPR1 >> ((self->uart_id - 1) * 3);
} else {
csel = RCC->CCIPR2 >> ((self->uart_id - 11) * 3);
}
bus_pclk = self->uart_id == 1 ? 2 : 1;
#elif defined(STM32H7A3xx) || defined(STM32H7A3xxQ) || defined(STM32H7B3xx) || defined(STM32H7B3xxQ)
if (self->uart_id == 1 || self->uart_id == 6 || self->uart_id == 9 || self->uart_id == 10) { if (self->uart_id == 1 || self->uart_id == 6 || self->uart_id == 9 || self->uart_id == 10) {
csel = RCC->CDCCIP2R >> 3; csel = RCC->CDCCIP2R >> 3;
bus_pclk = 2;
} else { } else {
csel = RCC->CDCCIP2R; csel = RCC->CDCCIP2R;
bus_pclk = 1;
} }
switch (csel & 3) { #else
case 0:
if (self->uart_id == 1 || self->uart_id == 6 || self->uart_id == 9 || self->uart_id == 10) {
uart_clk = HAL_RCC_GetPCLK2Freq();
} else {
uart_clk = HAL_RCC_GetPCLK1Freq();
}
break;
case 3:
uart_clk = HSI_VALUE;
break;
case 4:
uart_clk = CSI_VALUE;
break;
case 5:
uart_clk = LSE_VALUE;
break;
default:
break;
}
#elif defined(STM32H7)
uint32_t csel;
if (self->uart_id == 1 || self->uart_id == 6) { if (self->uart_id == 1 || self->uart_id == 6) {
csel = RCC->D2CCIP2R >> 3; csel = RCC->D2CCIP2R >> 3;
bus_pclk = 2;
} else { } else {
csel = RCC->D2CCIP2R; csel = RCC->D2CCIP2R;
bus_pclk = 1;
} }
switch (csel & 3) { #endif
switch (csel & 7) {
case 0: case 0:
if (self->uart_id == 1 || self->uart_id == 6) { if (bus_pclk == 1) {
uart_clk = HAL_RCC_GetPCLK2Freq();
} else {
uart_clk = HAL_RCC_GetPCLK1Freq(); uart_clk = HAL_RCC_GetPCLK1Freq();
} else {
uart_clk = HAL_RCC_GetPCLK2Freq();
} }
break; break;
case 1: {
LL_PLL_ClocksTypeDef PLL_Clocks;
LL_RCC_GetPLL2ClockFreq(&PLL_Clocks);
uart_clk = PLL_Clocks.PLL_Q_Frequency;
break;
}
case 2: {
LL_PLL_ClocksTypeDef PLL_Clocks;
LL_RCC_GetPLL3ClockFreq(&PLL_Clocks);
uart_clk = PLL_Clocks.PLL_Q_Frequency;
break;
}
case 3: case 3:
uart_clk = HSI_VALUE; uart_clk = HSI_VALUE;
break; break;