@@ -265,6 +265,18 @@ pub enum StopBits {
265
265
_2 = 3 ,
266
266
}
267
267
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
+
268
280
/// UART Configuration
269
281
#[ derive( Debug , Clone , Copy , procmacros:: BuilderLite ) ]
270
282
#[ cfg_attr( feature = "defmt" , derive( defmt:: Format ) ) ]
@@ -273,6 +285,9 @@ pub struct Config {
273
285
/// The baud rate (speed) of the UART communication in bits per second
274
286
/// (bps).
275
287
baudrate : u32 ,
288
+ /// Determines how close to the desired baud rate value the driver should
289
+ /// set the baud rate.
290
+ baudrate_tolerance : BaudrateTolerance ,
276
291
/// Number of data bits in each frame (5, 6, 7, or 8 bits).
277
292
data_bits : DataBits ,
278
293
/// Parity setting (None, Even, or Odd).
@@ -305,6 +320,7 @@ impl Default for Config {
305
320
rx : RxConfig :: default ( ) ,
306
321
tx : TxConfig :: default ( ) ,
307
322
baudrate : 115_200 ,
323
+ baudrate_tolerance : BaudrateTolerance :: default ( ) ,
308
324
data_bits : Default :: default ( ) ,
309
325
parity : Default :: default ( ) ,
310
326
stop_bits : Default :: default ( ) ,
@@ -463,6 +479,8 @@ pub struct UartRx<'d, Dm> {
463
479
#[ cfg_attr( feature = "defmt" , derive( defmt:: Format ) ) ]
464
480
#[ non_exhaustive]
465
481
pub enum ConfigError {
482
+ /// The requested baud rate is not achievable.
483
+ UnachievableBaudrate ,
466
484
/// The requested baud rate is not supported.
467
485
UnsupportedBaudrate ,
468
486
/// The requested timeout is not supported.
@@ -476,6 +494,9 @@ impl core::error::Error for ConfigError {}
476
494
impl core:: fmt:: Display for ConfigError {
477
495
fn fmt ( & self , f : & mut core:: fmt:: Formatter < ' _ > ) -> core:: fmt:: Result {
478
496
match self {
497
+ ConfigError :: UnachievableBaudrate => {
498
+ write ! ( f, "The requested baud rate is not achievable" )
499
+ }
479
500
ConfigError :: UnsupportedBaudrate => {
480
501
write ! ( f, "The requested baud rate is not supported" )
481
502
}
@@ -2205,7 +2226,7 @@ impl Info {
2205
2226
2206
2227
fn apply_config ( & self , config : & Config ) -> Result < ( ) , ConfigError > {
2207
2228
config. validate ( ) ?;
2208
- self . change_baud ( config) ;
2229
+ self . change_baud ( config) ? ;
2209
2230
self . change_data_bits ( config. data_bits ) ;
2210
2231
self . change_parity ( config. parity ) ;
2211
2232
self . change_stop_bits ( config. stop_bits ) ;
@@ -2394,7 +2415,7 @@ impl Info {
2394
2415
}
2395
2416
2396
2417
#[ cfg( any( esp32c2, esp32c3, esp32s3) ) ]
2397
- fn change_baud ( & self , config : & Config ) {
2418
+ fn change_baud ( & self , config : & Config ) -> Result < ( ) , ConfigError > {
2398
2419
use crate :: peripherals:: LPWR ;
2399
2420
2400
2421
let clocks = Clocks :: get ( ) ;
@@ -2433,6 +2454,24 @@ impl Info {
2433
2454
self . regs ( )
2434
2455
. clkdiv ( )
2435
2456
. 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 ( ( ) )
2436
2475
}
2437
2476
2438
2477
fn is_instance ( & self , other : impl Instance ) -> bool {
@@ -2444,7 +2483,7 @@ impl Info {
2444
2483
}
2445
2484
2446
2485
#[ cfg( any( esp32c6, esp32h2) ) ]
2447
- fn change_baud ( & self , config : & Config ) {
2486
+ fn change_baud ( & self , config : & Config ) -> Result < ( ) , ConfigError > {
2448
2487
let clocks = Clocks :: get ( ) ;
2449
2488
let clk = match config. clock_source {
2450
2489
ClockSource :: Apb => clocks. apb_clock . to_Hz ( ) ,
@@ -2499,10 +2538,28 @@ impl Info {
2499
2538
. write ( |w| unsafe { w. clkdiv ( ) . bits ( divider) . frag ( ) . bits ( 0 ) } ) ;
2500
2539
2501
2540
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 ( ( ) )
2502
2559
}
2503
2560
2504
2561
#[ cfg( any( esp32, esp32s2) ) ]
2505
- fn change_baud ( & self , config : & Config ) {
2562
+ fn change_baud ( & self , config : & Config ) -> Result < ( ) , ConfigError > {
2506
2563
let clk = match config. clock_source {
2507
2564
ClockSource :: Apb => Clocks :: get ( ) . apb_clock . to_Hz ( ) ,
2508
2565
// ESP32(/-S2) TRM, section 3.2.4.2 (6.2.4.2 for S2)
@@ -2519,6 +2576,24 @@ impl Info {
2519
2576
self . regs ( )
2520
2577
. clkdiv ( )
2521
2578
. 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 ( ( ) )
2522
2597
}
2523
2598
2524
2599
fn change_data_bits ( & self , data_bits : DataBits ) {
@@ -2575,6 +2650,32 @@ impl Info {
2575
2650
txfifo_rst ( self . regs ( ) , true ) ;
2576
2651
txfifo_rst ( self . regs ( ) , false ) ;
2577
2652
}
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
+ }
2578
2679
}
2579
2680
2580
2681
impl PartialEq for Info {
0 commit comments