embedded-trainings-2020/beginner/apps/src/bin/radio-puzzle-solution.rs

102 lines
2.8 KiB
Rust
Raw Normal View History

2020-06-24 14:38:01 +00:00
#![deny(unused_must_use)]
#![no_main]
#![no_std]
use core::str;
use cortex_m_rt::entry;
use dk::ieee802154::{Channel, Packet};
2021-06-04 11:41:31 +00:00
use heapless::{LinearMap, Vec};
2021-04-12 09:51:44 +00:00
// this imports `beginner/apps/lib.rs` to retrieve our global logger + panicking-behavior
use apps as _;
2020-06-24 14:38:01 +00:00
const TEN_MS: u32 = 10_000;
#[entry]
fn main() -> ! {
let board = dk::init().unwrap();
let mut radio = board.radio;
let mut timer = board.timer;
// puzzle.hex uses channel 25
radio.set_channel(Channel::_25);
/* # Build a dictionary */
2021-06-04 11:41:31 +00:00
let mut dict = LinearMap::<u8, u8, 128>::new();
// ^^^ NOTE larger capacity
2020-06-24 14:38:01 +00:00
// the IEEE 802.15.4 packet that will carry our data
2020-06-24 14:38:01 +00:00
let mut packet = Packet::new();
for plainletter in 0..=127 {
// ^^^^^^^ NOTE complete ASCII range
packet.copy_from_slice(&[plainletter]);
2021-03-16 12:26:51 +00:00
radio.send(&mut packet);
2020-06-24 14:38:01 +00:00
if radio.recv_timeout(&mut packet, &mut timer, TEN_MS).is_ok() {
// response should be one byte large
if packet.len() == 1 {
let cipherletter = packet[0];
// NOTE we want to map in reverse: from cipherletter to plainletter
dict.insert(cipherletter, plainletter)
.expect("dictionary full");
} else {
2021-04-12 09:51:44 +00:00
defmt::error!("response packet was not a single byte");
2020-06-24 14:38:01 +00:00
dk::exit()
}
} else {
2021-04-12 09:51:44 +00:00
defmt::error!("no response or response packet was corrupted");
2020-06-24 14:38:01 +00:00
dk::exit()
}
}
/* # Retrieve the secret string */
packet.copy_from_slice(&[]); // empty packet
2021-03-16 12:26:51 +00:00
radio.send(&mut packet);
2020-06-24 14:38:01 +00:00
if radio.recv_timeout(&mut packet, &mut timer, TEN_MS).is_err() {
2021-04-12 09:51:44 +00:00
defmt::error!("no response or response packet was corrupted");
2020-06-24 14:38:01 +00:00
dk::exit()
}
2022-01-07 16:24:21 +00:00
defmt::println!(
2020-06-24 14:38:01 +00:00
"ciphertext: {}",
str::from_utf8(&packet).expect("packet was not valid UTF-8")
);
/* # Decrypt the string */
2021-06-04 11:41:31 +00:00
let mut buffer = Vec::<u8, 128>::new();
2020-06-24 14:38:01 +00:00
// iterate over the bytes
for cipherletter in packet.iter() {
let key = cipherletter;
let value = dict[key];
let plainletter = value;
buffer.push(plainletter).expect("buffer full");
}
2022-01-07 16:24:21 +00:00
defmt::println!(
2020-06-24 14:38:01 +00:00
"plaintext: {}",
str::from_utf8(&buffer).expect("buffer contains non-UTF-8 data")
);
/* # Verify decrypted text */
packet.copy_from_slice(&buffer);
2021-03-16 12:26:51 +00:00
radio.send(&mut packet);
2020-06-24 14:38:01 +00:00
if radio.recv_timeout(&mut packet, &mut timer, TEN_MS).is_err() {
2021-04-12 09:51:44 +00:00
defmt::error!("no response or response packet was corrupted");
2020-06-24 14:38:01 +00:00
dk::exit()
}
2022-01-07 16:24:21 +00:00
defmt::println!(
2020-06-24 14:38:01 +00:00
"Dongle response: {}",
str::from_utf8(&packet).expect("response was not UTF-8")
);
dk::exit()
}