Use Result instead of Option for Frame creation.

This commit is contained in:
Corey Schuhen 2024-03-12 21:19:06 +10:00
parent 12a3af5043
commit 242759a600
9 changed files with 78 additions and 100 deletions

View file

@ -44,7 +44,6 @@ pub type Data = crate::can::frame::ClassicData;
pub type Frame = crate::can::frame::ClassicFrame;
use crate::can::bx::filter::MasterFilters;
use crate::can::frame::ClassicData;
/// A bxCAN peripheral instance.
///
@ -799,10 +798,7 @@ where
data[4..8].copy_from_slice(&mb.tdhr().read().0.to_ne_bytes());
let len = mb.tdtr().read().dlc();
Some(Frame::new(
Header::new(id, len, false),
ClassicData::new(&data).unwrap(),
))
Some(Frame::new(Header::new(id, len, false), &data).unwrap())
} else {
// Abort request failed because the frame was already sent (or being sent) on
// the bus. All mailboxes are now free. This can happen for small prescaler
@ -944,10 +940,7 @@ fn receive_fifo(canregs: crate::pac::can::Can, fifo_nr: usize) -> nb::Result<Fra
// Release the mailbox.
rfr.write(|w| w.set_rfom(true));
Ok(Frame::new(
Header::new(id, len, false),
ClassicData::new(&data).unwrap(),
))
Ok(Frame::new(Header::new(id, len, false), &data).unwrap())
}
/// Identifies one of the two receive FIFOs.

View file

