2024-02-20 23:29:33 +02:00
|
|
|
use crate::offline::{Header, HDR_SIZE};
|
2024-02-23 01:38:32 +02:00
|
|
|
use bracket_color::prelude::*;
|
2024-02-19 21:53:25 +02:00
|
|
|
use clap::Parser;
|
2024-02-17 15:43:20 +02:00
|
|
|
use dotenv::dotenv;
|
2024-02-23 01:48:11 +02:00
|
|
|
use std::time::SystemTime;
|
2024-02-23 01:38:32 +02:00
|
|
|
use std::{
|
|
|
|
io::Write,
|
|
|
|
sync::{Arc, Mutex},
|
|
|
|
thread::spawn,
|
|
|
|
};
|
2024-02-17 16:55:53 +02:00
|
|
|
use v4l::video::Output;
|
2024-02-16 00:41:39 +02:00
|
|
|
|
2024-02-19 21:53:25 +02:00
|
|
|
#[derive(Parser, Debug)]
|
|
|
|
#[command(version, about, long_about = None)]
|
|
|
|
struct Args {
|
|
|
|
#[arg(short, long, default_value_t = false)]
|
|
|
|
temperature: bool,
|
|
|
|
#[arg(short, long, default_value = "/dev/video0")]
|
|
|
|
device: String,
|
2024-02-19 22:02:27 +02:00
|
|
|
#[arg(short, long)]
|
|
|
|
red_cutoff: Option<f64>,
|
2024-02-19 21:53:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn pixel_to_celcius(x: u16) -> u16 {
|
|
|
|
let x: f64 = x.into();
|
|
|
|
let x = x / 256.0;
|
|
|
|
let t = (-1.665884e-08) * x.powf(4.)
|
|
|
|
+ (1.347094e-05) * x.powf(3.)
|
|
|
|
+ (-4.396264e-03) * x.powf(2.)
|
|
|
|
+ (9.506939e-01) * x
|
|
|
|
+ (-6.353247e+01);
|
|
|
|
(t * 256.0) as u16
|
|
|
|
}
|
|
|
|
|
2024-02-22 23:08:30 +02:00
|
|
|
/// https://en.wikipedia.org/wiki/HSL_and_HSV
|
|
|
|
/// convert to the expected dynamic range first. We insert values in [0..256)
|
|
|
|
/// h in [0, 360] degrees
|
|
|
|
/// s in [0, 1]
|
|
|
|
/// v in [0, 1]
|
|
|
|
fn once_upon_a_time_hsv2rgb(h: u8, s: u8, v: u8) -> (u8, u8, u8) {
|
|
|
|
let h = (h as f64) / 256.0 * 360.0;
|
|
|
|
let s = (s as f64) / 256.0;
|
|
|
|
let v = (v as f64) / 256.0;
|
|
|
|
(0, 0, 0)
|
|
|
|
}
|
|
|
|
|
2024-02-23 01:38:32 +02:00
|
|
|
fn rgb_to_u8s(rgb: &RGB) -> (u8, u8, u8) {
|
|
|
|
(
|
|
|
|
(rgb.r * 256.) as u8,
|
|
|
|
(rgb.g * 256.) as u8,
|
|
|
|
(rgb.b * 256.) as u8,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) struct Streamer {
|
2024-02-23 01:54:56 +02:00
|
|
|
pub(crate) min_cutoff: f64,
|
|
|
|
pub(crate) max_cutoff: f64,
|
|
|
|
pub(crate) freq_hz: f64,
|
2024-02-23 01:38:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn initialize() -> Arc<Mutex<Streamer>> {
|
2024-02-19 21:53:25 +02:00
|
|
|
let args = Args::parse();
|
2024-02-23 01:38:32 +02:00
|
|
|
Arc::new(Mutex::new(Streamer {
|
2024-02-23 01:54:56 +02:00
|
|
|
min_cutoff: args.red_cutoff.unwrap_or(26.),
|
|
|
|
max_cutoff: args.red_cutoff.unwrap_or(26.) + 10.0,
|
|
|
|
freq_hz: 1.0,
|
2024-02-23 01:38:32 +02:00
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main(streamer: Arc<Mutex<Streamer>>) -> anyhow::Result<()> {
|
2024-02-17 15:43:20 +02:00
|
|
|
dotenv().ok();
|
2024-02-23 01:38:32 +02:00
|
|
|
let args = Args::parse();
|
2024-02-17 15:43:20 +02:00
|
|
|
let device = match std::env::var("THERMALCAM_IFACE=enp1s0f0") {
|
|
|
|
Ok(d) => {
|
|
|
|
let device = pcap::Device::list()
|
|
|
|
.expect("device list failed")
|
|
|
|
.into_iter()
|
|
|
|
.find(|x| x.name == d)
|
|
|
|
.expect(&format!("could not find device {}", d));
|
|
|
|
device
|
|
|
|
}
|
|
|
|
Err(_) => pcap::Device::lookup()
|
|
|
|
.expect("device lookup failed")
|
|
|
|
.expect("no device available"),
|
|
|
|
};
|
2024-02-16 00:41:39 +02:00
|
|
|
// get the default Device
|
2024-02-17 15:43:20 +02:00
|
|
|
|
2024-02-16 00:41:39 +02:00
|
|
|
println!("Using device {}", device.name);
|
2024-02-19 21:53:25 +02:00
|
|
|
let output = args.device;
|
2024-02-17 16:55:53 +02:00
|
|
|
println!("Using output v4l2loopback device {}", output);
|
|
|
|
|
|
|
|
const WIDTH: usize = 288;
|
|
|
|
const HEIGHT: usize = 384;
|
2024-02-23 01:38:32 +02:00
|
|
|
println!("reading cutoff");
|
2024-02-23 01:48:11 +02:00
|
|
|
let start = SystemTime::now()
|
|
|
|
.duration_since(SystemTime::UNIX_EPOCH)
|
|
|
|
.unwrap()
|
|
|
|
.as_secs_f64();
|
2024-02-23 01:54:56 +02:00
|
|
|
let greyscale = !args.temperature;
|
2024-02-19 22:26:25 +02:00
|
|
|
let fourcc_repr = if greyscale {
|
|
|
|
[
|
|
|
|
b'Y', // | 0b10000000
|
|
|
|
b'1', b'6',
|
|
|
|
b' ', // Note: not using b' ' | 0x80, (V4L2_PIX_FMT_Y16_BE)
|
|
|
|
// because VID_S_FMT ioctl returns EINVAL, so just swap the bytes here
|
|
|
|
]
|
|
|
|
} else {
|
|
|
|
// RGB32 is 4 bytes R, G, B, A
|
|
|
|
[b'R', b'G', b'B', b'4']
|
|
|
|
};
|
2024-02-23 01:38:32 +02:00
|
|
|
println!("using four cc {:?}", fourcc_repr);
|
2024-02-19 22:26:25 +02:00
|
|
|
let bytes_per_pixel = if greyscale { 2 } else { 4 };
|
2024-02-17 16:59:29 +02:00
|
|
|
let fourcc = v4l::format::FourCC { repr: fourcc_repr };
|
2024-02-17 16:55:53 +02:00
|
|
|
let mut out = v4l::Device::with_path(output)?;
|
|
|
|
// To find the fourcc code, use v4l2-ctl --list-formats-out /dev/video0
|
|
|
|
// (or read the source :)
|
2024-02-19 22:31:07 +02:00
|
|
|
// flip axes
|
|
|
|
let format = v4l::Format::new(HEIGHT as u32, WIDTH as u32, fourcc);
|
2024-02-17 16:55:53 +02:00
|
|
|
Output::set_format(&out, &format)?;
|
2024-02-16 00:41:39 +02:00
|
|
|
|
|
|
|
// Setup Capture
|
|
|
|
let mut cap = pcap::Capture::from_device(device)
|
|
|
|
.unwrap()
|
|
|
|
.immediate_mode(true)
|
|
|
|
.open()
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
// get a packet and print its bytes
|
2024-02-17 15:56:05 +02:00
|
|
|
const PACKET_LEN: usize = 6972;
|
2024-02-19 22:26:25 +02:00
|
|
|
// input is grayscale 16 bits per pixel
|
2024-02-17 16:55:53 +02:00
|
|
|
const FRAME_LEN: usize = WIDTH * HEIGHT * 2;
|
2024-02-17 15:56:05 +02:00
|
|
|
let mut frame = [0u8; FRAME_LEN];
|
|
|
|
let mut len = 0;
|
2024-02-19 22:26:25 +02:00
|
|
|
let output_frame_len = WIDTH * HEIGHT * bytes_per_pixel;
|
|
|
|
let mut swapped_vec = vec![0u8; output_frame_len];
|
|
|
|
let swapped = &mut swapped_vec;
|
2024-02-16 00:41:39 +02:00
|
|
|
while let Ok(p) = cap.next_packet() {
|
|
|
|
let data = p.data;
|
2024-02-17 16:55:53 +02:00
|
|
|
if data.len() != PACKET_LEN {
|
2024-02-16 00:41:39 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
let data = &data[0x2a..];
|
|
|
|
let header = match Header::read(data) {
|
|
|
|
Ok(header) => header,
|
|
|
|
Err(_) => continue,
|
|
|
|
};
|
|
|
|
let data = &data[HDR_SIZE..];
|
2024-02-17 15:56:05 +02:00
|
|
|
if (header.part == 0 && len > 0)
|
|
|
|
// do not write out of bounds - would panic, instead just skip
|
|
|
|
|| (data.len() + len > FRAME_LEN)
|
|
|
|
{
|
|
|
|
if len == FRAME_LEN {
|
2024-02-23 01:38:32 +02:00
|
|
|
// read once per frame, can make it lower if need be
|
2024-02-23 01:54:56 +02:00
|
|
|
let state = streamer.lock().unwrap();
|
|
|
|
let mid = (state.min_cutoff + state.max_cutoff) / 2.0;
|
|
|
|
let range = state.max_cutoff - state.min_cutoff;
|
|
|
|
let hz = state.freq_hz;
|
2024-02-23 01:48:11 +02:00
|
|
|
let now = SystemTime::now()
|
|
|
|
.duration_since(SystemTime::UNIX_EPOCH)
|
|
|
|
.unwrap()
|
|
|
|
.as_secs_f64();
|
|
|
|
let dt = now - start;
|
2024-02-23 02:12:30 +02:00
|
|
|
let cutoff = mid + f64::sin(dt * hz) * 0.5 * range;
|
2024-02-17 17:21:01 +02:00
|
|
|
// swap the bytes, we are using LE, not BE, 16 bit grayscale
|
|
|
|
// possibly limitation of current v4l2loopback or v4l rust wrapper or libv4l2
|
|
|
|
for i in 0..FRAME_LEN / 2 {
|
2024-02-19 22:31:07 +02:00
|
|
|
let x = i % WIDTH;
|
|
|
|
let y = (i / WIDTH) % HEIGHT;
|
2024-02-19 21:53:25 +02:00
|
|
|
let mut pixel = u16::from_be_bytes([frame[i * 2], frame[i * 2 + 1]]);
|
2024-02-19 22:26:25 +02:00
|
|
|
if greyscale {
|
|
|
|
if args.temperature {
|
|
|
|
pixel = pixel_to_celcius(pixel);
|
2024-02-19 22:02:27 +02:00
|
|
|
}
|
2024-02-19 22:26:25 +02:00
|
|
|
let pixel_swapped = pixel.to_le_bytes();
|
2024-02-19 22:31:07 +02:00
|
|
|
let out_i = ((HEIGHT - 1 - y) + (WIDTH - 1 - x) * HEIGHT) * 2;
|
|
|
|
swapped[out_i..out_i + 2].copy_from_slice(&pixel_swapped);
|
2024-02-19 22:26:25 +02:00
|
|
|
} else {
|
|
|
|
pixel = pixel_to_celcius(pixel);
|
2024-02-20 21:02:14 +02:00
|
|
|
let (r, g, b) = if pixel > (256.0 * cutoff) as u16 {
|
|
|
|
let p = pixel - (256.0 * cutoff) as u16;
|
2024-02-23 01:38:32 +02:00
|
|
|
let rgb = HSV::from_f32(0.0, (p as f32) / 256.0, 0.0).to_rgb();
|
|
|
|
rgb_to_u8s(&rgb)
|
2024-02-19 22:26:25 +02:00
|
|
|
} else {
|
2024-02-23 01:38:32 +02:00
|
|
|
let rgb =
|
|
|
|
HSV::from_f32(pixel as f32 / 65536.0, 0.0, pixel as f32 / 65536.0)
|
|
|
|
.to_rgb();
|
|
|
|
rgb_to_u8s(&rgb)
|
2024-02-19 22:26:25 +02:00
|
|
|
};
|
2024-02-19 22:38:58 +02:00
|
|
|
let out_i = ((HEIGHT - 1 - y) + (WIDTH - 1 - x) * HEIGHT) * 4;
|
2024-02-19 22:31:07 +02:00
|
|
|
swapped[out_i..out_i + 4].copy_from_slice(&[0, r, g, b]);
|
2024-02-19 21:53:25 +02:00
|
|
|
}
|
2024-02-17 17:21:01 +02:00
|
|
|
}
|
|
|
|
out.write_all(&swapped[..])?;
|
2024-02-16 00:41:39 +02:00
|
|
|
}
|
2024-02-17 15:56:05 +02:00
|
|
|
len = 0;
|
2024-02-16 00:41:39 +02:00
|
|
|
}
|
2024-02-17 15:56:05 +02:00
|
|
|
frame[len..len + data.len()].copy_from_slice(data);
|
|
|
|
len += data.len();
|
2024-02-16 00:41:39 +02:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
2024-02-20 23:29:33 +02:00
|
|
|
|
2024-02-23 01:38:32 +02:00
|
|
|
pub(crate) fn start_stream_thread(streamer: Arc<Mutex<Streamer>>) {
|
2024-02-20 23:29:33 +02:00
|
|
|
spawn(move || {
|
2024-02-23 01:38:32 +02:00
|
|
|
if let Err(e) = main(streamer) {
|
2024-02-20 23:29:33 +02:00
|
|
|
println!("oops: {:?}", e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|