Skip to content

Commit

Permalink
Merge #408
Browse files Browse the repository at this point in the history
408: Add halfduplex to serial::config::Config and example r=richardeoin a=dsmcfarl

This allows the USART's to be used with serial devices that operate in a half-duplex mode (transmit and receive on one wire). An example was also added (serial-halfduplex) and has been tested with a NUCLEO-H723ZG board. It has also been tested with a Frysky RX4R RC receiver Smart Port interface.

Co-authored-by: Dan McFarlane <dsmcfarl@gmail.com>
Co-authored-by: Richard Meadows <962920+richardeoin@users.noreply.github.com>
  • Loading branch information
3 people authored Mar 8, 2023
2 parents 11518e0 + b482c05 commit c1a9d82
Show file tree
Hide file tree
Showing 2 changed files with 112 additions and 0 deletions.
86 changes: 86 additions & 0 deletions examples/serial-halfduplex.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
//! Example of half-duplex USART communication between two USART's.
//!
//! Connect the USART3_TX (PC10) and USART6_TX (PC6) and an external pullup resistor together.
//! i.e., connect one end of a resistor to a 3.3V pin and connect both PC10 and PC6 pins to the other end
//! of the resistor. 330 ohm resistor worked but exact value is not critical.
//!
//! It will print "Hello, world!" over and over again one character at a time.
#![deny(warnings)]
#![no_main]
#![no_std]

use cortex_m_rt::entry;
#[macro_use]
mod utilities;
use log::info;

use stm32h7xx_hal::{pac, prelude::*, serial};

use nb::block;

#[entry]
fn main() -> ! {
utilities::logger::init();
let dp = pac::Peripherals::take().unwrap();

// Constrain and Freeze power
let pwr = dp.PWR.constrain();
let pwrcfg = example_power!(pwr).freeze();

// Constrain and Freeze clock
let rcc = dp.RCC.constrain();
let ccdr = rcc.sys_ck(160.MHz()).freeze(pwrcfg, &dp.SYSCFG);

// Acquire the GPIOC peripheral. This also enables the clock for
// GPIOC in the RCC register.
let gpioc = dp.GPIOC.split(ccdr.peripheral.GPIOC);

let usart3txpin = gpioc.pc10.into_alternate::<7>();
let usart3rxpin = gpioc.pc11.into_alternate::<7>();
let usart6txpin = gpioc.pc6.into_alternate::<7>();
let usart6rxpin = gpioc.pc7.into_alternate::<7>();

info!("stm32h7xx-hal example - USART half-duplex");

// Configure the serial peripheral.
let config = serial::config::Config::new(9600.bps())
.parity_none()
.halfduplex(true)
.stopbits(serial::config::StopBits::Stop1);

let mut usart3 = dp
.USART3
.serial(
(usart3txpin, usart3rxpin),
config,
ccdr.peripheral.USART3,
&ccdr.clocks,
)
.unwrap();

let mut usart6 = dp
.USART6
.serial(
(usart6txpin, usart6rxpin),
config,
ccdr.peripheral.USART6,
&ccdr.clocks,
)
.unwrap();

loop {
// Tx from usart3
for c in "Hello, world!".chars() {
block!(usart3.write(c as u8)).unwrap();
let received = block!(usart6.read()).unwrap();
info!("usart6 rx {}", received as char);
}

// Tx from usart6
for c in "Hello, world!".chars() {
block!(usart6.write(c as u8)).unwrap();
let received = block!(usart3.read()).unwrap();
info!("usart3 rx {}", received as char);
}
}
}
26 changes: 26 additions & 0 deletions src/serial.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ use nb::block;
use stm32::usart1::cr2::{
CLKEN_A, CPHA_A, CPOL_A, LBCL_A, MSBFIRST_A, RXINV_A, TXINV_A,
};
use stm32::usart1::cr3::HDSEL_A;

use crate::gpio::{self, Alternate};
use crate::rcc::{rec, CoreClocks, ResetEnable};
Expand Down Expand Up @@ -144,6 +145,7 @@ pub mod config {
pub inverttx: bool,
pub rxfifothreshold: FifoThreshold,
pub txfifothreshold: FifoThreshold,
pub halfduplex: bool,
}

impl Config {
Expand All @@ -165,6 +167,7 @@ pub mod config {
inverttx: false,
rxfifothreshold: FifoThreshold::Eighth,
txfifothreshold: FifoThreshold::Eighth,
halfduplex: false,
}
}

Expand Down Expand Up @@ -256,6 +259,12 @@ pub mod config {
self.txfifothreshold = txfifothreshold;
self
}

/// If `true`, sets to half-duplex mode
pub fn halfduplex(mut self, halfduplex: bool) -> Self {
self.halfduplex = halfduplex;
self
}
}

#[derive(Debug)]
Expand Down Expand Up @@ -596,6 +605,14 @@ macro_rules! usart {
let mut serial = Serial { usart, ker_ck };
let config = config.into();
serial.usart.cr1.reset();

// If synchronous mode is supported, check that it is not
// enabled alongside half duplex mode
$(
if config.halfduplex & $synchronous {
return Err(config::InvalidConfig);
}
)?
serial.configure(&config $(, $synchronous )?);

Ok(serial)
Expand Down Expand Up @@ -650,6 +667,15 @@ macro_rules! usart {
self.usart.cr3.modify(|_, w| w.txftcfg().bits(fifo_threshold_bits));
}

// Configure half-duplex mode
self.usart.cr3.modify(|_, w| {
w.hdsel().variant(if config.halfduplex {
HDSEL_A::Selected
} else {
HDSEL_A::NotSelected
})
});

// Configure serial mode
self.usart.cr2.write(|w| {
w.stop().variant(match config.stopbits {
Expand Down

0 comments on commit c1a9d82

Please sign in to comment.