use core::borrow::BorrowMut;
use core::ffi::CStr;
use core::marker::PhantomData;
use core::num::NonZeroU32;
use enumset::{EnumSet, EnumSetType};
use esp_idf_sys::*;
use num_enum::TryFromPrimitive;
use crate::cpu::Core;
use crate::delay::{self, BLOCK, NON_BLOCK};
use crate::interrupt::InterruptType;
use crate::peripheral::{Peripheral, PeripheralRef};
use crate::task::asynch::Notification;
use crate::{gpio::*, task};
crate::embedded_hal_error!(CanError, embedded_can::Error, embedded_can::ErrorKind);
crate::embedded_hal_error!(
Can02Error,
embedded_hal_0_2::can::Error,
embedded_hal_0_2::can::ErrorKind
);
pub type CanConfig = config::Config;
pub mod config {
use enumset::EnumSet;
use esp_idf_sys::*;
use crate::interrupt::InterruptType;
use super::Alert;
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
pub enum Timing {
B25K,
B50K,
B100K,
B125K,
B250K,
B500K,
B800K,
B1M,
Custom {
baudrate_prescaler: u32,
timing_segment_1: u8,
timing_segment_2: u8,
synchronization_jump_width: u8,
triple_sampling: bool,
},
}
impl From<Timing> for twai_timing_config_t {
#[allow(clippy::needless_update)]
fn from(resolution: Timing) -> Self {
match resolution {
Timing::B25K => twai_timing_config_t {
brp: 128,
tseg_1: 16,
tseg_2: 8,
sjw: 3,
..Default::default()
},
Timing::B50K => twai_timing_config_t {
brp: 80,
tseg_1: 15,
tseg_2: 4,
sjw: 3,
..Default::default()
},
Timing::B100K => twai_timing_config_t {
brp: 40,
tseg_1: 15,
tseg_2: 4,
sjw: 3,
..Default::default()
},
Timing::B125K => twai_timing_config_t {
brp: 32,
tseg_1: 15,
tseg_2: 4,
sjw: 3,
..Default::default()
},
Timing::B250K => twai_timing_config_t {
brp: 16,
tseg_1: 15,
tseg_2: 4,
sjw: 3,
..Default::default()
},
Timing::B500K => twai_timing_config_t {
brp: 8,
tseg_1: 15,
tseg_2: 4,
sjw: 3,
..Default::default()
},
Timing::B800K => twai_timing_config_t {
brp: 4,
tseg_1: 16,
tseg_2: 8,
sjw: 3,
..Default::default()
},
Timing::B1M => twai_timing_config_t {
brp: 4,
tseg_1: 15,
tseg_2: 4,
sjw: 3,
..Default::default()
},
Timing::Custom {
baudrate_prescaler,
timing_segment_1,
timing_segment_2,
synchronization_jump_width,
triple_sampling,
} => twai_timing_config_t {
brp: baudrate_prescaler,
tseg_1: timing_segment_1,
tseg_2: timing_segment_2,
sjw: synchronization_jump_width,
#[cfg(any(
esp_idf_version_major = "4",
esp_idf_version = "5.0",
esp_idf_version = "5.1",
esp_idf_version = "5.2",
esp_idf_version = "5.3",
esp_idf_version = "5.4"
))]
triple_sampling,
#[cfg(not(any(
esp_idf_version_major = "4",
esp_idf_version = "5.0",
esp_idf_version = "5.1",
esp_idf_version = "5.2",
esp_idf_version = "5.3",
esp_idf_version = "5.4"
)))]
__bindgen_anon_1: twai_timing_config_t__bindgen_ty_1 { triple_sampling },
..Default::default()
},
}
}
}
impl Default for Timing {
fn default() -> Self {
Self::B500K
}
}
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
pub enum Filter {
Standard { filter: u16, mask: u16 },
Dual {
filter1: u16,
mask1: u16,
filter2: u16,
mask2: u16,
},
Extended { filter: u32, mask: u32 },
}
impl Filter {
pub fn standard_allow_all() -> Self {
Self::Standard { filter: 0, mask: 0 }
}
pub fn extended_allow_all() -> Self {
Self::Extended { filter: 0, mask: 0 }
}
}
impl Default for Filter {
fn default() -> Self {
Filter::standard_allow_all()
}
}
#[derive(Debug, Copy, Clone, Default)]
pub enum Mode {
#[default]
Normal,
NoAck,
ListenOnly,
}
impl From<Mode> for twai_mode_t {
fn from(val: Mode) -> Self {
match val {
Mode::Normal => twai_mode_t_TWAI_MODE_NORMAL,
Mode::NoAck => twai_mode_t_TWAI_MODE_NO_ACK,
Mode::ListenOnly => twai_mode_t_TWAI_MODE_LISTEN_ONLY,
}
}
}
#[derive(Debug, Clone)]
pub struct Config {
pub timing: Timing,
pub filter: Filter,
pub tx_queue_len: u32,
pub rx_queue_len: u32,
pub mode: Mode,
pub alerts: EnumSet<Alert>,
pub intr_flags: EnumSet<InterruptType>,
}
impl Config {
pub fn new() -> Self {
Self {
timing: Default::default(),
filter: Default::default(),
tx_queue_len: 5,
rx_queue_len: 5,
mode: Default::default(),
alerts: Default::default(),
intr_flags: EnumSet::<InterruptType>::empty(),
}
}
#[must_use]
pub fn timing(mut self, timing: Timing) -> Self {
self.timing = timing;
self
}
#[must_use]
pub fn filter(mut self, filter: Filter) -> Self {
self.filter = filter;
self
}
#[must_use]
pub fn tx_queue_len(mut self, tx_queue_len: u32) -> Self {
self.tx_queue_len = tx_queue_len;
self
}
#[must_use]
pub fn rx_queue_len(mut self, rx_queue_len: u32) -> Self {
self.rx_queue_len = rx_queue_len;
self
}
#[must_use]
pub fn mode(mut self, mode: Mode) -> Self {
self.mode = mode;
self
}
#[must_use]
pub fn alerts(mut self, alerts: EnumSet<Alert>) -> Self {
self.alerts = alerts;
self
}
#[must_use]
pub fn intr_flags(mut self, flags: EnumSet<InterruptType>) -> Self {
self.intr_flags = flags;
self
}
}
impl Default for Config {
fn default() -> Self {
Self::new()
}
}
}
#[derive(Debug, EnumSetType, TryFromPrimitive)]
#[enumset(repr = "u32")]
#[repr(u32)]
pub enum Alert {
TransmitIdle = 1,
Success = 2,
Received = 3,
BelowErrorWarning = 4,
ActiveError = 5,
RecoveryInProgress = 6,
BusRecovered = 7,
ArbLost = 8,
AboveErrorWarning = 9,
BusError = 10,
TransmitFailed = 11,
ReceiveQueueFull = 12,
ErrorPass = 13,
BusOffline = 14,
ReceiveFifoOverflow = 15,
TransmitRetried = 16,
PeripheralReset = 17,
AlertAndLog = 18,
}
pub struct CanDriver<'d>(PeripheralRef<'d, CAN>, EnumSet<Alert>, bool);
impl<'d> CanDriver<'d> {
pub fn new(
can: impl Peripheral<P = CAN> + 'd,
tx: impl Peripheral<P = impl OutputPin> + 'd,
rx: impl Peripheral<P = impl InputPin> + 'd,
config: &config::Config,
) -> Result<Self, EspError> {
crate::into_ref!(can, tx, rx);
#[allow(clippy::needless_update)]
let general_config = twai_general_config_t {
mode: config.mode.into(),
tx_io: tx.pin(),
rx_io: rx.pin(),
clkout_io: -1,
bus_off_io: -1,
tx_queue_len: config.tx_queue_len,
rx_queue_len: config.rx_queue_len,
alerts_enabled: config.alerts.as_repr(),
clkout_divider: 0,
intr_flags: InterruptType::to_native(config.intr_flags) as _,
..Default::default()
};
let timing_config = config.timing.into();
let (filter, mask, single_filter) = match config.filter {
config::Filter::Standard { filter, mask } => {
((filter as u32) << 21, !((mask as u32) << 21), true)
}
config::Filter::Extended { filter, mask } => (filter << 3, !(mask << 3), true),
config::Filter::Dual {
filter1,
mask1,
filter2,
mask2,
} => (
((filter1 as u32) << 21) | ((filter2 as u32) << 5),
!(((mask1 as u32) << 21) | ((mask2 as u32) << 5)),
false,
),
};
let filter_config = twai_filter_config_t {
acceptance_code: filter,
acceptance_mask: mask,
single_filter,
};
esp!(unsafe { twai_driver_install(&general_config, &timing_config, &filter_config) })?;
Ok(Self(can, config.alerts, config.tx_queue_len > 0))
}
pub fn start(&mut self) -> Result<(), EspError> {
esp!(unsafe { twai_start() })
}
pub fn stop(&mut self) -> Result<(), EspError> {
esp!(unsafe { twai_stop() })
}
pub fn transmit(&self, frame: &Frame, timeout: TickType_t) -> Result<(), EspError> {
esp!(unsafe { twai_transmit(&frame.0, timeout) })
}
pub fn receive(&self, timeout: TickType_t) -> Result<Frame, EspError> {
let mut rx_msg = Default::default();
match esp_result!(unsafe { twai_receive(&mut rx_msg, timeout) }, ()) {
Ok(_) => Ok(Frame(rx_msg)),
Err(err) => Err(err),
}
}
pub fn read_alerts(&self, timeout: TickType_t) -> Result<EnumSet<Alert>, EspError> {
let mut alerts = 0;
esp!(unsafe { twai_read_alerts(&mut alerts, timeout) })?;
Ok(EnumSet::from_repr_truncated(alerts))
}
}
impl Drop for CanDriver<'_> {
fn drop(&mut self) {
let _ = self.stop();
esp!(unsafe { twai_driver_uninstall() }).unwrap();
}
}
unsafe impl Send for CanDriver<'_> {}
impl embedded_hal_0_2::blocking::can::Can for CanDriver<'_> {
type Frame = Frame;
type Error = Can02Error;
fn transmit(&mut self, frame: &Self::Frame) -> Result<(), Self::Error> {
CanDriver::transmit(self, frame, BLOCK).map_err(Can02Error::other)
}
fn receive(&mut self) -> Result<Self::Frame, Self::Error> {
CanDriver::receive(self, BLOCK).map_err(Can02Error::other)
}
}
impl embedded_can::blocking::Can for CanDriver<'_> {
type Frame = Frame;
type Error = CanError;
fn transmit(&mut self, frame: &Self::Frame) -> Result<(), Self::Error> {
CanDriver::transmit(self, frame, BLOCK).map_err(CanError::other)
}
fn receive(&mut self) -> Result<Self::Frame, Self::Error> {
CanDriver::receive(self, BLOCK).map_err(CanError::other)
}
}
impl embedded_hal_0_2::can::nb::Can for CanDriver<'_> {
type Frame = Frame;
type Error = Can02Error;
fn transmit(&mut self, frame: &Self::Frame) -> nb::Result<Option<Self::Frame>, Self::Error> {
match CanDriver::transmit(self, frame, NON_BLOCK) {
Ok(_) => Ok(None),
Err(e) if e.code() == ESP_FAIL => Err(nb::Error::WouldBlock),
Err(e) if e.code() == ESP_ERR_TIMEOUT => Err(nb::Error::WouldBlock),
Err(e) => Err(nb::Error::Other(Can02Error::other(e))),
}
}
fn receive(&mut self) -> nb::Result<Self::Frame, Self::Error> {
match CanDriver::receive(self, NON_BLOCK) {
Ok(frame) => Ok(frame),
Err(e) if e.code() == ESP_ERR_TIMEOUT => Err(nb::Error::WouldBlock),
Err(e) => Err(nb::Error::Other(Can02Error::other(e))),
}
}
}
impl embedded_can::nb::Can for CanDriver<'_> {
type Frame = Frame;
type Error = CanError;
fn transmit(&mut self, frame: &Self::Frame) -> nb::Result<Option<Self::Frame>, Self::Error> {
match CanDriver::transmit(self, frame, NON_BLOCK) {
Ok(_) => Ok(None),
Err(e) if e.code() == ESP_FAIL => Err(nb::Error::WouldBlock),
Err(e) if e.code() == ESP_ERR_TIMEOUT => Err(nb::Error::WouldBlock),
Err(e) => Err(nb::Error::Other(CanError::other(e))),
}
}
fn receive(&mut self) -> nb::Result<Self::Frame, Self::Error> {
match CanDriver::receive(self, NON_BLOCK) {
Ok(frame) => Ok(frame),
Err(e) if e.code() == ESP_ERR_TIMEOUT => Err(nb::Error::WouldBlock),
Err(e) => Err(nb::Error::Other(CanError::other(e))),
}
}
}
fn read_alerts() -> EnumSet<Alert> {
Alert::Success | Alert::Received | Alert::ReceiveQueueFull
}
fn write_alerts() -> EnumSet<Alert> {
Alert::Success | Alert::TransmitIdle | Alert::TransmitFailed | Alert::TransmitRetried
}
pub type OwnedAsyncCanDriver<'d> = AsyncCanDriver<'d, CanDriver<'d>>;
pub struct AsyncCanDriver<'d, T>
where
T: BorrowMut<CanDriver<'d>>,
{
driver: T,
task: TaskHandle_t,
_data: PhantomData<&'d ()>,
}
impl<'d> AsyncCanDriver<'d, CanDriver<'d>> {
pub fn new(
can: impl Peripheral<P = CAN> + 'd,
tx: impl Peripheral<P = impl OutputPin> + 'd,
rx: impl Peripheral<P = impl InputPin> + 'd,
config: &config::Config,
) -> Result<Self, EspError> {
Self::wrap(CanDriver::new(can, tx, rx, config)?)
}
}
impl<'d, T> AsyncCanDriver<'d, T>
where
T: BorrowMut<CanDriver<'d>>,
{
pub fn wrap(driver: T) -> Result<Self, EspError> {
Self::wrap_custom(driver, None, None)
}
pub fn wrap_custom(
mut driver: T,
priority: Option<u8>,
pin_to_core: Option<Core>,
) -> Result<Self, EspError> {
let _ = driver.borrow_mut().stop();
let mut alerts = 0;
esp!(unsafe {
twai_reconfigure_alerts(
driver
.borrow()
.1
.union(read_alerts())
.union(write_alerts())
.as_repr(),
&mut alerts,
)
})?;
let task = unsafe {
task::create(
Self::process_alerts,
CStr::from_bytes_until_nul(b"CAN - Alerts task\0").unwrap(),
2048,
core::ptr::null_mut(),
priority.unwrap_or(6),
pin_to_core,
)?
};
Ok(Self {
driver,
task,
_data: PhantomData,
})
}
pub fn driver(&self) -> &CanDriver<'d> {
self.driver.borrow()
}
pub fn driver_mut(&mut self) -> &mut CanDriver<'d> {
self.driver.borrow_mut()
}
pub fn start(&mut self) -> Result<(), EspError> {
self.driver.borrow_mut().start()
}
pub fn stop(&mut self) -> Result<(), EspError> {
self.driver.borrow_mut().stop()
}
pub async fn transmit(&self, frame: &Frame) -> Result<(), EspError> {
loop {
let res = self.driver.borrow().transmit(frame, delay::NON_BLOCK);
match res {
Ok(()) => return Ok(()),
Err(e)
if e.code() != ESP_ERR_TIMEOUT
&& (e.code() != ESP_FAIL || self.driver.borrow().2) =>
{
return Err(e)
}
_ => (),
}
WRITE_NOTIFICATION.wait().await;
}
}
pub async fn receive(&self) -> Result<Frame, EspError> {
loop {
let res = self.driver.borrow().receive(delay::NON_BLOCK);
match res {
Ok(frame) => return Ok(frame),
Err(e) if e.code() != ESP_ERR_TIMEOUT => return Err(e),
_ => (),
}
READ_NOTIFICATION.wait().await;
}
}
pub async fn read_alerts(&self) -> Result<EnumSet<Alert>, EspError> {
let alerts = loop {
let alerts = EnumSet::from_repr(ALERT_NOTIFICATION.wait().await.into())
.intersection(self.driver.borrow().1);
if !alerts.is_empty() {
break alerts;
}
};
Ok(alerts)
}
extern "C" fn process_alerts(_arg: *mut core::ffi::c_void) {
let mut alerts = 0;
loop {
if unsafe { twai_read_alerts(&mut alerts, delay::BLOCK) } == 0 {
let ealerts: EnumSet<Alert> = EnumSet::from_repr_truncated(alerts);
if !ealerts.is_disjoint(read_alerts()) {
READ_NOTIFICATION.notify_lsb();
}
if !ealerts.is_disjoint(write_alerts()) {
WRITE_NOTIFICATION.notify_lsb();
}
if let Some(alerts) = NonZeroU32::new(alerts) {
ALERT_NOTIFICATION.notify(alerts);
}
}
}
}
}
impl<'d, T> Drop for AsyncCanDriver<'d, T>
where
T: BorrowMut<CanDriver<'d>>,
{
fn drop(&mut self) {
let _ = self.stop();
unsafe { task::destroy(self.task) };
let mut alerts = 0;
esp!(unsafe { twai_reconfigure_alerts(self.driver.borrow().1.as_repr(), &mut alerts) })
.unwrap();
READ_NOTIFICATION.reset();
WRITE_NOTIFICATION.reset();
ALERT_NOTIFICATION.reset();
}
}
static READ_NOTIFICATION: Notification = Notification::new();
static WRITE_NOTIFICATION: Notification = Notification::new();
static ALERT_NOTIFICATION: Notification = Notification::new();
#[derive(Debug, EnumSetType)]
pub enum Flags {
Extended,
Remote,
SingleShot,
SelfReception,
DlcNonComp,
None,
}
pub struct Frame(twai_message_t);
impl Frame {
pub fn new(id: u32, flags: EnumSet<Flags>, data: &[u8]) -> Option<Self> {
let dlc = data.len();
if dlc <= 8 {
let mut twai_flags = twai_message_t__bindgen_ty_1::default();
for flag in flags.iter() {
match flag {
Flags::Extended => unsafe { twai_flags.__bindgen_anon_1.set_extd(1) },
Flags::Remote => unsafe { twai_flags.__bindgen_anon_1.set_rtr(1) },
Flags::SingleShot => unsafe { twai_flags.__bindgen_anon_1.set_ss(1) },
Flags::SelfReception => unsafe { twai_flags.__bindgen_anon_1.set_self(1) },
Flags::DlcNonComp => unsafe { twai_flags.__bindgen_anon_1.set_dlc_non_comp(1) },
Flags::None => {}
}
}
let mut payload = [0; 8];
payload[..dlc].copy_from_slice(data);
let twai_message = twai_message_t {
__bindgen_anon_1: twai_flags,
identifier: id,
data_length_code: dlc as u8,
data: payload,
};
Some(Frame(twai_message))
} else {
None
}
}
pub fn new_remote(id: u32, extended: bool, dlc: usize) -> Option<Self> {
if dlc <= 8 {
let mut flags = twai_message_t__bindgen_ty_1::default();
unsafe { flags.__bindgen_anon_1.set_rtr(1) };
if extended {
unsafe { flags.__bindgen_anon_1.set_extd(1) };
}
let twai_message = twai_message_t {
__bindgen_anon_1: flags,
identifier: id,
data_length_code: dlc as u8,
data: [0; 8],
};
Some(Frame(twai_message))
} else {
None
}
}
pub fn is_extended(&self) -> bool {
unsafe { self.0.__bindgen_anon_1.__bindgen_anon_1.extd() == 1 }
}
pub fn is_remote_frame(&self) -> bool {
unsafe { self.0.__bindgen_anon_1.__bindgen_anon_1.rtr() == 1 }
}
pub fn identifier(&self) -> u32 {
self.0.identifier
}
pub fn dlc(&self) -> usize {
self.0.data_length_code as usize
}
pub fn data(&self) -> &[u8] {
&self.0.data[..self.dlc()]
}
}
impl core::fmt::Display for Frame {
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
if self.is_extended() {
write!(
f,
"Frame {{ id: {:08x}, remote: {}, data: {:?} }}",
self.identifier(),
self.is_remote_frame(),
self.data()
)
} else {
write!(
f,
"Frame {{ id: {:03x}, remote: {}, data: {:?} }}",
self.identifier(),
self.is_remote_frame(),
self.data()
)
}
}
}
impl embedded_hal_0_2::can::Frame for Frame {
fn new(id: impl Into<embedded_hal_0_2::can::Id>, data: &[u8]) -> Option<Self> {
let (id, flags) = match id.into() {
embedded_hal_0_2::can::Id::Standard(id) => (id.as_raw() as u32, Flags::None),
embedded_hal_0_2::can::Id::Extended(id) => (id.as_raw(), Flags::Extended),
};
Self::new(id, flags.into(), data)
}
fn new_remote(id: impl Into<embedded_hal_0_2::can::Id>, dlc: usize) -> Option<Self> {
let (id, extended) = match id.into() {
embedded_hal_0_2::can::Id::Standard(id) => (id.as_raw() as u32, false),
embedded_hal_0_2::can::Id::Extended(id) => (id.as_raw(), true),
};
Self::new_remote(id, extended, dlc)
}
fn is_extended(&self) -> bool {
Frame::is_extended(self)
}
fn is_standard(&self) -> bool {
!self.is_extended()
}
fn is_remote_frame(&self) -> bool {
Frame::is_remote_frame(self)
}
fn is_data_frame(&self) -> bool {
!self.is_remote_frame()
}
fn id(&self) -> embedded_hal_0_2::can::Id {
if self.is_standard() {
let id = unsafe {
embedded_hal_0_2::can::StandardId::new_unchecked(self.identifier() as u16)
};
embedded_hal_0_2::can::Id::Standard(id)
} else {
let id = unsafe { embedded_hal_0_2::can::ExtendedId::new_unchecked(self.identifier()) };
embedded_hal_0_2::can::Id::Extended(id)
}
}
fn dlc(&self) -> usize {
Frame::dlc(self)
}
fn data(&self) -> &[u8] {
Frame::data(self)
}
}
impl embedded_can::Frame for Frame {
fn new(id: impl Into<embedded_can::Id>, data: &[u8]) -> Option<Self> {
let (id, flags) = match id.into() {
embedded_can::Id::Standard(id) => (id.as_raw() as u32, Flags::None),
embedded_can::Id::Extended(id) => (id.as_raw(), Flags::Extended),
};
Self::new(id, flags.into(), data)
}
fn new_remote(id: impl Into<embedded_can::Id>, dlc: usize) -> Option<Self> {
let (id, extended) = match id.into() {
embedded_can::Id::Standard(id) => (id.as_raw() as u32, false),
embedded_can::Id::Extended(id) => (id.as_raw(), true),
};
Self::new_remote(id, extended, dlc)
}
fn is_extended(&self) -> bool {
Frame::is_extended(self)
}
fn is_standard(&self) -> bool {
!self.is_extended()
}
fn is_remote_frame(&self) -> bool {
Frame::is_remote_frame(self)
}
fn is_data_frame(&self) -> bool {
!self.is_remote_frame()
}
fn id(&self) -> embedded_can::Id {
if self.is_standard() {
let id = unsafe { embedded_can::StandardId::new_unchecked(self.identifier() as u16) };
embedded_can::Id::Standard(id)
} else {
let id = unsafe { embedded_can::ExtendedId::new_unchecked(self.identifier()) };
embedded_can::Id::Extended(id)
}
}
fn dlc(&self) -> usize {
Frame::dlc(self)
}
fn data(&self) -> &[u8] {
Frame::data(self)
}
}
crate::impl_peripheral!(CAN);