X-Git-Url: https://code.octet-stream.net/m17rt/blobdiff_plain/21b7d95e42fd374a33e0f18d3e3f406642e46322..HEAD:/m17app/src/app.rs diff --git a/m17app/src/app.rs b/m17app/src/app.rs index 4a7a345..561a2b7 100644 --- a/m17app/src/app.rs +++ b/m17app/src/app.rs @@ -1,7 +1,7 @@ use crate::adapter::{PacketAdapter, StreamAdapter}; use crate::tnc::Tnc; use m17core::kiss::{KissBuffer, KissCommand, KissFrame}; -use m17core::protocol::{EncryptionType, LsfFrame, PacketType}; +use m17core::protocol::{EncryptionType, LsfFrame, PacketType, StreamFrame}; use log::debug; use std::collections::HashMap; @@ -9,7 +9,7 @@ use std::sync::mpsc; use std::sync::{Arc, RwLock}; pub struct M17App { - listeners: Arc>, + adapters: Arc>, event_tx: mpsc::SyncSender, } @@ -17,45 +17,47 @@ impl M17App { pub fn new(mut tnc: T) -> Self { let write_tnc = tnc.try_clone().unwrap(); let (event_tx, event_rx) = mpsc::sync_channel(128); - let listeners = Arc::new(RwLock::new(Listeners::new())); + let listeners = Arc::new(RwLock::new(Adapters::new())); spawn_reader(tnc, listeners.clone()); spawn_writer(write_tnc, event_rx); Self { - listeners, + adapters: listeners, event_tx, } } - pub fn add_packet_listener(&self, listener: P) -> usize { - let mut listeners = self.listeners.write().unwrap(); - let id = listeners.next; - listeners.next += 1; - listeners.packet.insert(id, Arc::new(listener)); + pub fn add_packet_adapter(&self, adapter: P) -> usize { + let adapter = Arc::new(adapter); + let mut adapters = self.adapters.write().unwrap(); + let id = adapters.next; + adapters.next += 1; + adapters.packet.insert(id, adapter.clone()); + drop(adapters); + adapter.adapter_registered(id, self.tx()); id } - pub fn add_stream_listener(&self, listener: S) -> usize { - let mut listeners = self.listeners.write().unwrap(); - let id = listeners.next; - listeners.next += 1; - listeners.stream.insert(id, Arc::new(listener)); + pub fn add_stream_adapter(&self, adapter: S) -> usize { + let adapter = Arc::new(adapter); + let mut adapters = self.adapters.write().unwrap(); + let id = adapters.next; + adapters.next += 1; + adapters.stream.insert(id, adapter.clone()); + drop(adapters); + adapter.adapter_registered(id, self.tx()); id } - pub fn remove_packet_listener(&self, id: usize) { - self.listeners.write().unwrap().packet.remove(&id); - } - - pub fn remove_stream_listener(&self, id: usize) { - self.listeners.write().unwrap().stream.remove(&id); + pub fn remove_packet_adapter(&self, id: usize) { + if let Some(a) = self.adapters.write().unwrap().packet.remove(&id) { + a.adapter_removed(); + } } - pub fn transmit_packet(&self, type_code: PacketType, payload: &[u8]) { - // hang on where do we get the LSF details from? We need a destination obviously - // our source address needs to be configured here too - // also there is possible CAN, encryption, meta payload - - // we will immediately convert this into a KISS payload before sending into channel so we only need borrow on data + pub fn remove_stream_adapter(&self, id: usize) { + if let Some(a) = self.adapters.write().unwrap().stream.remove(&id) { + a.adapter_removed(); + } } /// Create a handle that can be used to transmit data on the TNC @@ -79,26 +81,42 @@ pub struct TxHandle { } impl TxHandle { + pub fn transmit_packet(&self, packet_type: PacketType, payload: &[u8]) { + // hang on where do we get the LSF details from? We need a destination obviously + // our source address needs to be configured here too + // also there is possible CAN, encryption, meta payload + + // we will immediately convert this into a KISS payload before sending into channel so we only need borrow on data + } + // add more methods here for stream outgoing - pub fn transmit_stream_start(&self /* lsf?, payload? what needs to be configured ?! */) {} + pub fn transmit_stream_start(&self, lsf: LsfFrame) { + // TODO: is asking for an LsfFrame a good idea or unfriendly API? + // What I should do here is create a LinkSetup struct which wraps an LsfFrame and can be loaded with a raw one + let kiss_frame = KissFrame::new_stream_setup(&lsf.0).unwrap(); + let _ = self.event_tx.send(TncControlEvent::Kiss(kiss_frame)); + } // as long as there is only one TNC it is implied there is only ever one stream transmission in flight - pub fn transmit_stream_next(&self, /* next payload, */ end_of_stream: bool) {} + pub fn transmit_stream_next(&self, stream: StreamFrame) { + let kiss_frame = KissFrame::new_stream_data(&stream).unwrap(); + let _ = self.event_tx.send(TncControlEvent::Kiss(kiss_frame)); + } } /// Synchronised structure for listeners subscribing to packets and streams. /// /// Each listener will be notified in turn of each event. -struct Listeners { +struct Adapters { /// Identifier to be assigned to the next listener, starting from 0 next: usize, packet: HashMap>, stream: HashMap>, } -impl Listeners { +impl Adapters { fn new() -> Self { Self { next: 0, @@ -115,9 +133,10 @@ enum TncControlEvent { Close, } -fn spawn_reader(mut tnc: T, listeners: Arc>) { +fn spawn_reader(mut tnc: T, adapters: Arc>) { std::thread::spawn(move || { let mut kiss_buffer = KissBuffer::new(); + let mut stream_running = false; loop { let mut buf = kiss_buffer.buf_remaining(); let n = match tnc.read(&mut buf) { @@ -145,7 +164,7 @@ fn spawn_reader(mut tnc: T, listeners: Arc(mut tnc: T, listeners: Arc = - listeners.read().unwrap().packet.values().cloned().collect(); + adapters.read().unwrap().packet.values().cloned().collect(); for s in subs { s.packet_received( lsf.clone(), @@ -181,7 +200,47 @@ fn spawn_reader(mut tnc: T, listeners: Arc { - // handle stream and send it to subscribers + let mut payload = [0u8; 32]; + let Ok(n) = frame.decode_payload(&mut payload) else { + debug!("failed to decode stream payload from KISS frame"); + continue; + }; + if n == 30 { + let lsf = LsfFrame(payload[0..30].try_into().unwrap()); + if lsf.check_crc() != 0 { + debug!("initial LSF in stream did not pass CRC"); + continue; + } + stream_running = true; + let subs: Vec<_> = + adapters.read().unwrap().stream.values().cloned().collect(); + for s in subs { + s.stream_began(lsf.clone()); + } + } else if n == 26 { + if !stream_running { + debug!("ignoring stream data as we didn't get a valid LSF first"); + continue; + } + // TODO: parse LICH and handle the different changing subvalues META could have + if m17core::crc::m17_crc(&payload[6..n]) != 0 { + debug!("stream data CRC mismatch"); + continue; + } + let mut frame_number = u16::from_be_bytes([payload[6], payload[7]]); + let is_final = (frame_number & 0x8000) > 0; + frame_number &= 0x7fff; + let data: [u8; 16] = payload[8..24].try_into().unwrap(); + let data = Arc::new(data); + if is_final { + stream_running = false; + } + let subs: Vec<_> = + adapters.read().unwrap().stream.values().cloned().collect(); + for s in subs { + s.stream_data(frame_number, is_final, data.clone()); + } + } } _ => (), } @@ -190,7 +249,7 @@ fn spawn_reader(mut tnc: T, listeners: Arc(mut tnc: T, event_rx: mpsc::Receiver) { +fn spawn_writer(mut tnc: T, event_rx: mpsc::Receiver) { std::thread::spawn(move || { while let Ok(ev) = event_rx.recv() { match ev {