Skip to content
Snippets Groups Projects
Commit c301a632 authored by Stephen D's avatar Stephen D
Browse files

Merge branch 'spi2' into 'master'

Spi

See merge request !4
parents 98cd6ac2 0befd735
No related branches found
No related tags found
1 merge request!4Spi
...@@ -40,8 +40,8 @@ dependencies = [ ...@@ -40,8 +40,8 @@ dependencies = [
"crc", "crc",
"image", "image",
"kiss-tnc", "kiss-tnc",
"rppal",
"serde", "serde",
"serial",
"sha3", "sha3",
"subprocess", "subprocess",
"tokio", "tokio",
...@@ -383,15 +383,6 @@ dependencies = [ ...@@ -383,15 +383,6 @@ dependencies = [
"hashbrown", "hashbrown",
] ]
[[package]]
name = "ioctl-rs"
version = "0.1.6"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f7970510895cee30b3e9128319f2cefd4bde883a39f38baa279567ba3a7eb97d"
dependencies = [
"libc",
]
[[package]] [[package]]
name = "jpeg-decoder" name = "jpeg-decoder"
version = "0.3.0" version = "0.3.0"
...@@ -687,6 +678,15 @@ dependencies = [ ...@@ -687,6 +678,15 @@ dependencies = [
"bitflags", "bitflags",
] ]
[[package]]
name = "rppal"
version = "0.14.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "612e1a22e21f08a246657c6433fe52b773ae43d07c9ef88ccfc433cc8683caba"
dependencies = [
"libc",
]
[[package]] [[package]]
name = "scopeguard" name = "scopeguard"
version = "1.1.0" version = "1.1.0"
...@@ -722,48 +722,6 @@ dependencies = [ ...@@ -722,48 +722,6 @@ dependencies = [
"serde", "serde",
] ]
[[package]]
name = "serial"
version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a1237a96570fc377c13baa1b88c7589ab66edced652e43ffb17088f003db3e86"
dependencies = [
"serial-core",
"serial-unix",
"serial-windows",
]
[[package]]
name = "serial-core"
version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3f46209b345401737ae2125fe5b19a77acce90cd53e1658cda928e4fe9a64581"
dependencies = [
"libc",
]
[[package]]
name = "serial-unix"
version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f03fbca4c9d866e24a459cbca71283f545a37f8e3e002ad8c70593871453cab7"
dependencies = [
"ioctl-rs",
"libc",
"serial-core",
"termios",
]
[[package]]
name = "serial-windows"
version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "15c6d3b776267a75d31bbdfd5d36c0ca051251caafc285827052bc53bcdc8162"
dependencies = [
"libc",
"serial-core",
]
[[package]] [[package]]
name = "sha3" name = "sha3"
version = "0.10.8" version = "0.10.8"
...@@ -841,15 +799,6 @@ version = "1.0.1" ...@@ -841,15 +799,6 @@ version = "1.0.1"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "55937e1799185b12863d447f42597ed69d9928686b8d88a1df17376a097d8369" checksum = "55937e1799185b12863d447f42597ed69d9928686b8d88a1df17376a097d8369"
[[package]]
name = "termios"
version = "0.2.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d5d9cf598a6d7ce700a4e6a9199da127e6819a61e64b68609683cc9a01b5683a"
dependencies = [
"libc",
]
[[package]] [[package]]
name = "thiserror" name = "thiserror"
version = "1.0.40" version = "1.0.40"
......
...@@ -23,10 +23,9 @@ bitvec = "1.0.1" ...@@ -23,10 +23,9 @@ bitvec = "1.0.1"
crc = "3.0.1" crc = "3.0.1"
image = "0.24.6" image = "0.24.6"
kiss-tnc = "0.2.2" kiss-tnc = "0.2.2"
rppal = "0.14.1"
serde = { version = "1.0.160", features = ["derive"] } serde = { version = "1.0.160", features = ["derive"] }
serial = "0.4.0"
sha3 = "0.10.7" sha3 = "0.10.7"
subprocess = "0.2.9" subprocess = "0.2.9"
tokio = { version = "1.27.0", features = ["full"] } tokio = { version = "1.27.0", features = ["full"] }
toml = "0.7.3" toml = "0.7.3"
...@@ -14,9 +14,6 @@ callsign = "NOCALL" ...@@ -14,9 +14,6 @@ callsign = "NOCALL"
# Max image dimension. Comment this out to send full-sized images # Max image dimension. Comment this out to send full-sized images
max_img_dimension = 1024 max_img_dimension = 1024
# radio uart
uart = "/dev/ttyAMA0"
# Comment this section out if you're not using # Comment this section out if you're not using
# APRS for control # APRS for control
[control] [control]
......
...@@ -14,7 +14,6 @@ pub struct ControlConfig { ...@@ -14,7 +14,6 @@ pub struct ControlConfig {
pub struct Config { pub struct Config {
pub paths: Vec<PathBuf>, pub paths: Vec<PathBuf>,
pub callsign: String, pub callsign: String,
pub uart: String,
pub max_img_dimension: Option<u32>, pub max_img_dimension: Option<u32>,
pub control: Option<ControlConfig>, pub control: Option<ControlConfig>,
......
...@@ -11,13 +11,12 @@ use crate::{ ...@@ -11,13 +11,12 @@ use crate::{
config::Config, config::Config,
img::ImgManager, img::ImgManager,
packet::{FecPacket, Packet}, packet::{FecPacket, Packet},
radio::UartRadio, radio::McuRadio,
ssdv::ssdv_encode, ssdv::ssdv_encode,
}; };
const IMAGE_PACKET_QUEUE_LENGTH: usize = 8192; const IMAGE_PACKET_QUEUE_LENGTH: usize = 8192;
const TEMP_REFRESH_INTERVAL: Duration = Duration::from_secs(5); const TEMP_REFRESH_INTERVAL: Duration = Duration::from_secs(5);
const SYNC_REFRESH_INTERVAL: Duration = Duration::from_secs(7);
pub struct Controller { pub struct Controller {
config: Config, config: Config,
...@@ -35,8 +34,7 @@ impl Controller { ...@@ -35,8 +34,7 @@ impl Controller {
{ {
let callsign = self.config.callsign.clone(); let callsign = self.config.callsign.clone();
let uart = self.config.uart.clone(); thread::spawn(|| Self::tx_thread(callsign, img_rx, telem_rx));
thread::spawn(|| Self::tx_thread(callsign, uart, img_rx, telem_rx));
} }
{ {
...@@ -86,14 +84,9 @@ impl Controller { ...@@ -86,14 +84,9 @@ impl Controller {
} }
// manages our transceiver // manages our transceiver
fn tx_thread( fn tx_thread(callsign: String, image_rx: Receiver<FecPacket>, telem_rx: Receiver<Packet>) {
callsign: String,
uart: String,
image_rx: Receiver<FecPacket>,
telem_rx: Receiver<Packet>,
) {
let mut radio = loop { let mut radio = loop {
let r = UartRadio::new(&uart); let r = McuRadio::new();
match r { match r {
Ok(r) => break r, Ok(r) => break r,
...@@ -107,49 +100,53 @@ impl Controller { ...@@ -107,49 +100,53 @@ impl Controller {
let mut text_msg_id = 0; let mut text_msg_id = 0;
let mut last_got_temp = Instant::now(); let mut last_got_temp = Instant::now();
let mut last_synced_at = Instant::now();
loop { loop {
while let Ok(tm) = telem_rx.try_recv() { if let Err(e) = Self::tx_thread_single_iter(
let tm = tm.into_raw(&mut text_msg_id).into(); &callsign,
&image_rx,
if let Err(e) = radio.send_packet(&tm) { &telem_rx,
eprintln!("Could not send packet: {}", e); &mut text_msg_id,
} &mut last_got_temp,
&mut radio,
) {
eprintln!("Radio error: {}", e);
radio.reset();
} }
}
}
if let Ok(img) = image_rx.try_recv() { fn tx_thread_single_iter(
if let Err(e) = radio.send_packet(&img) { callsign: &str,
eprintln!("Could not send packet: {}", e); image_rx: &Receiver<FecPacket>,
} telem_rx: &Receiver<Packet>,
} else { text_msg_id: &mut u16,
thread::sleep(Duration::from_millis(50)); last_got_temp: &mut Instant,
} radio: &mut McuRadio,
) -> anyhow::Result<()> {
while let Ok(tm) = telem_rx.try_recv() {
let tm = tm.into_raw(text_msg_id).into();
radio.send_packet(&tm)?;
}
if Instant::now() - last_got_temp > TEMP_REFRESH_INTERVAL { if let Ok(img) = image_rx.try_recv() {
last_got_temp = Instant::now(); radio.send_packet(&img)?;
} else {
radio.flush()?;
let temp = match radio.get_temp() { thread::sleep(Duration::from_millis(50));
Ok(x) => x, }
Err(e) => {
eprintln!("Could not get radio temp: {}", e);
continue;
}
};
let packet = Packet::new_text_message(&callsign, &format!("Temp: {}", temp)); if Instant::now() - *last_got_temp > TEMP_REFRESH_INTERVAL {
*last_got_temp = Instant::now();
if let Err(e) = radio.send_packet(&packet.into_raw(&mut text_msg_id).into()) { let temp = radio.get_temp()?;
eprintln!("Could not send packet: {}", e);
}
}
if Instant::now() - last_synced_at > SYNC_REFRESH_INTERVAL { let packet = Packet::new_text_message(callsign, &format!("Temp: {}", temp));
last_synced_at = Instant::now();
if let Err(e) = radio.sync() { radio.send_packet(&packet.into_raw(text_msg_id).into())?;
eprintln!("Could not sync: {}", e);
}
}
} }
Ok(())
} }
} }
...@@ -23,16 +23,13 @@ impl Packet { ...@@ -23,16 +23,13 @@ impl Packet {
.iter() .iter()
.chain(b": ".iter()) .chain(b": ".iter())
.chain(msg.as_bytes().iter()); .chain(msg.as_bytes().iter());
let mut len = 0;
for (i, b) in iter.enumerate().take(TEXT_MESSAGE_LEN) { for (i, b) in iter.enumerate().take(TEXT_MESSAGE_LEN) {
out[i] = *b; out[i] = *b;
len += 1;
} }
let len = msg let len = len.min(TEXT_MESSAGE_LEN).try_into().unwrap();
.as_bytes()
.len()
.min(TEXT_MESSAGE_LEN)
.try_into()
.unwrap();
Self::TextMessage(len, out) Self::TextMessage(len, out)
} }
......
use serial::{unix::TTYPort, BaudRate::BaudOther, SerialPort}; use anyhow::anyhow;
use std::{ use rppal::{
io::{self, Read, Write}, gpio::{Gpio, OutputPin},
thread::sleep, spi::{Bus, Mode, SlaveSelect, Spi},
time::Duration,
}; };
use std::{thread::sleep, time::Duration};
use crate::packet::FecPacket; use crate::packet::FecPacket;
const BUFFER_STATUS_CMD: u8 = 0x00;
const SEND_PACKET_CMD: u8 = 0x01; const SEND_PACKET_CMD: u8 = 0x01;
const GET_TEMP_CMD: u8 = 0x02; const NOP_CMD: u8 = 0x55;
const SYNC_CMD: u8 = 0x03;
const PACKET_SPACE: u8 = 0x02; const PREAMBLE: u8 = 0xE5;
const NO_SPACE: u8 = 0x55;
const FREE_SPACE: u8 = 0xAA;
// Used for talking to an RF4463 via a microcontroller // Used for talking to an RF4463 via a microcontroller
// The microcontroller takes in packet data over UART and sends it to the RF4463 // The microcontroller takes in packet data over SPI and sends it to the RF4463
pub struct UartRadio { pub struct McuRadio {
uart: TTYPort, spi: Spi,
reset_pin: OutputPin,
} }
impl UartRadio { impl McuRadio {
pub fn new(uart: &str) -> anyhow::Result<Self> { pub fn new() -> anyhow::Result<Self> {
let mut uart = serial::open(uart)?; let spi = Spi::new(Bus::Spi0, SlaveSelect::Ss1, 1_000_000, Mode::Mode0)?;
uart.reconfigure(&|settings| settings.set_baud_rate(BaudOther(921_600)))?; let reset_pin = Gpio::new()?.get(8)?.into_output_high();
Ok(Self { uart }) let mut s = Self { reset_pin, spi };
}
pub fn send_packet(&mut self, packet: &FecPacket) -> anyhow::Result<()> { s.reset();
self.uart.write_all(&[SEND_PACKET_CMD])?;
self.uart.write_all(&packet.0)?;
let mut buf = [0; 1]; Ok(s)
self.uart.read_exact(&mut buf)?; }
// wait until we have space for packets pub fn reset(&mut self) {
while buf != [PACKET_SPACE] { self.reset_pin.set_low();
sleep(Duration::from_millis(10)); sleep(Duration::from_millis(10));
self.reset_pin.set_high();
sleep(Duration::from_millis(1000));
}
// poll to see if the buffer has emptied yet pub fn send_packet(&mut self, packet: &FecPacket) -> anyhow::Result<()> {
self.uart.write_all(&[BUFFER_STATUS_CMD])?; // wait until we have packet space
self.uart.read_exact(&mut buf)?; while !self.get_info()?.0 {
sleep(Duration::from_millis(10));
} }
self.spi.write(&[SEND_PACKET_CMD])?;
self.spi.write(&packet.0)?;
Ok(()) Ok(())
} }
pub fn get_temp(&mut self) -> anyhow::Result<f32> { pub fn get_temp(&mut self) -> anyhow::Result<f32> {
self.uart.write_all(&[GET_TEMP_CMD])?; Ok(self.get_info()?.1)
}
let mut buf = [0; 4];
self.uart.read_exact(&mut buf)?;
let temp = f32::from_le_bytes(buf); // sends a bunch of nops over SPI to clear out its DMA buffer
pub fn flush(&mut self) -> anyhow::Result<()> {
self.spi.write(&[NOP_CMD; 256])?;
Ok(temp) Ok(())
} }
pub fn sync(&mut self) -> anyhow::Result<()> { // try 10 times and then give up
// clear any pending packets fn get_info(&mut self) -> anyhow::Result<(bool, f32)> {
let mut tries = 0; for _ in 0..9 {
let mut buf = [0; 100]; if let Ok(x) = self.get_info_once() {
loop { return Ok(x);
match self.uart.read(&mut buf) {
Ok(0) => break,
Ok(_) => {
tries += 1;
}
Err(e) if matches!(e.kind(), io::ErrorKind::TimedOut) => break,
Err(e) => {
return Err(e.into());
}
} }
} }
if tries > 0 { self.get_info_once()
eprintln!("[WARN] Sync had pending packets(tries={tries})"); }
}
tries = 0; fn get_info_once(&mut self) -> anyhow::Result<(bool, f32)> {
let mut buf = [0; 10]; // get 256 bytes from the mcu
loop { let mut buf = [0; 256];
self.uart.write_all(&[SYNC_CMD; 10])?; self.spi.transfer(&mut buf, &[NOP_CMD; 256])?;
match self.uart.read_exact(&mut buf) {
Ok(()) => break,
Err(e) if matches!(e.kind(), io::ErrorKind::TimedOut) => {
tries += 1;
}
Err(e) => {
return Err(e.into());
}
}
}
eprintln!("Synced in {tries} tries"); // look for our data
// it goes [0xE5, 0xE5, 0xAA || 0x55, ?, ?, ?, ?, 0xE5, 0xE5]
for i in 0..(buf.len() - 9) {
let chunk = &buf[i..(i + 9)];
Ok(()) if chunk[0..2] == [PREAMBLE, PREAMBLE]
&& (chunk[2] == FREE_SPACE || chunk[2] == NO_SPACE)
&& chunk[7..9] == [PREAMBLE, PREAMBLE]
{
let free = chunk[2] == FREE_SPACE;
let temp = f32::from_le_bytes(chunk[3..7].try_into().unwrap());
return Ok((free, temp));
}
}
Err(anyhow!("could not find valid data"))
} }
} }
use anyhow::{bail, Context}; use anyhow::{bail, Context};
use subprocess::{Popen, PopenConfig, Redirection};
use crate::packet::RawPacket; use crate::packet::RawPacket;
// TODO eventually rewrite Ssdv in Rust? // TODO eventually rewrite Ssdv in Rust?
// Don't want to use FFI because then segfaults can hurt us // Don't want to use FFI because then segfaults can hurt us
pub fn ssdv_encode(callsign: &str, img: &[u8], img_idx: u8) -> anyhow::Result<Vec<RawPacket>> { pub fn ssdv_encode(callsign: &str, img: &[u8], img_idx: u8) -> anyhow::Result<Vec<RawPacket>> {
let (stdout, stderr) = subprocess::Exec::cmd("ssdv") let mut p = Popen::create(
.args(&[ &[
"ssdv",
"-e", "-e",
"-c", "-c",
callsign, callsign,
...@@ -15,10 +17,16 @@ pub fn ssdv_encode(callsign: &str, img: &[u8], img_idx: u8) -> anyhow::Result<Ve ...@@ -15,10 +17,16 @@ pub fn ssdv_encode(callsign: &str, img: &[u8], img_idx: u8) -> anyhow::Result<Ve
"6", "6",
"-i", "-i",
&format!("{}", img_idx), &format!("{}", img_idx),
]) ],
.stdin(img.to_vec()) PopenConfig {
.communicate()? stdin: Redirection::Pipe,
.read()?; stdout: Redirection::Pipe,
stderr: Redirection::Pipe,
..Default::default()
},
)?;
let (stdout, stderr) = p.communicate_bytes(Some(img))?;
if let Some(stderr) = stderr { if let Some(stderr) = stderr {
if !stderr.is_empty() { if !stderr.is_empty() {
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment