use embassy_time::{Duration, Timer}; use embedded_hal::digital::v2::OutputPin; use embedded_hal_async::digital::Wait; use embedded_hal_async::spi::SpiBus; use super::mod_params::RadioError::*; use super::mod_params::*; use super::LoRa; // Defines the time required for the TCXO to wakeup [ms]. const BRD_TCXO_WAKEUP_TIME: u32 = 10; // Provides board-specific functionality for Semtech SX126x-based boards. impl LoRa where SPI: SpiBus, CTRL: OutputPin, WAIT: Wait, { // De-initialize the radio I/Os pins interface. Useful when going into MCU low power modes. pub(super) async fn brd_io_deinit(&mut self) -> Result<(), RadioError> { Ok(()) // no operation currently } // Initialize the TCXO power pin pub(super) async fn brd_io_tcxo_init(&mut self) -> Result<(), RadioError> { let timeout = self.brd_get_board_tcxo_wakeup_time() << 6; self.sub_set_dio3_as_tcxo_ctrl(TcxoCtrlVoltage::Ctrl1V7, timeout) .await?; Ok(()) } // Initialize RF switch control pins pub(super) async fn brd_io_rf_switch_init(&mut self) -> Result<(), RadioError> { self.sub_set_dio2_as_rf_switch_ctrl(true).await?; Ok(()) } // Initialize the radio debug pins pub(super) async fn brd_io_dbg_init(&mut self) -> Result<(), RadioError> { Ok(()) // no operation currently } // Hardware reset of the radio pub(super) async fn brd_reset(&mut self) -> Result<(), RadioError> { Timer::after(Duration::from_millis(10)).await; self.reset.set_low().map_err(|_| Reset)?; Timer::after(Duration::from_millis(20)).await; self.reset.set_high().map_err(|_| Reset)?; Timer::after(Duration::from_millis(10)).await; Ok(()) } // Wait while the busy pin is high pub(super) async fn brd_wait_on_busy(&mut self) -> Result<(), RadioError> { self.busy.wait_for_low().await.map_err(|_| Busy)?; Ok(()) } // Wake up the radio pub(super) async fn brd_wakeup(&mut self) -> Result<(), RadioError> { self.cs.set_low().map_err(|_| CS)?; self.spi.write(&[OpCode::GetStatus.value()]).await.map_err(SPI)?; self.spi.write(&[0x00]).await.map_err(SPI)?; self.cs.set_high().map_err(|_| CS)?; self.brd_wait_on_busy().await?; self.brd_set_operating_mode(RadioMode::StandbyRC); Ok(()) } // Send a command that writes data to the radio pub(super) async fn brd_write_command(&mut self, op_code: OpCode, buffer: &[u8]) -> Result<(), RadioError> { self.sub_check_device_ready().await?; self.cs.set_low().map_err(|_| CS)?; self.spi.write(&[op_code.value()]).await.map_err(SPI)?; self.spi.write(buffer).await.map_err(SPI)?; self.cs.set_high().map_err(|_| CS)?; if op_code != OpCode::SetSleep { self.brd_wait_on_busy().await?; } Ok(()) } // Send a command that reads data from the radio, filling the provided buffer and returning a status pub(super) async fn brd_read_command(&mut self, op_code: OpCode, buffer: &mut [u8]) -> Result> { let mut status = [0u8]; let mut input = [0u8]; self.sub_check_device_ready().await?; self.cs.set_low().map_err(|_| CS)?; self.spi.write(&[op_code.value()]).await.map_err(SPI)?; self.spi.transfer(&mut status, &[0x00]).await.map_err(SPI)?; for i in 0..buffer.len() { self.spi.transfer(&mut input, &[0x00]).await.map_err(SPI)?; buffer[i] = input[0]; } self.cs.set_high().map_err(|_| CS)?; self.brd_wait_on_busy().await?; Ok(status[0]) } // Write one or more bytes of data to the radio memory pub(super) async fn brd_write_registers( &mut self, start_register: Register, buffer: &[u8], ) -> Result<(), RadioError> { self.sub_check_device_ready().await?; self.cs.set_low().map_err(|_| CS)?; self.spi.write(&[OpCode::WriteRegister.value()]).await.map_err(SPI)?; self.spi .write(&[ ((start_register.addr() & 0xFF00) >> 8) as u8, (start_register.addr() & 0x00FF) as u8, ]) .await .map_err(SPI)?; self.spi.write(buffer).await.map_err(SPI)?; self.cs.set_high().map_err(|_| CS)?; self.brd_wait_on_busy().await?; Ok(()) } // Read one or more bytes of data from the radio memory pub(super) async fn brd_read_registers( &mut self, start_register: Register, buffer: &mut [u8], ) -> Result<(), RadioError> { let mut input = [0u8]; self.sub_check_device_ready().await?; self.cs.set_low().map_err(|_| CS)?; self.spi.write(&[OpCode::ReadRegister.value()]).await.map_err(SPI)?; self.spi .write(&[ ((start_register.addr() & 0xFF00) >> 8) as u8, (start_register.addr() & 0x00FF) as u8, 0x00u8, ]) .await .map_err(SPI)?; for i in 0..buffer.len() { self.spi.transfer(&mut input, &[0x00]).await.map_err(SPI)?; buffer[i] = input[0]; } self.cs.set_high().map_err(|_| CS)?; self.brd_wait_on_busy().await?; Ok(()) } // Write data to the buffer holding the payload in the radio pub(super) async fn brd_write_buffer(&mut self, offset: u8, buffer: &[u8]) -> Result<(), RadioError> { self.sub_check_device_ready().await?; self.cs.set_low().map_err(|_| CS)?; self.spi.write(&[OpCode::WriteBuffer.value()]).await.map_err(SPI)?; self.spi.write(&[offset]).await.map_err(SPI)?; self.spi.write(buffer).await.map_err(SPI)?; self.cs.set_high().map_err(|_| CS)?; self.brd_wait_on_busy().await?; Ok(()) } // Read data from the buffer holding the payload in the radio pub(super) async fn brd_read_buffer(&mut self, offset: u8, buffer: &mut [u8]) -> Result<(), RadioError> { let mut input = [0u8]; self.sub_check_device_ready().await?; self.cs.set_low().map_err(|_| CS)?; self.spi.write(&[OpCode::ReadBuffer.value()]).await.map_err(SPI)?; self.spi.write(&[offset]).await.map_err(SPI)?; self.spi.write(&[0x00]).await.map_err(SPI)?; for i in 0..buffer.len() { self.spi.transfer(&mut input, &[0x00]).await.map_err(SPI)?; buffer[i] = input[0]; } self.cs.set_high().map_err(|_| CS)?; self.brd_wait_on_busy().await?; Ok(()) } // Set the radio output power pub(super) async fn brd_set_rf_tx_power(&mut self, power: i8) -> Result<(), RadioError> { self.sub_set_tx_params(power, RampTime::Ramp40Us).await?; Ok(()) } // Get the radio type pub(super) fn brd_get_radio_type(&mut self) -> RadioType { RadioType::SX1262 } // Quiesce the antenna(s). pub(super) fn brd_ant_sleep(&mut self) -> Result<(), RadioError> { self.antenna_tx.set_low().map_err(|_| AntTx)?; self.antenna_rx.set_low().map_err(|_| AntRx)?; Ok(()) } // Prepare the antenna(s) for a receive operation pub(super) fn brd_ant_set_rx(&mut self) -> Result<(), RadioError> { self.antenna_tx.set_low().map_err(|_| AntTx)?; self.antenna_rx.set_high().map_err(|_| AntRx)?; Ok(()) } // Prepare the antenna(s) for a send operation pub(super) fn brd_ant_set_tx(&mut self) -> Result<(), RadioError> { self.antenna_rx.set_low().map_err(|_| AntRx)?; self.antenna_tx.set_high().map_err(|_| AntTx)?; Ok(()) } // Check if the given RF frequency is supported by the hardware pub(super) async fn brd_check_rf_frequency(&mut self, _frequency: u32) -> Result> { Ok(true) } // Get the duration required for the TCXO to wakeup [ms]. pub(super) fn brd_get_board_tcxo_wakeup_time(&mut self) -> u32 { BRD_TCXO_WAKEUP_TIME } /* Get current state of the DIO1 pin - not currently needed if waiting on DIO1 instead of using an IRQ process pub(super) async fn brd_get_dio1_pin_state( &mut self, ) -> Result> { Ok(0) } */ // Get the current radio operatiing mode pub(super) fn brd_get_operating_mode(&mut self) -> RadioMode { self.operating_mode } // Set/Update the current radio operating mode This function is only required to reflect the current radio operating mode when processing interrupts. pub(super) fn brd_set_operating_mode(&mut self, mode: RadioMode) { self.operating_mode = mode; } }