flacenc 0.5.1

Pure rust library for embedding FLAC encoder in your application.
Documentation
// Copyright 2022-2024 Google LLC
// Copyright 2025- flacenc-rs developers
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//      http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

#![allow(clippy::missing_panics_doc)]

use std::collections::BTreeMap;
use std::io::Write;

use once_cell::sync::Lazy;
use rand::distributions::Distribution;
use rand::distributions::Uniform;
use rand::Rng;
use tempfile::NamedTempFile;

use super::arrayutils::le_bytes_to_i32s;
use super::bitsink::ByteSink;
use super::component::BitRepr;
use super::component::BlockSizeSpec;
use super::component::ChannelAssignment;
use super::component::Frame;
use super::component::FrameOffset;
use super::component::Residual;
use super::component::SampleRateSpec;
use super::component::SampleSizeSpec;
use super::component::Stream;
use super::component::StreamInfo;
use super::component::Verbatim;
use super::error::Verify;
use super::source::MemSource;
use super::source::Seekable;
use super::source::Source;

#[macro_export]
macro_rules! assert_close {
    ($actual:expr, $expected:expr, rtol = $rtol:expr, atol = $atol:expr) => {{
        let err = ($actual - $expected).abs();
        #[allow(clippy::suboptimal_flops)]
        let tol = $rtol * ($expected).abs() + $atol;
        assert!(err < tol);
    }};
    ($actual:expr, $expected:expr) => {{
        assert_close!($actual, $expected, rtol = 0.00001, atol = 0.00001);
    }};
}

#[macro_export]
macro_rules! assert_finite {
    ($result:expr) => {{
        for (i, &value) in $result.iter().enumerate() {
            assert!(
                value.is_normal() || value == 0.0,
                "{}-th element in a vector is not finite ({}), x={:?}.",
                i,
                value,
                $result
            );
        }
    }};
}

#[allow(clippy::similar_names)]
fn read_le16(src: &[u8]) -> Vec<i32> {
    assert!(src.len() % 2 == 0);
    let mut ret = vec![0i32; src.len() / 2];
    le_bytes_to_i32s(src, &mut ret, 2);
    ret
}

static TEST_SIGNALS: Lazy<BTreeMap<(&str, usize), Vec<i32>>> = Lazy::new(|| {
    BTreeMap::from([
        (
            ("sus109", 0),
            read_le16(include_bytes!("resource/testsignal.sus109.ch0.bin")),
        ),
        (
            ("sus109", 1),
            read_le16(include_bytes!("resource/testsignal.sus109.ch1.bin")),
        ),
        (
            ("sus6", 0),
            read_le16(include_bytes!("resource/testsignal.sus6.ch0.bin")),
        ),
        (
            ("sus6", 1),
            read_le16(include_bytes!("resource/testsignal.sus6.ch1.bin")),
        ),
        (
            ("ras22", 0),
            read_le16(include_bytes!("resource/testsignal.ras22.ch0.bin")),
        ),
        (
            ("ras22", 1),
            read_le16(include_bytes!("resource/testsignal.ras22.ch1.bin")),
        ),
        (
            ("ras103", 0),
            read_le16(include_bytes!("resource/testsignal.ras103.ch0.bin")),
        ),
        (
            ("ras103", 1),
            read_le16(include_bytes!("resource/testsignal.ras103.ch1.bin")),
        ),
    ])
});

/// Loads monoral test signal by key and the channel specifier.
#[allow(dead_code)]
pub fn test_signal(src: &str, ch: usize) -> Vec<i32> {
    TEST_SIGNALS
        .get(&(src, ch))
        .expect("Specified test signal not found.")
        .clone()
}

