static inline void uart_init_lpuart(uart_t uart, uint32_t baudrate) { uint32_t clk; switch (uart_config[uart].clk_src) { case 0: clk = periph_apb_clk(uart_config[uart].bus); break; case RCC_CCIPR_LPUART1SEL_0: clk = CLOCK_CORECLOCK; break; case (RCC_CCIPR_LPUART1SEL_0 | RCC_CCIPR_LPUART1SEL_1): clk = 32768; break; default: /* HSI is not supported */ return; } RCC->CCIPR |= uart_config[uart].clk_src; /* LSE can only be used with baudrate <= 9600 */ if ( (clk < (3 * baudrate)) || (clk > (4096 * baudrate))) { return; } /* LPUARTDIV = f_clk * 256 / baudrate */ uint32_t brr = (uint32_t)(((uint64_t)clk << 8) / baudrate); dev(uart)->BRR = brr; }
uint32_t pwm_init(pwm_t pwm, pwm_mode_t mode, uint32_t freq, uint16_t res) { uint32_t bus_clk = periph_apb_clk(pwm_config[pwm].bus); /* verify parameters */ assert((pwm < PWM_NUMOF) && ((freq * res) < bus_clk)); /* power on the used timer */ pwm_poweron(pwm); /* reset configuration and CC channels */ dev(pwm)->CR1 = 0; dev(pwm)->CR2 = 0; for (int i = 0; i < TIMER_CHAN; i++) { dev(pwm)->CCR[i] = 0; } /* configure the used pins */ unsigned i = 0; while ((i < TIMER_CHAN) && (pwm_config[pwm].chan[i].pin != GPIO_UNDEF)) { gpio_init(pwm_config[pwm].chan[i].pin, GPIO_OUT); gpio_init_af(pwm_config[pwm].chan[i].pin, pwm_config[pwm].af); i++; } /* configure the PWM frequency and resolution by setting the auto-reload * and prescaler registers */ dev(pwm)->PSC = (bus_clk / (res * freq)) - 1; dev(pwm)->ARR = res - 1; /* set PWM mode */ switch (mode) { case PWM_LEFT: dev(pwm)->CCMR1 = CCMR_LEFT; dev(pwm)->CCMR2 = CCMR_LEFT; break; case PWM_RIGHT: dev(pwm)->CCMR1 = CCMR_RIGHT; dev(pwm)->CCMR2 = CCMR_RIGHT; break; case PWM_CENTER: dev(pwm)->CCMR1 = 0; dev(pwm)->CCMR2 = 0; dev(pwm)->CR1 |= (TIM_CR1_CMS_0 | TIM_CR1_CMS_1); break; } /* enable PWM outputs and start PWM generation */ #ifdef TIM_BDTR_MOE dev(pwm)->BDTR = TIM_BDTR_MOE; #endif dev(pwm)->CCER = (TIM_CCER_CC1E | TIM_CCER_CC2E | TIM_CCER_CC3E | TIM_CCER_CC4E); dev(pwm)->CR1 |= TIM_CR1_CEN; /* return the actual used PWM frequency */ return (bus_clk / (res * (dev(pwm)->PSC + 1))); }
static inline void uart_init_usart(uart_t uart, uint32_t baudrate) { uint16_t mantissa; uint8_t fraction; uint32_t clk; /* calculate and apply baudrate */ clk = periph_apb_clk(uart_config[uart].bus) / baudrate; mantissa = (uint16_t)(clk / 16); fraction = (uint8_t)(clk - (mantissa * 16)); dev(uart)->BRR = ((mantissa & 0x0fff) << 4) | (fraction & 0x0f); }
int uart_init(uart_t uart, uint32_t baudrate, uart_rx_cb_t rx_cb, void *arg) { uint16_t mantissa; uint8_t fraction; uint32_t clk; assert(uart < UART_NUMOF); /* save ISR context */ isr_ctx[uart].rx_cb = rx_cb; isr_ctx[uart].arg = arg; /* configure TX pin */ gpio_init(uart_config[uart].tx_pin, GPIO_OUT); /* set TX pin high to avoid garbage during further initialization */ gpio_set(uart_config[uart].tx_pin); #ifdef CPU_FAM_STM32F1 gpio_init_af(uart_config[uart].tx_pin, GPIO_AF_OUT_PP); #else gpio_init_af(uart_config[uart].tx_pin, uart_config[uart].tx_af); #endif /* configure RX pin */ if (rx_cb) { gpio_init(uart_config[uart].rx_pin, GPIO_IN); #ifndef CPU_FAM_STM32F1 gpio_init_af(uart_config[uart].rx_pin, uart_config[uart].rx_af); #endif } #ifdef UART_USE_HW_FC if (uart_config[uart].cts_pin != GPIO_UNDEF) { gpio_init(uart_config[uart].cts_pin, GPIO_IN); gpio_init(uart_config[uart].rts_pin, GPIO_OUT); #ifdef CPU_FAM_STM32F1 gpio_init_af(uart_config[uart].rts_pin, GPIO_AF_OUT_PP); #else gpio_init_af(uart_config[uart].cts_pin, uart_config[uart].cts_af); gpio_init_af(uart_config[uart].rts_pin, uart_config[uart].rts_af); #endif } #endif /* enable the clock */ periph_clk_en(uart_config[uart].bus, uart_config[uart].rcc_mask); /* reset UART configuration -> defaults to 8N1 mode */ dev(uart)->CR1 = 0; dev(uart)->CR2 = 0; dev(uart)->CR3 = 0; /* calculate and apply baudrate */ clk = periph_apb_clk(uart_config[uart].bus) / baudrate; mantissa = (uint16_t)(clk / 16); fraction = (uint8_t)(clk - (mantissa * 16)); dev(uart)->BRR = ((mantissa & 0x0fff) << 4) | (fraction & 0x0f); /* enable RX interrupt if applicable */ if (rx_cb) { NVIC_EnableIRQ(uart_config[uart].irqn); dev(uart)->CR1 = (USART_CR1_UE | USART_CR1_TE | RXENABLE); } else { dev(uart)->CR1 = (USART_CR1_UE | USART_CR1_TE); } #ifdef UART_USE_HW_FC if (uart_config[uart].cts_pin != GPIO_UNDEF) { /* configure hardware flow control */ dev(uart)->CR3 = (USART_CR3_RTSE | USART_CR3_CTSE); } #endif return UART_OK; }