@ -274,7 +274,7 @@ impl<'d, T: Instance> Can<'d, T> {
data[0..4].copy_from_slice(&fifo.rdlr().read().0.to_ne_bytes());
data[4..8].copy_from_slice(&fifo.rdhr().read().0.to_ne_bytes());
let frame = Frame::new(Header::new(id, data_len, false), Data::new(&data).unwrap());
let frame = Frame::new(Header::new(id, data_len, false), &data).unwrap();
let envelope = Envelope {
#[cfg(feature = "time")]
ts,

View file

@ -28,3 +28,15 @@ pub enum BusError {
/// At least one of error counter has reached the Error_Warning limit of 96.
BusWarning,
}
/// Frame Create Errors
#[derive(Debug)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
pub enum FrameCreateError {
/// Data in header does not match supplied.
NotEnoughData,
/// Invalid data length not 0-8 for Classic packet or valid for FD.
InvalidDataLength,
/// Invalid ID.
InvalidCanId,
}

View file

@ -56,7 +56,10 @@ impl Registers {
match maybe_header {
Some((header, ts)) => {
let data = &buffer[0..header.len() as usize];
Some((F::from_header(header, data)?, ts))
match F::from_header(header, data) {
Ok(frame) => Some((frame, ts)),
Err(_) => None,
}
}
None => None,
}

View file

@ -1,6 +1,8 @@
//! Definition for CAN Frames
use bit_field::BitField;
use crate::can::enums::FrameCreateError;
/// CAN Header, without meta data
#[derive(Debug, Copy, Clone)]
pub struct Header {
@ -82,7 +84,7 @@ impl Header {
/// and to retrieve the Header from a frame
pub trait CanHeader: Sized {
/// Construct frame from header and payload
fn from_header(header: Header, data: &[u8]) -> Option<Self>;
fn from_header(header: Header, data: &[u8]) -> Result<Self, FrameCreateError>;
/// Get this frame's header struct
fn header(&self) -> &Header;
@ -103,15 +105,15 @@ impl ClassicData {
///
/// Returns `None` if `data` is more than 64 bytes (which is the maximum) or
/// cannot be represented with an FDCAN DLC.
pub fn new(data: &[u8]) -> Option<Self> {
if !FdData::is_valid_len(data.len()) {
return None;
pub fn new(data: &[u8]) -> Result<Self, FrameCreateError> {
if data.len() > 8 {
return Err(FrameCreateError::InvalidDataLength);
}
let mut bytes = [0; 8];
bytes[..data.len()].copy_from_slice(data);
Some(Self { bytes })
Ok(Self { bytes })
}
/// Raw read access to data.
@ -134,12 +136,6 @@ impl ClassicData {
}
}
impl From<&[u8]> for ClassicData {
fn from(d: &[u8]) -> Self {
ClassicData::new(d).unwrap()
}
}
/// Frame with up to 8 bytes of data payload as per Classic CAN
#[derive(Debug, Copy, Clone)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
@ -150,59 +146,42 @@ pub struct ClassicFrame {
impl ClassicFrame {
/// Create a new CAN classic Frame
pub fn new(can_header: Header, data: impl Into<ClassicData>) -> ClassicFrame {
ClassicFrame {
can_header,
data: data.into(),
}
pub fn new(can_header: Header, raw_data: &[u8]) -> Result<Self, FrameCreateError> {
let data = ClassicData::new(raw_data)?;
Ok(ClassicFrame { can_header, data: data })
}
/// Creates a new data frame.
pub fn new_data(id: impl Into<embedded_can::Id>, data: &[u8]) -> Self {
pub fn new_data(id: impl Into<embedded_can::Id>, data: &[u8]) -> Result<Self, FrameCreateError> {
let eid: embedded_can::Id = id.into();
let header = Header::new(eid, data.len() as u8, false);
Self::new(header, data)
}
/// Create new extended frame
pub fn new_extended(raw_id: u32, raw_data: &[u8]) -> Option<Self> {
pub fn new_extended(raw_id: u32, raw_data: &[u8]) -> Result<Self, FrameCreateError> {
if let Some(id) = embedded_can::ExtendedId::new(raw_id) {
match ClassicData::new(raw_data) {
Some(data) => Some(ClassicFrame::new(
Header::new(id.into(), raw_data.len() as u8, false),
data,
)),
None => None,
}
Self::new(Header::new(id.into(), raw_data.len() as u8, false), raw_data)
} else {
None
Err(FrameCreateError::InvalidCanId)
}
}
/// Create new standard frame
pub fn new_standard(raw_id: u16, raw_data: &[u8]) -> Option<Self> {
pub fn new_standard(raw_id: u16, raw_data: &[u8]) -> Result<Self, FrameCreateError> {
if let Some(id) = embedded_can::StandardId::new(raw_id) {
match ClassicData::new(raw_data) {
Some(data) => Some(ClassicFrame::new(
Header::new(id.into(), raw_data.len() as u8, false),
data,
)),
None => None,
}
Self::new(Header::new(id.into(), raw_data.len() as u8, false), raw_data)
} else {
None
Err(FrameCreateError::InvalidCanId)
}
}
/// Create new remote frame
pub fn new_remote(id: impl Into<embedded_can::Id>, len: usize) -> Option<Self> {
pub fn new_remote(id: impl Into<embedded_can::Id>, len: usize) -> Result<Self, FrameCreateError> {
if len <= 8usize {
Some(ClassicFrame::new(
Header::new(id.into(), len as u8, true),
ClassicData::empty(),
))
Self::new(Header::new(id.into(), len as u8, true), &[0; 8])
} else {
None
Err(FrameCreateError::InvalidDataLength)
}
}
@ -229,20 +208,19 @@ impl ClassicFrame {
impl embedded_can::Frame for ClassicFrame {
fn new(id: impl Into<embedded_can::Id>, raw_data: &[u8]) -> Option<Self> {
match ClassicData::new(raw_data) {
Some(data) => Some(ClassicFrame::new(
Header::new(id.into(), raw_data.len() as u8, false),
data,
)),
None => None,
let frameopt = ClassicFrame::new(Header::new(id.into(), raw_data.len() as u8, false), raw_data);
match frameopt {
Ok(frame) => Some(frame),
Err(_) => None,
}
}
fn new_remote(id: impl Into<embedded_can::Id>, len: usize) -> Option<Self> {
if len <= 8 {
Some(ClassicFrame::new(
Header::new(id.into(), len as u8, true),
ClassicData::empty(),
))
let frameopt = ClassicFrame::new(Header::new(id.into(), len as u8, true), &[0; 8]);
match frameopt {
Ok(frame) => Some(frame),
Err(_) => None,
}
} else {
None
}
@ -268,8 +246,8 @@ impl embedded_can::Frame for ClassicFrame {
}
impl CanHeader for ClassicFrame {
fn from_header(header: Header, data: &[u8]) -> Option<Self> {
Some(Self::new(header, ClassicData::new(data)?))
fn from_header(header: Header, data: &[u8]) -> Result<Self, FrameCreateError> {
Self::new(header, data)
}
fn header(&self) -> &Header {
@ -290,15 +268,15 @@ impl FdData {
///
/// Returns `None` if `data` is more than 64 bytes (which is the maximum) or
/// cannot be represented with an FDCAN DLC.
pub fn new(data: &[u8]) -> Option<Self> {
pub fn new(data: &[u8]) -> Result<Self, FrameCreateError> {
if !FdData::is_valid_len(data.len()) {
return None;
return Err(FrameCreateError::InvalidDataLength);
}
let mut bytes = [0; 64];
bytes[..data.len()].copy_from_slice(data);
Some(Self { bytes })
Ok(Self { bytes })
}
/// Raw read access to data.
@ -337,40 +315,35 @@ pub struct FdFrame {
impl FdFrame {
/// Create a new CAN classic Frame
pub fn new(can_header: Header, data: FdData) -> FdFrame {
FdFrame { can_header, data }
pub fn new(can_header: Header, raw_data: &[u8]) -> Result<Self, FrameCreateError> {
let data = FdData::new(raw_data)?;
Ok(FdFrame { can_header, data })
}
/// Create new extended frame
pub fn new_extended(raw_id: u32, raw_data: &[u8]) -> Option<Self> {
pub fn new_extended(raw_id: u32, raw_data: &[u8]) -> Result<Self, FrameCreateError> {
if let Some(id) = embedded_can::ExtendedId::new(raw_id) {
match FdData::new(raw_data) {
Some(data) => Some(FdFrame::new(Header::new(id.into(), raw_data.len() as u8, false), data)),
None => None,
}
Self::new(Header::new(id.into(), raw_data.len() as u8, false), raw_data)
} else {
None
Err(FrameCreateError::InvalidCanId)
}
}
/// Create new standard frame
pub fn new_standard(raw_id: u16, raw_data: &[u8]) -> Option<Self> {
pub fn new_standard(raw_id: u16, raw_data: &[u8]) -> Result<Self, FrameCreateError> {
if let Some(id) = embedded_can::StandardId::new(raw_id) {
match FdData::new(raw_data) {
Some(data) => Some(FdFrame::new(Header::new(id.into(), raw_data.len() as u8, false), data)),
None => None,
}
Self::new(Header::new(id.into(), raw_data.len() as u8, false), raw_data)
} else {
None
Err(FrameCreateError::InvalidCanId)
}
}
/// Create new remote frame
pub fn new_remote(id: impl Into<embedded_can::Id>, len: usize) -> Option<Self> {
pub fn new_remote(id: impl Into<embedded_can::Id>, len: usize) -> Result<Self, FrameCreateError> {
if len <= 8 {
Some(FdFrame::new(Header::new(id.into(), len as u8, true), FdData::empty()))
Self::new(Header::new(id.into(), len as u8, true), &[0; 8])
} else {
None
Err(FrameCreateError::InvalidDataLength)
}
}
@ -392,20 +365,17 @@ impl FdFrame {
impl embedded_can::Frame for FdFrame {
fn new(id: impl Into<embedded_can::Id>, raw_data: &[u8]) -> Option<Self> {
match FdData::new(raw_data) {
Some(data) => Some(FdFrame::new(
Header::new_fd(id.into(), raw_data.len() as u8, false, true),
data,
)),
None => None,
match FdFrame::new(Header::new_fd(id.into(), raw_data.len() as u8, false, true), raw_data) {
Ok(frame) => Some(frame),
Err(_) => None,
}
}
fn new_remote(id: impl Into<embedded_can::Id>, len: usize) -> Option<Self> {
if len <= 8 {
Some(FdFrame::new(
Header::new_fd(id.into(), len as u8, true, true),
FdData::empty(),
))
match FdFrame::new(Header::new_fd(id.into(), len as u8, true, true), &[0; 64]) {
Ok(frame) => Some(frame),
Err(_) => None,
}
} else {
None
}
@ -432,8 +402,8 @@ impl embedded_can::Frame for FdFrame {
}
impl CanHeader for FdFrame {
fn from_header(header: Header, data: &[u8]) -> Option<Self> {
Some(Self::new(header, FdData::new(data)?))
fn from_header(header: Header, data: &[u8]) -> Result<Self, FrameCreateError> {
Self::new(header, data)
}
fn header(&self) -> &Header {

View file

@ -46,7 +46,7 @@ async fn main(_spawner: Spawner) {
let mut i: u8 = 0;
loop {
let tx_frame = Frame::new_data(unwrap!(StandardId::new(i as _)), &[i, 0, 1, 2, 3, 4, 5, 6]);
let tx_frame = Frame::new_data(unwrap!(StandardId::new(i as _)), &[i, 0, 1, 2, 3, 4, 5, 6]).unwrap();
can.write(&tx_frame).await;
match can.read().await {

View file

@ -51,7 +51,7 @@ async fn main(_spawner: Spawner) {
let mut i: u8 = 0;
loop {
let tx_frame = Frame::new_data(unwrap!(StandardId::new(i as _)), &[i]);
let tx_frame = Frame::new_data(unwrap!(StandardId::new(i as _)), &[i]).unwrap();
let tx_ts = Instant::now();
can.write(&tx_frame).await;

View file

@ -26,7 +26,7 @@ bind_interrupts!(struct Irqs {
#[embassy_executor::task]
pub async fn send_can_message(tx: &'static mut CanTx<'static, CAN3>) {
loop {
let frame = Frame::new_data(unwrap!(StandardId::new(0 as _)), &[0]);
let frame = Frame::new_data(unwrap!(StandardId::new(0 as _)), &[0]).unwrap();
tx.write(&frame).await;
embassy_time::Timer::after_secs(1).await;
}

View file

@ -60,7 +60,7 @@ async fn main(_spawner: Spawner) {
let mut i: u8 = 0;
loop {
let tx_frame = Frame::new_data(unwrap!(StandardId::new(i as _)), &[i]);
let tx_frame = Frame::new_data(unwrap!(StandardId::new(i as _)), &[i]).unwrap();
info!("Transmitting frame...");
let tx_ts = Instant::now();