slowrx 0.5.0

Pure-Rust SSTV (Slow-Scan TV) decoder library — a port of slowrx by Oona Räisänen
Documentation
//! Synthetic RGB-sequential encoder for round-trip testing —
//! handles both Scottie (1/2/DX) and Martin (1/2) families.
//!
//! Test-only — gated behind `cfg(any(test, feature = "test-support"))`.
//!
//! **Per-line tone emission order branches on
//! [`crate::modespec::SyncPosition`]:**
//!
//! ```text
//! Scottie (sync_position::Scottie):
//!   [septr 1500 Hz][G pixels 1500-2300 Hz][septr 1500 Hz]
//!   [B pixels 1500-2300 Hz][SYNC 1200 Hz][porch 1500 Hz]
//!   [R pixels 1500-2300 Hz]
//!
//! Martin (sync_position::LineStart):
//!   [SYNC 1200 Hz][porch 1500 Hz][G pixels 1500-2300 Hz]
//!   [septr 1500 Hz][B pixels 1500-2300 Hz][septr 1500 Hz]
//!   [R pixels 1500-2300 Hz]
//! ```
//!
//! Total per line = LineTime exactly (defensive pad fills the
//! boundary if float arithmetic rounds short).

#![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
}

/// Emit samples up to absolute sample index `target_n`, advancing phase.
/// Cumulative-target pattern (matches `robot_test_encoder::fill_to`) so
/// per-tone rounding doesn't compound across the line.
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;
        }
    }
}

/// Encode an image as Scottie 1 / 2 / DX audio. `rgb` is row-major,
/// `line_pixels × image_lines` `[R, G, B]` triples (320×256 for all
/// three Scottie modes). Returns f32 PCM at [`WORKING_SAMPLE_RATE_HZ`]
/// (11_025 Hz).
///
/// Per radio line, emits in this order:
///   1. Septr 1 at `SEPTR_HZ`
///   2. G channel at `pixel_seconds` per pixel
///   3. Septr 2 at `SEPTR_HZ`
///   4. B channel at `pixel_seconds` per pixel
///   5. Sync at `SYNC_HZ` (mid-line, between B and R)
///   6. Porch at `PORCH_HZ`
///   7. R channel at `pixel_seconds` per pixel
#[must_use]
/// Encode an RGB image as continuous-phase FM audio for either
/// Scottie (S1/S2/DX) or Martin (M1/M2). The per-line tone emission
/// order branches on `spec.sync_position`:
///
/// - [`crate::modespec::SyncPosition::Scottie`] —
///   `[septr][G][septr][B][SYNC][porch][R]` (sync mid-line).
/// - [`crate::modespec::SyncPosition::LineStart`] —
///   `[SYNC][porch][G][septr][B][septr][R]` (Martin / standard
///   PD-Robot order).
///
/// Panics if `mode` is not one of the five supported variants or if
/// `rgb.len() != line_pixels * image_lines`.
#[doc(hidden)]
#[allow(dead_code, clippy::too_many_lines)]
pub fn encode_scottie(mode: SstvMode, rgb: &[[u8; 3]]) -> Vec<f32> {
    assert!(matches!(
        mode,
        SstvMode::Scottie1
            | SstvMode::Scottie2
            | SstvMode::ScottieDx
            | SstvMode::Martin1
            | SstvMode::Martin2
    ));
    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 {
        match spec.sync_position {
            crate::modespec::SyncPosition::Scottie => {
                // Septr 1.
                fill_to(
                    &mut out,
                    SEPTR_HZ,
                    advance(&mut t, spec.septr_seconds),
                    &mut phase,
                );

                // G channel.
                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,
                    );
                }

                // Septr 2.
                fill_to(
                    &mut out,
                    SEPTR_HZ,
                    advance(&mut t, spec.septr_seconds),
                    &mut phase,
                );

                // B channel.
                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,
                    );
                }

                // Sync (mid-line, between B and R).
                fill_to(
                    &mut out,
                    SYNC_HZ,
                    advance(&mut t, spec.sync_seconds),
                    &mut phase,
                );

                // Porch.
                fill_to(
                    &mut out,
                    PORCH_HZ,
                    advance(&mut t, spec.porch_seconds),
                    &mut phase,
                );

                // R channel.
                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,
                    );
                }
            }
            crate::modespec::SyncPosition::LineStart => {
                // Martin layout: sync at line start, then porch, then
                // G/septr/B/septr/R.

                // Sync.
                fill_to(
                    &mut out,
                    SYNC_HZ,
                    advance(&mut t, spec.sync_seconds),
                    &mut phase,
                );

                // Porch.
                fill_to(
                    &mut out,
                    PORCH_HZ,
                    advance(&mut t, spec.porch_seconds),
                    &mut phase,
                );

                // G channel.
                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,
                    );
                }

                // Septr 1.
                fill_to(
                    &mut out,
                    SEPTR_HZ,
                    advance(&mut t, spec.septr_seconds),
                    &mut phase,
                );

                // B channel.
                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,
                    );
                }

                // Septr 2.
                fill_to(
                    &mut out,
                    SEPTR_HZ,
                    advance(&mut t, spec.septr_seconds),
                    &mut phase,
                );

                // R channel.
                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,
                    );
                }
            }
        }

        // Defensive pad to the line_seconds boundary (existing logic
        // — shape unchanged).
        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;
        // Allow 1-sample rounding slack at end-of-image.
        assert!(audio.len() >= expected_len);
        assert!(audio.len() <= expected_len + 1);
    }
}