Skip to content

Commit f443e60

Browse files
committed
upd
1 parent 1f19da3 commit f443e60

File tree

1 file changed

+105
-4
lines changed

1 file changed

+105
-4
lines changed

esp-hal/src/uart.rs

+105-4
Original file line numberDiff line numberDiff line change
@@ -265,6 +265,18 @@ pub enum StopBits {
265265
_2 = 3,
266266
}
267267

268+
/// Defines how strictly the requested baud rate must be met.
269+
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)]
270+
pub enum BaudrateTolerance {
271+
/// Accept the closest achievable baud rate without restriction.
272+
#[default]
273+
Closest,
274+
/// Require an exact match, otherwise return an error.
275+
Exact,
276+
/// Allow a certain percentage of deviation.
277+
ErrorPercent(u8),
278+
}
279+
268280
/// UART Configuration
269281
#[derive(Debug, Clone, Copy, procmacros::BuilderLite)]
270282
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
@@ -273,6 +285,9 @@ pub struct Config {
273285
/// The baud rate (speed) of the UART communication in bits per second
274286
/// (bps).
275287
baudrate: u32,
288+
/// Determines how close to the desired baud rate value the driver should
289+
/// set the baud rate.
290+
baudrate_tolerance: BaudrateTolerance,
276291
/// Number of data bits in each frame (5, 6, 7, or 8 bits).
277292
data_bits: DataBits,
278293
/// Parity setting (None, Even, or Odd).
@@ -305,6 +320,7 @@ impl Default for Config {
305320
rx: RxConfig::default(),
306321
tx: TxConfig::default(),
307322
baudrate: 115_200,
323+
baudrate_tolerance: BaudrateTolerance::default(),
308324
data_bits: Default::default(),
309325
parity: Default::default(),
310326
stop_bits: Default::default(),
@@ -463,6 +479,8 @@ pub struct UartRx<'d, Dm> {
463479
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
464480
#[non_exhaustive]
465481
pub enum ConfigError {
482+
/// The requested baud rate is not achievable.
483+
UnachievableBaudrate,
466484
/// The requested baud rate is not supported.
467485
UnsupportedBaudrate,
468486
/// The requested timeout is not supported.
@@ -476,6 +494,9 @@ impl core::error::Error for ConfigError {}
476494
impl core::fmt::Display for ConfigError {
477495
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
478496
match self {
497+
ConfigError::UnachievableBaudrate => {
498+
write!(f, "The requested baud rate is not achievable")
499+
}
479500
ConfigError::UnsupportedBaudrate => {
480501
write!(f, "The requested baud rate is not supported")
481502
}
@@ -2205,7 +2226,7 @@ impl Info {
22052226

22062227
fn apply_config(&self, config: &Config) -> Result<(), ConfigError> {
22072228
config.validate()?;
2208-
self.change_baud(config);
2229+
self.change_baud(config)?;
22092230
self.change_data_bits(config.data_bits);
22102231
self.change_parity(config.parity);
22112232
self.change_stop_bits(config.stop_bits);
@@ -2394,7 +2415,7 @@ impl Info {
23942415
}
23952416

23962417
#[cfg(any(esp32c2, esp32c3, esp32s3))]
2397-
fn change_baud(&self, config: &Config) {
2418+
fn change_baud(&self, config: &Config) -> Result<(), ConfigError> {
23982419
use crate::peripherals::LPWR;
23992420

24002421
let clocks = Clocks::get();
@@ -2433,6 +2454,24 @@ impl Info {
24332454
self.regs()
24342455
.clkdiv()
24352456
.write(|w| unsafe { w.clkdiv().bits(divider_integer).frag().bits(divider_frag) });
2457+
2458+
let actual_baud = self.get_baudrate(clk);
2459+
2460+
match config.baudrate_tolerance {
2461+
BaudrateTolerance::Exact if actual_baud != config.baudrate => {
2462+
return Err(ConfigError::UnachievableBaudrate)
2463+
}
2464+
BaudrateTolerance::ErrorPercent(percent) => {
2465+
let deviation = ((config.baudrate as i64 - actual_baud as i64).abs() as u64 * 100)
2466+
/ actual_baud as u64;
2467+
if deviation > percent as u64 {
2468+
return Err(ConfigError::UnachievableBaudrate);
2469+
}
2470+
}
2471+
_ => {}
2472+
}
2473+
2474+
Ok(())
24362475
}
24372476

24382477
fn is_instance(&self, other: impl Instance) -> bool {
@@ -2444,7 +2483,7 @@ impl Info {
24442483
}
24452484

24462485
#[cfg(any(esp32c6, esp32h2))]
2447-
fn change_baud(&self, config: &Config) {
2486+
fn change_baud(&self, config: &Config) -> Result<(), ConfigError> {
24482487
let clocks = Clocks::get();
24492488
let clk = match config.clock_source {
24502489
ClockSource::Apb => clocks.apb_clock.to_Hz(),
@@ -2499,10 +2538,28 @@ impl Info {
24992538
.write(|w| unsafe { w.clkdiv().bits(divider).frag().bits(0) });
25002539

25012540
self.sync_regs();
2541+
2542+
let actual_baud = self.get_baudrate(clk);
2543+
2544+
match config.baudrate_tolerance {
2545+
BaudrateTolerance::Exact if actual_baud != config.baudrate => {
2546+
return Err(ConfigError::UnachievableBaudrate)
2547+
}
2548+
BaudrateTolerance::ErrorPercent(percent) => {
2549+
let deviation = ((config.baudrate as i64 - actual_baud as i64).abs() as u64 * 100)
2550+
/ actual_baud as u64;
2551+
if deviation > percent as u64 {
2552+
return Err(ConfigError::UnachievableBaudrate);
2553+
}
2554+
}
2555+
_ => {}
2556+
}
2557+
2558+
Ok(())
25022559
}
25032560

25042561
#[cfg(any(esp32, esp32s2))]
2505-
fn change_baud(&self, config: &Config) {
2562+
fn change_baud(&self, config: &Config) -> Result<(), ConfigError> {
25062563
let clk = match config.clock_source {
25072564
ClockSource::Apb => Clocks::get().apb_clock.to_Hz(),
25082565
// ESP32(/-S2) TRM, section 3.2.4.2 (6.2.4.2 for S2)
@@ -2519,6 +2576,24 @@ impl Info {
25192576
self.regs()
25202577
.clkdiv()
25212578
.write(|w| unsafe { w.clkdiv().bits(divider).frag().bits(0) });
2579+
2580+
let actual_baud = self.get_baudrate(clk);
2581+
2582+
match config.baudrate_tolerance {
2583+
BaudrateTolerance::Exact if actual_baud != config.baudrate => {
2584+
return Err(ConfigError::UnachievableBaudrate)
2585+
}
2586+
BaudrateTolerance::ErrorPercent(percent) => {
2587+
let deviation = ((config.baudrate as i64 - actual_baud as i64).abs() as u64 * 100)
2588+
/ actual_baud as u64;
2589+
if deviation > percent as u64 {
2590+
return Err(ConfigError::UnachievableBaudrate);
2591+
}
2592+
}
2593+
_ => {}
2594+
}
2595+
2596+
Ok(())
25222597
}
25232598

25242599
fn change_data_bits(&self, data_bits: DataBits) {
@@ -2575,6 +2650,32 @@ impl Info {
25752650
txfifo_rst(self.regs(), true);
25762651
txfifo_rst(self.regs(), false);
25772652
}
2653+
2654+
fn get_baudrate(&self, clk: u32) -> u32 {
2655+
// taken from https://github.com/espressif/esp-idf/blob/master/components/hal/esp32c6/include/hal/uart_ll.h#L433-L444
2656+
// (it's different for different chips)
2657+
cfg_if::cfg_if! {
2658+
if #[cfg(any(esp32, esp32s2))] {
2659+
let clkdiv_reg = self.regs().clkdiv().read();
2660+
let clkdiv = clkdiv_reg.clkdiv().bits() as u32;
2661+
let clkdiv_frag = clkdiv_reg.frag().bits() as u32;
2662+
(clk << 4) / ((clkdiv << 4) | clkdiv_frag)
2663+
} else if #[cfg(any(esp32c2, esp32c3, esp32s3))] {
2664+
let clkdiv_reg = self.regs().clkdiv().read();
2665+
let clkdiv = clkdiv_reg.clkdiv().bits() as u32;
2666+
let clkdiv_frag = clkdiv_reg.frag().bits() as u32;
2667+
let sclk_div_num = self.regs().clk_conf().read().sclk_div_num().bits() as u32;
2668+
(clk << 4) / (((clkdiv << 4) | clkdiv_frag) * (sclk_div_num + 1))
2669+
} else if #[cfg(any(esp32c6, esp32h2))] {
2670+
let pcr = crate::peripherals::PCR::regs();
2671+
let clkdiv_reg = self.regs().clkdiv().read();
2672+
let clkdiv = clkdiv_reg.clkdiv().bits() as u32;
2673+
let clkdiv_frag = clkdiv_reg.frag().bits() as u32;
2674+
let sclk_div_num = pcr.uart0_sclk_conf().read().uart0_sclk_div_num().bits() as u32;
2675+
(clk << 4) / (((clkdiv << 4) | clkdiv_frag) * (sclk_div_num + 1))
2676+
}
2677+
}
2678+
}
25782679
}
25792680

25802681
impl PartialEq for Info {

0 commit comments

Comments
 (0)