#![allow(
clippy::cast_precision_loss,
clippy::cast_possible_truncation,
clippy::cast_sign_loss,
clippy::cast_possible_wrap
)]
use crate::modespec::SstvMode;
use crate::resample::WORKING_SAMPLE_RATE_HZ;
use std::f64::consts::PI;
const SYNC_HZ: f64 = 1200.0;
const PORCH_HZ: f64 = 1500.0;
const SEPTR_HZ: f64 = 1500.0;
const BLACK_HZ: f64 = 1500.0;
const WHITE_HZ: f64 = 2300.0;
fn lum_to_freq(lum: u8) -> f64 {
BLACK_HZ + (WHITE_HZ - BLACK_HZ) * f64::from(lum) / 255.0
}
fn fill_to(out: &mut Vec<f32>, freq_hz: f64, target_n: usize, phase: &mut f64) {
let dphi = 2.0 * PI * freq_hz / f64::from(WORKING_SAMPLE_RATE_HZ);
while out.len() < target_n {
out.push(phase.sin() as f32);
*phase += dphi;
if *phase > 2.0 * PI {
*phase -= 2.0 * PI;
}
}
}
#[must_use]
#[doc(hidden)]
#[allow(dead_code)]
pub fn encode_scottie(mode: SstvMode, rgb: &[[u8; 3]]) -> Vec<f32> {
assert!(matches!(
mode,
SstvMode::Scottie1 | SstvMode::Scottie2 | SstvMode::ScottieDx
));
let spec = crate::modespec::for_mode(mode);
let w = spec.line_pixels;
let h = spec.image_lines;
assert_eq!(rgb.len() as u32, w * h);
let sr = f64::from(WORKING_SAMPLE_RATE_HZ);
let mut out: Vec<f32> = Vec::new();
let mut phase = 0.0_f64;
let mut t = 0.0_f64;
let advance = |t: &mut f64, secs: f64| -> usize {
*t += secs;
(*t * sr).round() as usize
};
for y in 0..h {
fill_to(
&mut out,
SEPTR_HZ,
advance(&mut t, spec.septr_seconds),
&mut phase,
);
for x in 0..w {
let g = rgb[(y * w + x) as usize][1];
fill_to(
&mut out,
lum_to_freq(g),
advance(&mut t, spec.pixel_seconds),
&mut phase,
);
}
fill_to(
&mut out,
SEPTR_HZ,
advance(&mut t, spec.septr_seconds),
&mut phase,
);
for x in 0..w {
let b = rgb[(y * w + x) as usize][2];
fill_to(
&mut out,
lum_to_freq(b),
advance(&mut t, spec.pixel_seconds),
&mut phase,
);
}
fill_to(
&mut out,
SYNC_HZ,
advance(&mut t, spec.sync_seconds),
&mut phase,
);
fill_to(
&mut out,
PORCH_HZ,
advance(&mut t, spec.porch_seconds),
&mut phase,
);
for x in 0..w {
let r = rgb[(y * w + x) as usize][0];
fill_to(
&mut out,
lum_to_freq(r),
advance(&mut t, spec.pixel_seconds),
&mut phase,
);
}
let line_end_target = f64::from(y + 1) * spec.line_seconds;
let pad_secs = line_end_target - t;
if pad_secs > 0.0 {
fill_to(&mut out, PORCH_HZ, advance(&mut t, pad_secs), &mut phase);
}
}
out
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn lum_to_freq_endpoints() {
assert!((lum_to_freq(0) - BLACK_HZ).abs() < 1e-9);
assert!((lum_to_freq(255) - WHITE_HZ).abs() < 1e-9);
}
#[test]
fn scottie1_encode_total_length() {
let rgb = vec![[128u8; 3]; 320 * 256];
let audio = encode_scottie(SstvMode::Scottie1, &rgb);
let spec = crate::modespec::for_mode(SstvMode::Scottie1);
let expected_len = (spec.line_seconds
* f64::from(spec.image_lines)
* f64::from(WORKING_SAMPLE_RATE_HZ)) as usize;
assert!(audio.len() >= expected_len);
assert!(audio.len() <= expected_len + 1);
}
}