From 16dfbb96b16f9bf5bef5ec5bd1649ddcf1e45039 Mon Sep 17 00:00:00 2001 From: Priit Laes Date: Tue, 7 Nov 2023 13:23:28 +0200 Subject: [PATCH] sx127x: Add IRQ's for DIO1 and DIO3 and use it for ValidHeader irq --- src/sx1276_7_8_9/mod.rs | 20 +++++++++----- src/sx1276_7_8_9/radio_kind_params.rs | 40 ++++++++++++++++++++++++++- 2 files changed, 52 insertions(+), 8 deletions(-) diff --git a/src/sx1276_7_8_9/mod.rs b/src/sx1276_7_8_9/mod.rs index 3831821..3079d1c 100644 --- a/src/sx1276_7_8_9/mod.rs +++ b/src/sx1276_7_8_9/mod.rs @@ -478,7 +478,9 @@ where .await } - // Set the IRQ mask to disable unwanted interrupts, enable interrupts on DIO0 (the IRQ pin), and allow interrupts. + // Set the IRQ mask to disable unwanted interrupts, + // enable interrupts on DIO pins (sx127x has multiple), + // and allow interrupts. async fn set_irq_params(&mut self, radio_mode: Option) -> Result<(), RadioError> { match radio_mode { Some(RadioMode::Transmit) => { @@ -500,15 +502,19 @@ where self.write_register( Register::RegIrqFlagsMask, IrqMask::All.value() - ^ (IrqMask::RxDone.value() | IrqMask::RxTimeout.value() | IrqMask::CRCError.value()), + ^ (IrqMask::RxDone.value() | IrqMask::RxTimeout.value() | IrqMask::CRCError.value() | IrqMask::HeaderValid.value()), false, ) .await?; - let mut dio_mapping_1 = self.read_register(Register::RegDioMapping1).await?; - dio_mapping_1 = (dio_mapping_1 & DioMapping1Dio0::Mask.value()) | DioMapping1Dio0::RxDone.value(); - self.write_register(Register::RegDioMapping1, dio_mapping_1, false) - .await?; + // HeaderValid and CRCError are mutually exclusive when attempting to + // trigger DIO-based interrupt, so our approach is to trigger HeaderValid + // as this is required for preamble detection. + // TODO: RxTimeout should be configured on DIO1 + let dio_mapping_1 = self.read_register(Register::RegDioMapping1).await?; + let val = (dio_mapping_1 & DioMapping1Dio0::Mask.value() & DioMapping1Dio3::Mask.value()) + | (DioMapping1Dio0::RxDone.value() | DioMapping1Dio3::ValidHeader.value()); + self.write_register(Register::RegDioMapping1, val, false).await?; self.write_register(Register::RegIrqFlags, 0x00u8, false).await?; } @@ -578,7 +584,7 @@ where self.write_register(Register::RegIrqFlags, 0xffu8, false).await?; // clear all interrupts debug!( - "process_irq satisfied: irq_flags = 0x{:x} in radio mode {}", + "process_irq: irq_flags = 0b{:08b} in radio mode {}", irq_flags, radio_mode ); diff --git a/src/sx1276_7_8_9/radio_kind_params.rs b/src/sx1276_7_8_9/radio_kind_params.rs index 8ae9505..7f8cc42 100644 --- a/src/sx1276_7_8_9/radio_kind_params.rs +++ b/src/sx1276_7_8_9/radio_kind_params.rs @@ -18,7 +18,14 @@ impl LoRaMode { } } -#[derive(Clone, Copy)] +// IRQ mapping for sx127x chips: +// DIO0 - RxDone, TxDone, CadDone +// DIO1 - RxTimeout, FhssChangeChannel, CadDetected +// DIO2 - 3x FhssChangeChannel +// DIO3 - CadDone, ValidHeader, PayloadCrcError +// DIO4 - CadDetected, *PllLock, *PllLock +// DIO5 - *ModeReady, *ClkOut, *ClkOut + #[allow(dead_code)] pub enum DioMapping1Dio0 { RxDone = 0x00, @@ -34,6 +41,37 @@ impl DioMapping1Dio0 { } } +#[allow(dead_code)] +pub enum DioMapping1Dio1 { + RxTimeOut = 0b00 << 2, + FhssChangeChannel = 0b01 << 2, + CadDetected = 0b10 << 2, + Other = 0b11 << 2, + Mask = 0xf3, +} + +#[allow(dead_code)] +impl DioMapping1Dio1 { + pub fn value(self) -> u8 { + self as u8 + } +} + +#[allow(dead_code)] +pub enum DioMapping1Dio3 { + CadDone = 0, + ValidHeader = 0b01, + PayloadCrcError = 0b10, + Other = 0b11, + Mask = 0xfc, +} + +impl DioMapping1Dio3 { + pub fn value(self) -> u8 { + self as u8 + } +} + #[derive(Clone, Copy)] #[allow(dead_code)] pub enum IrqMask {