use core::fmt; use spacepackets::{ ecss::{tc::IsPusTelecommand, PusPacket}, ByteConversionError, CcsdsPacket, }; use thiserror::Error; use crate::TargetId; pub type Apid = u16; #[derive(Debug, Error)] pub enum TargetIdCreationError { #[error("byte conversion")] ByteConversion(#[from] ByteConversionError), #[error("not enough app data to generate target ID")] NotEnoughAppData(usize), } #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)] pub struct TargetAndApidId { pub apid: Apid, pub target: u32, } impl TargetAndApidId { pub fn new(apid: Apid, target: u32) -> Self { Self { apid, target } } pub fn apid(&self) -> Apid { self.apid } pub fn target(&self) -> u32 { self.target } pub fn raw(&self) -> TargetId { ((self.apid as u64) << 32) | (self.target as u64) } pub fn target_id(&self) -> TargetId { self.raw() } pub fn from_pus_tc( tc: &(impl CcsdsPacket + PusPacket + IsPusTelecommand), ) -> Result { if tc.user_data().len() < 4 { return Err(ByteConversionError::FromSliceTooSmall { found: tc.user_data().len(), expected: 8, } .into()); } Ok(Self { apid: tc.apid(), target: u32::from_be_bytes(tc.user_data()[0..4].try_into().unwrap()), }) } } impl From for TargetAndApidId { fn from(raw: u64) -> Self { Self { apid: (raw >> 32) as u16, target: raw as u32, } } } impl From for u64 { fn from(target_and_apid_id: TargetAndApidId) -> Self { target_and_apid_id.raw() } } impl fmt::Display for TargetAndApidId { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "{}, {}", self.apid, self.target) } }