/// Runs an integrity test over the given encoding function.
///
/// This runs `encoder` function followed by `claxon`-based FLAC decoding, and
/// compares the waveforms of the original signal and reconstructed signal.
pub fn integrity_test<Enc>(encoder: Enc, src: &MemSource) -> Stream
where
    Enc: Fn(MemSource) -> Stream,
{
    let stream = encoder(src.clone());

    assert!(stream.verify().is_ok());

    let mut file = NamedTempFile::new().expect("Failed to create temp file.");

    let mut bv: ByteSink = ByteSink::with_capacity(stream.count_bits());
    stream.write(&mut bv).expect("Bitstream formatting failed.");
    file.write_all(bv.as_slice()).expect("File write failed.");

    let flac_path = file.into_temp_path();

    let mut reader = claxon::FlacReader::open(&flac_path).unwrap();
    let streaminfo = reader.streaminfo();
    assert_eq!(streaminfo.channels as usize, src.channels());
    assert_eq!(streaminfo.sample_rate as usize, src.sample_rate());
    assert_eq!(streaminfo.bits_per_sample as usize, src.bits_per_sample());
    assert_eq!(streaminfo.samples, Some(src.len() as u64));

    let mut offsets: Vec<usize> = vec![];
    let mut offset = 0;
    for frame in stream.frames() {
        offsets.push(offset);
        offset += frame.header().block_size();
    }
    offsets.push(offset);

    let channels = streaminfo.channels as usize;
    let loaded = reader.samples().map(Result::unwrap).collect::<Vec<i32>>();
    let mut head = 0;
    for t in 0..src.len() {
        if t >= offsets[head] {
            head += 1;
        }
        let current_offset = offsets[head - 1];

        for ch in 0..channels {
            assert_eq!(
                loaded.as_slice()[t * channels + ch],
                src.as_slice()[t * channels + ch],
                "Failed at t={} of ch={} (block={}, in-block-t={})\n{:?}",
                t,
                ch,
                head,
                t - current_offset,
                stream.frame(head).unwrap().subframe(ch).unwrap()
            );
        }
    }
    stream
}

pub fn make_random_residual<R: Rng>(mut rng: R, warmup_length: usize) -> Residual {
    // when blocksize is minimum (64), the partition length is 16, and warmup samples
    // are not expected to go beyond the partition boundary.
    assert!(warmup_length <= 16);
    let block_size = 4 * Uniform::from(16..=1024).sample(&mut rng);
    let partition_order: usize = 2;
    let nparts = 2usize.pow(partition_order as u32);
    let part_len = block_size / nparts;
    let params = vec![7, 8, 6, 7];
    let mut quotients: Vec<u32> = vec![];
    let mut remainders: Vec<u32> = vec![];

    for t in 0..block_size {
        if t < warmup_length {
            quotients.push(0u32);
            remainders.push(0u32);
        } else {
            let part_id = t / part_len;
            let p = params[part_id];
            let denom = 1u32 << p;

            quotients.push(255 / denom);
            remainders.push(255 % denom);
        }
    }
    Residual::new(
        partition_order,
        block_size,
        warmup_length,
        &params,
        &quotients,
        &remainders,
    )
    .expect("Error in random construction of Residual")
}

pub fn make_verbatim_frame(stream_info: &StreamInfo, samples: &[i32], offset: u64) -> Frame {
    let channels = stream_info.channels();
    let block_size = samples.len() / channels;
    let bits_per_sample: u8 = stream_info.bits_per_sample() as u8;
    let ch_info = ChannelAssignment::Independent(channels as u8);
    let mut frame = Frame::new_empty(
        BlockSizeSpec::from_size(block_size as u16),
        ch_info,
        SampleSizeSpec::from_bits(bits_per_sample).unwrap_or(SampleSizeSpec::Unspecified),
        SampleRateSpec::from_freq(stream_info.sample_rate() as u32)
            .unwrap_or(SampleRateSpec::Unspecified),
    );
    frame
        .header_mut()
        .set_frame_offset(FrameOffset::StartSample(offset));
    for ch in 0..channels {
        frame.add_subframe(
            Verbatim::from_samples(
                &samples[block_size * ch..block_size * (ch + 1)],
                bits_per_sample,
            )
            .into(),
        );
    }
    frame
}