Skip to main content

rns_net/interface/
serial_iface.rs

1//! Serial interface with HDLC framing.
2//!
3//! Matches Python `SerialInterface.py` — opens a serial port,
4//! reads with HDLC framing, reconnects on failure.
5
6use std::io::{self, Read, Write};
7use std::thread;
8use std::time::Duration;
9
10use rns_core::transport::types::InterfaceId;
11
12use crate::event::{Event, EventSender};
13use crate::hdlc;
14use crate::interface::Writer;
15use crate::serial::{Parity, SerialConfig, SerialPort};
16
17/// Configuration for a Serial interface.
18#[derive(Debug, Clone)]
19pub struct SerialIfaceConfig {
20    pub name: String,
21    pub port: String,
22    pub speed: u32,
23    pub data_bits: u8,
24    pub parity: Parity,
25    pub stop_bits: u8,
26    pub interface_id: InterfaceId,
27}
28
29impl Default for SerialIfaceConfig {
30    fn default() -> Self {
31        SerialIfaceConfig {
32            name: String::new(),
33            port: String::new(),
34            speed: 9600,
35            data_bits: 8,
36            parity: Parity::None,
37            stop_bits: 1,
38            interface_id: InterfaceId(0),
39        }
40    }
41}
42
43/// Writer that sends HDLC-framed data over a serial port.
44struct SerialWriter {
45    file: std::fs::File,
46}
47
48impl Writer for SerialWriter {
49    fn send_frame(&mut self, data: &[u8]) -> io::Result<()> {
50        self.file.write_all(&hdlc::frame(data))
51    }
52}
53
54/// Start the serial interface. Opens the port, spawns reader thread.
55/// Returns the writer for the driver.
56pub fn start(config: SerialIfaceConfig, tx: EventSender) -> io::Result<Box<dyn Writer>> {
57    let serial_config = SerialConfig {
58        path: config.port.clone(),
59        baud: config.speed,
60        data_bits: config.data_bits,
61        parity: config.parity,
62        stop_bits: config.stop_bits,
63    };
64
65    let port = SerialPort::open(&serial_config)?;
66    let reader_file = port.reader()?;
67    let writer_file = port.writer()?;
68
69    let id = config.interface_id;
70
71    // Signal interface up
72    let _ = tx.send(Event::InterfaceUp(id, None, None));
73
74    // Short delay matching Python's configure_device sleep
75    thread::sleep(Duration::from_millis(500));
76
77    // Spawn reader thread
78    thread::Builder::new()
79        .name(format!("serial-reader-{}", id.0))
80        .spawn(move || {
81            reader_loop(reader_file, id, config, tx);
82        })?;
83
84    Ok(Box::new(SerialWriter { file: writer_file }))
85}
86
87/// Reader thread: reads from serial, HDLC-decodes, sends frames to driver.
88fn reader_loop(
89    mut reader: std::fs::File,
90    id: InterfaceId,
91    config: SerialIfaceConfig,
92    tx: EventSender,
93) {
94    let mut decoder = hdlc::Decoder::new();
95    let mut buf = [0u8; 4096];
96
97    loop {
98        match reader.read(&mut buf) {
99            Ok(0) => {
100                // EOF — port closed
101                log::warn!("[{}] serial port closed", config.name);
102                let _ = tx.send(Event::InterfaceDown(id));
103                match reconnect(&config, &tx) {
104                    Some(new_reader) => {
105                        reader = new_reader;
106                        decoder = hdlc::Decoder::new();
107                        continue;
108                    }
109                    None => return,
110                }
111            }
112            Ok(n) => {
113                for frame in decoder.feed(&buf[..n]) {
114                    if tx
115                        .send(Event::Frame {
116                            interface_id: id,
117                            data: frame,
118                            rssi: None,
119                            snr: None,
120                        })
121                        .is_err()
122                    {
123                        return; // driver shut down
124                    }
125                }
126            }
127            Err(e) => {
128                log::warn!("[{}] serial read error: {}", config.name, e);
129                let _ = tx.send(Event::InterfaceDown(id));
130                match reconnect(&config, &tx) {
131                    Some(new_reader) => {
132                        reader = new_reader;
133                        decoder = hdlc::Decoder::new();
134                        continue;
135                    }
136                    None => return,
137                }
138            }
139        }
140    }
141}
142
143/// Attempt to reconnect the serial port. Returns new reader file on success.
144fn reconnect(config: &SerialIfaceConfig, tx: &EventSender) -> Option<std::fs::File> {
145    loop {
146        thread::sleep(Duration::from_secs(5));
147        log::info!(
148            "[{}] attempting to reconnect serial port {}...",
149            config.name,
150            config.port
151        );
152
153        let serial_config = SerialConfig {
154            path: config.port.clone(),
155            baud: config.speed,
156            data_bits: config.data_bits,
157            parity: config.parity,
158            stop_bits: config.stop_bits,
159        };
160
161        match SerialPort::open(&serial_config) {
162            Ok(port) => match (port.reader(), port.writer()) {
163                (Ok(reader), Ok(writer_file)) => {
164                    log::info!("[{}] serial port reconnected", config.name);
165                    let new_writer: Box<dyn Writer> = Box::new(SerialWriter { file: writer_file });
166                    let _ = tx.send(Event::InterfaceUp(
167                        config.interface_id,
168                        Some(new_writer),
169                        None,
170                    ));
171                    thread::sleep(Duration::from_millis(500));
172                    return Some(reader);
173                }
174                _ => {
175                    log::warn!(
176                        "[{}] failed to get reader/writer from serial port",
177                        config.name
178                    );
179                }
180            },
181            Err(e) => {
182                log::warn!("[{}] serial reconnect failed: {}", config.name, e);
183            }
184        }
185    }
186}
187
188// --- Factory implementation ---
189
190use super::{InterfaceConfigData, InterfaceFactory, StartContext, StartResult};
191use rns_core::transport::types::InterfaceInfo;
192use std::collections::HashMap;
193
194/// Factory for `SerialInterface`.
195pub struct SerialFactory;
196
197impl InterfaceFactory for SerialFactory {
198    fn type_name(&self) -> &str {
199        "SerialInterface"
200    }
201
202    fn default_ifac_size(&self) -> usize {
203        8
204    }
205
206    fn parse_config(
207        &self,
208        name: &str,
209        id: InterfaceId,
210        params: &HashMap<String, String>,
211    ) -> Result<Box<dyn InterfaceConfigData>, String> {
212        let port = params
213            .get("port")
214            .cloned()
215            .ok_or_else(|| "SerialInterface requires 'port'".to_string())?;
216
217        let speed = params
218            .get("speed")
219            .and_then(|v| v.parse().ok())
220            .unwrap_or(9600u32);
221
222        let data_bits = params
223            .get("databits")
224            .and_then(|v| v.parse().ok())
225            .unwrap_or(8u8);
226
227        let parity = params
228            .get("parity")
229            .map(|v| match v.to_lowercase().as_str() {
230                "e" | "even" => crate::serial::Parity::Even,
231                "o" | "odd" => crate::serial::Parity::Odd,
232                _ => crate::serial::Parity::None,
233            })
234            .unwrap_or(crate::serial::Parity::None);
235
236        let stop_bits = params
237            .get("stopbits")
238            .and_then(|v| v.parse().ok())
239            .unwrap_or(1u8);
240
241        Ok(Box::new(SerialIfaceConfig {
242            name: name.to_string(),
243            port,
244            speed,
245            data_bits,
246            parity,
247            stop_bits,
248            interface_id: id,
249        }))
250    }
251
252    fn start(
253        &self,
254        config: Box<dyn InterfaceConfigData>,
255        ctx: StartContext,
256    ) -> std::io::Result<StartResult> {
257        let serial_config = *config
258            .into_any()
259            .downcast::<SerialIfaceConfig>()
260            .map_err(|_| {
261                std::io::Error::new(std::io::ErrorKind::InvalidData, "wrong config type")
262            })?;
263
264        let id = serial_config.interface_id;
265        let name = serial_config.name.clone();
266        let bitrate = Some(serial_config.speed as u64);
267
268        let info = InterfaceInfo {
269            id,
270            name,
271            mode: ctx.mode,
272            out_capable: true,
273            in_capable: true,
274            bitrate,
275            airtime_profile: None,
276            announce_rate_target: None,
277            announce_rate_grace: 0,
278            announce_rate_penalty: 0.0,
279            announce_cap: rns_core::constants::ANNOUNCE_CAP,
280            is_local_client: false,
281            wants_tunnel: false,
282            tunnel_id: None,
283            mtu: rns_core::constants::MTU as u32,
284            ingress_control: rns_core::transport::types::IngressControlConfig::disabled(),
285            ia_freq: 0.0,
286            ip_freq: 0.0,
287            op_freq: 0.0,
288            op_samples: 0,
289            started: crate::time::now(),
290        };
291
292        let writer = start(serial_config, ctx.tx)?;
293
294        Ok(StartResult::Simple {
295            id,
296            info,
297            writer,
298            interface_type_name: "SerialInterface".to_string(),
299        })
300    }
301}
302
303#[cfg(test)]
304mod tests {
305    use super::*;
306    use crate::serial::open_pty_pair;
307    use std::os::unix::io::{AsRawFd, FromRawFd};
308    use std::sync::mpsc;
309    use std::time::Duration;
310
311    /// Helper: poll an fd for reading with timeout (ms). Returns true if data available.
312    fn poll_read(fd: i32, timeout_ms: i32) -> bool {
313        let mut pfd = libc::pollfd {
314            fd,
315            events: libc::POLLIN,
316            revents: 0,
317        };
318        let ret = unsafe { libc::poll(&mut pfd, 1, timeout_ms) };
319        ret > 0
320    }
321
322    #[test]
323    fn serial_hdlc_roundtrip() {
324        let (master_fd, slave_fd) = open_pty_pair().unwrap();
325        let mut master_file = unsafe { std::fs::File::from_raw_fd(master_fd) };
326        let mut slave_file = unsafe { std::fs::File::from_raw_fd(slave_fd) };
327
328        // Write an HDLC frame to the master side
329        let payload: Vec<u8> = (0..32).collect();
330        let framed = hdlc::frame(&payload);
331        master_file.write_all(&framed).unwrap();
332        master_file.flush().unwrap();
333
334        // Read from slave using HDLC decoder
335        assert!(
336            poll_read(slave_file.as_raw_fd(), 2000),
337            "should have data available"
338        );
339
340        let mut decoder = hdlc::Decoder::new();
341        let mut buf = [0u8; 4096];
342        let n = slave_file.read(&mut buf).unwrap();
343        let frames = decoder.feed(&buf[..n]);
344        assert_eq!(frames.len(), 1);
345        assert_eq!(frames[0], payload);
346    }
347
348    #[test]
349    fn serial_writer_frames() {
350        let (master_fd, slave_fd) = open_pty_pair().unwrap();
351
352        let writer_file = unsafe { std::fs::File::from_raw_fd(slave_fd) };
353        let mut writer = SerialWriter { file: writer_file };
354
355        let payload = vec![0x01, 0x02, 0x7E, 0x7D, 0x03]; // includes bytes that need HDLC escaping
356        writer.send_frame(&payload).unwrap();
357
358        // Read from master
359        let mut master_file = unsafe { std::fs::File::from_raw_fd(master_fd) };
360        assert!(poll_read(master_file.as_raw_fd(), 2000), "should have data");
361
362        let expected = hdlc::frame(&payload);
363        let mut buf = [0u8; 256];
364        let n = master_file.read(&mut buf).unwrap();
365        assert_eq!(&buf[..n], &expected[..]);
366    }
367
368    #[test]
369    fn serial_fragmented_read() {
370        let (master_fd, slave_fd) = open_pty_pair().unwrap();
371        let mut master_file = unsafe { std::fs::File::from_raw_fd(master_fd) };
372        let slave_file = unsafe { std::fs::File::from_raw_fd(slave_fd) };
373
374        let (tx, rx) = mpsc::channel::<Event>();
375
376        // Write a frame in two parts to the master
377        let payload: Vec<u8> = (0..32).collect();
378        let framed = hdlc::frame(&payload);
379        let mid = framed.len() / 2;
380
381        // Spawn reader thread first (it will block waiting for data)
382        let reader_thread = thread::spawn(move || {
383            let mut reader = slave_file;
384            let mut decoder = hdlc::Decoder::new();
385            let mut buf = [0u8; 4096];
386
387            loop {
388                match reader.read(&mut buf) {
389                    Ok(n) if n > 0 => {
390                        if let Some(frame) = decoder.feed(&buf[..n]).into_iter().next() {
391                            let _ = tx.send(Event::Frame {
392                                interface_id: InterfaceId(0),
393                                data: frame,
394                                rssi: None,
395                                snr: None,
396                            });
397                            return;
398                        }
399                    }
400                    _ => return,
401                }
402            }
403        });
404
405        // Write first half
406        master_file.write_all(&framed[..mid]).unwrap();
407        master_file.flush().unwrap();
408
409        // Small delay then write second half
410        thread::sleep(Duration::from_millis(50));
411        master_file.write_all(&framed[mid..]).unwrap();
412        master_file.flush().unwrap();
413
414        let event = rx.recv_timeout(Duration::from_secs(2)).unwrap();
415        match event {
416            Event::Frame { data, .. } => assert_eq!(data, payload),
417            other => panic!("expected Frame, got {:?}", other),
418        }
419
420        let _ = reader_thread.join();
421    }
422
423    #[test]
424    fn serial_reconnect_on_close() {
425        let (master_fd, slave_fd) = open_pty_pair().unwrap();
426
427        let (tx, rx) = mpsc::channel::<Event>();
428        let id = InterfaceId(42);
429
430        // Spawn a reader thread on the slave side
431        let slave_file = unsafe { std::fs::File::from_raw_fd(slave_fd) };
432        let reader_thread = thread::spawn(move || {
433            let mut reader = slave_file;
434            let mut buf = [0u8; 4096];
435            let mut decoder = hdlc::Decoder::new();
436            loop {
437                match reader.read(&mut buf) {
438                    Ok(0) => {
439                        let _ = tx.send(Event::InterfaceDown(id));
440                        return;
441                    }
442                    Ok(n) => {
443                        for frame in decoder.feed(&buf[..n]) {
444                            let _ = tx.send(Event::Frame {
445                                interface_id: id,
446                                data: frame,
447                                rssi: None,
448                                snr: None,
449                            });
450                        }
451                    }
452                    Err(_) => {
453                        let _ = tx.send(Event::InterfaceDown(id));
454                        return;
455                    }
456                }
457            }
458        });
459
460        // Close the master fd — this should cause the reader to get EOF
461        unsafe { libc::close(master_fd) };
462
463        let event = rx.recv_timeout(Duration::from_secs(2)).unwrap();
464        assert!(matches!(event, Event::InterfaceDown(InterfaceId(42))));
465
466        let _ = reader_thread.join();
467    }
468}