extern crate rubato; use rubato::{FastFixedIn, PolynomialDegree, Resampler}; use std::convert::TryInto; use std::env; use std::fs::File; use std::io::prelude::{Read, Seek, Write}; use std::io::Cursor; use std::time::Instant; extern crate env_logger; extern crate log; use env_logger::Builder; use log::LevelFilter; ///! A resampler app that reads a raw file of little-endian 64 bit floats, and writes the output in the same format. ///! The command line arguments are input filename, output filename, input samplerate, output samplerate, ///! number of channels, final relative ratio in percent, and ramp duration in seconds. ///! To resample the file `sine_f64_2ch.raw` from 44.1kHz to 192kHz, and assuming the file has two channels, /// and that the resampling ratio should be ramped to 150% during 3 seconds, the command is: ///! ``` ///! cargo run --release --example fastfixedin_ramp64 sine_f64_2ch.raw test.raw 44100 192000 2 150 3 ///! ``` ///! There are two helper python scripts for testing. `makesineraw.py` simply writes a stereo file ///! with a 1 second long 1kHz tone (at 44.1kHz). This script takes no aruments. Modify as needed to create other test files. ///! To analyze the result, use the `analyze_result.py` script. This takes three arguments: number of channels, samplerate, and number of bits per sample (32 or 64). ///! Example, to analyze the file created above: ///! ``` ///! python examples/analyze_result.py test.raw 2 192000 64 ///! ``` /// Helper to read frames from a buffer fn read_frames(inbuffer: &mut R, nbr: usize, channels: usize) -> Vec> { let mut buffer = vec![0u8; 8]; let mut wfs = Vec::with_capacity(channels); for _chan in 0..channels { wfs.push(Vec::with_capacity(nbr)); } let mut value: f64; for _frame in 0..nbr { for wf in wfs.iter_mut().take(channels) { inbuffer.read_exact(&mut buffer).unwrap(); value = f64::from_le_bytes(buffer.as_slice().try_into().unwrap()); //idx += 8; wf.push(value); } } wfs } /// Helper to write frames to a buffer fn write_frames(waves: Vec>, outbuffer: &mut W, channels: usize) { let nbr = waves[0].len(); for frame in 0..nbr { for wave in waves.iter().take(channels) { let value64 = wave[frame]; let bytes = value64.to_le_bytes(); outbuffer.write_all(&bytes).unwrap(); } } } fn main() { // init logger let mut builder = Builder::from_default_env(); builder.filter(None, LevelFilter::Debug).init(); let file_in = env::args().nth(1).expect("Please specify an input file."); let file_out = env::args().nth(2).expect("Please specify an output file."); println!("Opening files: {}, {}", file_in, file_out); let fs_in_str = env::args() .nth(3) .expect("Please specify an input sample rate"); let fs_out_str = env::args() .nth(4) .expect("Please specify an output sample rate"); let fs_in = fs_in_str.parse::().unwrap(); let fs_out = fs_out_str.parse::().unwrap(); println!("Resampling from {} to {}", fs_in, fs_out); let channels_str = env::args() .nth(5) .expect("Please specify number of channels"); let channels = channels_str.parse::().unwrap(); let ratio_str = env::args() .nth(6) .expect("Please specify final resampling ratio in percent"); let final_ratio = ratio_str.parse::().unwrap(); let duration_str = env::args() .nth(7) .expect("Please specify ramp time in seconds"); let duration = duration_str.parse::().unwrap(); //open files let mut f_in_disk = File::open(file_in).expect("Can't open file"); let mut f_in_ram: Vec = vec![]; //let mut f_out_ram: Vec = vec![]; println!("Copy input file to buffer"); std::io::copy(&mut f_in_disk, &mut f_in_ram).unwrap(); let file_size = f_in_ram.len(); let mut f_out_ram: Vec = Vec::with_capacity(2 * (file_size as f32 * fs_out as f32 / fs_in as f32) as usize); let mut f_in = Cursor::new(&f_in_ram); let mut f_out = Cursor::new(&mut f_out_ram); // parameters let f_ratio = fs_out as f64 / fs_in as f64; let chunksize = 1024; let target_ratio = final_ratio / 100.0; let mut resampler = FastFixedIn::::new( f_ratio, target_ratio, PolynomialDegree::Cubic, chunksize, channels, ) .unwrap(); let num_chunks = f_in_ram.len() / (8 * channels * chunksize); let mut output_time = 0.0; let start = Instant::now(); for _chunk in 0..num_chunks { let waves = read_frames(&mut f_in, chunksize, channels); let waves_out = resampler.process(&waves, None).unwrap(); let new_frames = waves_out[0].len(); write_frames(waves_out, &mut f_out, channels); output_time += new_frames as f64 / fs_out as f64; if output_time < duration { let rel_time = output_time / duration; let rel_ratio = 1.0 + (target_ratio - 1.0) * rel_time; println!("time {}, rel ratio {}", output_time, rel_ratio); resampler .set_resample_ratio_relative(rel_ratio, false) .unwrap(); } } let duration = start.elapsed(); println!("Resampling took: {:?}", duration); let mut f_out_disk = File::create(file_out).unwrap(); f_out.seek(std::io::SeekFrom::Start(0)).unwrap(); std::io::copy(&mut f_out, &mut f_out_disk).unwrap(); }