Skip to content

Commit 77fc276

Browse files
committed
stmhal: For UART, check that baudrate is within 5% of desired value.
Also includes documentation about minimum baudrate. Addresses issue micropython#1090.
1 parent e06cf89 commit 77fc276

File tree

2 files changed

+29
-0
lines changed

2 files changed

+29
-0
lines changed

docs/library/pyb.UART.rst

+7
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,13 @@ Methods
7575
- ``timeout_char`` is the timeout in milliseconds to wait between characters.
7676
- ``read_buf_len`` is the character length of the read buffer (0 to disable).
7777

78+
This method will raise an exception if the baudrate could not be set within
79+
5% of the desired value. The minimum baudrate is dictated by the frequency
80+
of the bus that the UART is on; UART(1) and UART(6) are APB2, the rest are on
81+
APB1. The default bus frequencies give a minimum baudrate of 1300 for
82+
UART(1) and UART(6) and 650 for the others. Use :func:`pyb.freq <pyb.freq>`
83+
to reduce the bus frequencies to get lower baudrates.
84+
7885
*Note:* with parity=None, only 8 and 9 bits are supported. With parity enabled,
7986
only 7 and 8 bits are supported.
8087

stmhal/uart.c

+22
Original file line numberDiff line numberDiff line change
@@ -457,6 +457,28 @@ STATIC mp_obj_t pyb_uart_init_helper(pyb_uart_obj_t *self, mp_uint_t n_args, con
457457
HAL_NVIC_EnableIRQ(self->irqn);
458458
}
459459

460+
// compute actual baudrate that was configured
461+
// (this formula assumes UART_OVERSAMPLING_16)
462+
uint32_t actual_baudrate;
463+
if (self->uart.Instance == USART1 || self->uart.Instance == USART6) {
464+
actual_baudrate = HAL_RCC_GetPCLK2Freq();
465+
} else {
466+
actual_baudrate = HAL_RCC_GetPCLK1Freq();
467+
}
468+
actual_baudrate /= self->uart.Instance->BRR;
469+
470+
// check we could set the baudrate within 5%
471+
uint32_t baudrate_diff;
472+
if (actual_baudrate > init->BaudRate) {
473+
baudrate_diff = actual_baudrate - init->BaudRate;
474+
} else {
475+
baudrate_diff = init->BaudRate - actual_baudrate;
476+
}
477+
init->BaudRate = actual_baudrate; // remember actual baudrate for printing
478+
if (20 * baudrate_diff > init->BaudRate) {
479+
nlr_raise(mp_obj_new_exception_msg_varg(&mp_type_ValueError, "set baudrate %d is not within 5%% of desired value", actual_baudrate));
480+
}
481+
460482
return mp_const_none;
461483
}
462484

0 commit comments

Comments
 (0)