Make the window overlap configurable
It gets super whacky now.
This commit is contained in:
parent
115d03a34a
commit
715ba467a9
1 changed files with 31 additions and 8 deletions
|
@ -25,7 +25,9 @@ use std::sync::Arc;
|
||||||
const MIN_WINDOW_SIZE: usize = 64;
|
const MIN_WINDOW_SIZE: usize = 64;
|
||||||
const DEFAULT_WINDOW_SIZE: usize = 1024;
|
const DEFAULT_WINDOW_SIZE: usize = 1024;
|
||||||
const MAX_WINDOW_SIZE: usize = 32768;
|
const MAX_WINDOW_SIZE: usize = 32768;
|
||||||
const OVERLAP_TIMES: usize = 4;
|
const MIN_OVERLAP_TIMES: usize = 2;
|
||||||
|
const DEFAULT_OVERLAP_TIMES: usize = 4;
|
||||||
|
const MAX_OVERLAP_TIMES: usize = 32;
|
||||||
|
|
||||||
struct PubertySimulator {
|
struct PubertySimulator {
|
||||||
params: Pin<Box<PubertySimulatorParams>>,
|
params: Pin<Box<PubertySimulatorParams>>,
|
||||||
|
@ -62,6 +64,10 @@ struct PubertySimulatorParams {
|
||||||
/// The size of the FFT window as a power of two (to prevent invalid inputs).
|
/// The size of the FFT window as a power of two (to prevent invalid inputs).
|
||||||
#[id = "wndsz"]
|
#[id = "wndsz"]
|
||||||
window_size_order: IntParam,
|
window_size_order: IntParam,
|
||||||
|
/// The amount of overlap to use in the overlap-add algorithm as a power of two (again to
|
||||||
|
/// prevent invalid inputs).
|
||||||
|
#[id = "ovrlap"]
|
||||||
|
overlap_times_order: IntParam,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for PubertySimulator {
|
impl Default for PubertySimulator {
|
||||||
|
@ -84,6 +90,10 @@ impl Default for PubertySimulator {
|
||||||
|
|
||||||
impl Default for PubertySimulatorParams {
|
impl Default for PubertySimulatorParams {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
|
let power_of_two_val2str = Arc::new(|value| format!("{}", 1 << value));
|
||||||
|
let power_of_two_str2val =
|
||||||
|
Arc::new(|string: &str| string.parse().ok().map(|n: i32| (n as f32).log2() as i32));
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
pitch_octaves: FloatParam::new(
|
pitch_octaves: FloatParam::new(
|
||||||
"Pitch",
|
"Pitch",
|
||||||
|
@ -109,10 +119,18 @@ impl Default for PubertySimulatorParams {
|
||||||
max: (MAX_WINDOW_SIZE as f32).log2() as i32,
|
max: (MAX_WINDOW_SIZE as f32).log2() as i32,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
.with_value_to_string(Arc::new(|value| format!("{}", 1 << value)))
|
.with_value_to_string(power_of_two_val2str.clone())
|
||||||
.with_string_to_value(Arc::new(|string| {
|
.with_string_to_value(power_of_two_str2val.clone()),
|
||||||
string.parse().ok().map(|n: i32| (n as f32).log2() as i32)
|
overlap_times_order: IntParam::new(
|
||||||
})),
|
"Window Overlap",
|
||||||
|
(DEFAULT_OVERLAP_TIMES as f32).log2() as i32,
|
||||||
|
IntRange::Linear {
|
||||||
|
min: (MIN_OVERLAP_TIMES as f32).log2() as i32,
|
||||||
|
max: (MAX_OVERLAP_TIMES as f32).log2() as i32,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.with_value_to_string(power_of_two_val2str)
|
||||||
|
.with_string_to_value(power_of_two_str2val),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -161,8 +179,9 @@ impl Plugin for PubertySimulator {
|
||||||
// Compensate for the window function, the overlap, and the extra gain introduced by the
|
// Compensate for the window function, the overlap, and the extra gain introduced by the
|
||||||
// IDFT operation
|
// IDFT operation
|
||||||
let window_size = self.window_size();
|
let window_size = self.window_size();
|
||||||
|
let overlap_times = self.overlap_times();
|
||||||
let sample_rate = context.transport().sample_rate;
|
let sample_rate = context.transport().sample_rate;
|
||||||
let gain_compensation: f32 = 2.0 / OVERLAP_TIMES as f32 / window_size as f32;
|
let gain_compensation: f32 = 2.0 / overlap_times as f32 / window_size as f32;
|
||||||
|
|
||||||
// If the window size has changed since the last process call, reset the buffers and chance
|
// If the window size has changed since the last process call, reset the buffers and chance
|
||||||
// our latency. All of these buffers already have enough capacity
|
// our latency. All of these buffers already have enough capacity
|
||||||
|
@ -180,7 +199,7 @@ impl Plugin for PubertySimulator {
|
||||||
self.stft.process_overlap_add(
|
self.stft.process_overlap_add(
|
||||||
buffer,
|
buffer,
|
||||||
&self.window_function,
|
&self.window_function,
|
||||||
OVERLAP_TIMES,
|
overlap_times,
|
||||||
|channel_idx, real_fft_scratch_buffer| {
|
|channel_idx, real_fft_scratch_buffer| {
|
||||||
// This loop runs whenever there's a block ready, so we can't easily do any post- or
|
// This loop runs whenever there's a block ready, so we can't easily do any post- or
|
||||||
// pre-processing without muddying up the interface. But if this is channel 0, then
|
// pre-processing without muddying up the interface. But if this is channel 0, then
|
||||||
|
@ -190,7 +209,7 @@ impl Plugin for PubertySimulator {
|
||||||
.params
|
.params
|
||||||
.pitch_octaves
|
.pitch_octaves
|
||||||
.smoothed
|
.smoothed
|
||||||
.next_step((window_size / OVERLAP_TIMES) as u32);
|
.next_step((window_size / overlap_times) as u32);
|
||||||
}
|
}
|
||||||
// Negated because pitching down should cause us to take values from higher frequency bins
|
// Negated because pitching down should cause us to take values from higher frequency bins
|
||||||
let frequency_multiplier = 2.0f32.powf(-smoothed_pitch_value);
|
let frequency_multiplier = 2.0f32.powf(-smoothed_pitch_value);
|
||||||
|
@ -258,6 +277,10 @@ impl PubertySimulator {
|
||||||
1 << self.params.window_size_order.value as usize
|
1 << self.params.window_size_order.value as usize
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn overlap_times(&self) -> usize {
|
||||||
|
1 << self.params.overlap_times_order.value as usize
|
||||||
|
}
|
||||||
|
|
||||||
/// `window_size` should not exceed `MAX_WINDOW_SIZE` or this will allocate.
|
/// `window_size` should not exceed `MAX_WINDOW_SIZE` or this will allocate.
|
||||||
fn resize_for_window(&mut self, window_size: usize) {
|
fn resize_for_window(&mut self, window_size: usize) {
|
||||||
self.stft.set_block_size(window_size);
|
self.stft.set_block_size(window_size);
|
||||||
|
|
Loading…
Add table
Reference in a new issue