use firewheel_core::{
channel_config::{ChannelConfig, ChannelCount},
diff::{Diff, Patch},
dsp::volume::{Volume, DEFAULT_AMP_EPSILON},
event::NodeEventList,
log::RealtimeLogger,
node::{
AudioNode, AudioNodeInfo, AudioNodeProcessor, ConstructProcessorContext, ProcBuffers,
ProcInfo, ProcessStatus,
},
param::smoother::{SmoothedParam, SmootherConfig},
SilenceMask,
};
const COEFF_A: [i32; 5] = [14055, 12759, 10733, 12273, 15716];
const COEFF_SUM: [i16; 5] = [22347, 27917, 29523, 29942, 30007];
#[derive(Diff, Patch, Debug, Clone, Copy, PartialEq)]
#[cfg_attr(feature = "bevy_reflect", derive(bevy_reflect::Reflect))]
pub struct PinkNoiseGenNode {
pub volume: Volume,
pub enabled: bool,
}
impl Default for PinkNoiseGenNode {
fn default() -> Self {
Self {
volume: Volume::Linear(0.4),
enabled: true,
}
}
}
#[derive(Debug, Clone)]
#[cfg_attr(feature = "bevy_reflect", derive(bevy_reflect::Reflect))]
pub struct PinkNoiseGenConfig {
pub seed: i32,
pub smooth_secs: f32,
}
impl Default for PinkNoiseGenConfig {
fn default() -> Self {
Self {
seed: 17,
smooth_secs: 10.0 / 1_000.0,
}
}
}
impl AudioNode for PinkNoiseGenNode {
type Configuration = PinkNoiseGenConfig;
fn info(&self, _config: &Self::Configuration) -> AudioNodeInfo {
AudioNodeInfo::new()
.debug_name("pink_noise_gen")
.channel_config(ChannelConfig {
num_inputs: ChannelCount::ZERO,
num_outputs: ChannelCount::MONO,
})
}
fn construct_processor(
&self,
config: &Self::Configuration,
cx: ConstructProcessorContext,
) -> impl AudioNodeProcessor {
let seed = if config.seed == 0 { 17 } else { config.seed };
Processor {
gain: SmoothedParam::new(
self.volume.amp_clamped(DEFAULT_AMP_EPSILON),
SmootherConfig {
smooth_secs: config.smooth_secs,
..Default::default()
},
cx.stream_info.sample_rate,
),
params: *self,
fpd: seed,
contrib: [0; 5],
accum: 0,
}
}
}
struct Processor {
params: PinkNoiseGenNode,
gain: SmoothedParam,
fpd: i32,
contrib: [i32; 5],
accum: i32,
}
impl AudioNodeProcessor for Processor {
fn process(
&mut self,
buffers: ProcBuffers,
_proc_info: &ProcInfo,
events: &mut NodeEventList,
_logger: &mut RealtimeLogger,
) -> ProcessStatus {
for patch in events.drain_patches::<PinkNoiseGenNode>() {
if let PinkNoiseGenNodePatch::Volume(vol) = patch {
self.gain.set_value(vol.amp_clamped(DEFAULT_AMP_EPSILON));
}
self.params.apply(patch);
}
if !self.params.enabled || (self.gain.target_value() == 0.0 && !self.gain.is_smoothing()) {
self.gain.reset();
return ProcessStatus::ClearAllOutputs;
}
for s in buffers.outputs[0].iter_mut() {
let randu: i16 = (rng(&mut self.fpd) & 0x7fff) as i16;
let r_bytes = rng(&mut self.fpd).to_ne_bytes();
let randv: i32 = i16::from_ne_bytes([r_bytes[0], r_bytes[1]]) as i32;
if randu < COEFF_SUM[0] {
update_contrib::<0>(&mut self.accum, &mut self.contrib, randv);
} else if randu < COEFF_SUM[1] {
update_contrib::<1>(&mut self.accum, &mut self.contrib, randv);
} else if randu < COEFF_SUM[2] {
update_contrib::<2>(&mut self.accum, &mut self.contrib, randv);
} else if randu < COEFF_SUM[3] {
update_contrib::<3>(&mut self.accum, &mut self.contrib, randv);
} else if randu < COEFF_SUM[4] {
update_contrib::<4>(&mut self.accum, &mut self.contrib, randv);
}
let r = self.accum as f32 * (1.0 / 2_147_483_648.0);
*s = r * self.gain.next_smoothed();
}
ProcessStatus::OutputsModified {
out_silence_mask: SilenceMask::NONE_SILENT,
}
}
}
#[inline(always)]
fn rng(fpd: &mut i32) -> i32 {
*fpd ^= *fpd << 13;
*fpd ^= *fpd >> 17;
*fpd ^= *fpd << 5;
*fpd
}
#[inline(always)]
fn update_contrib<const I: usize>(accum: &mut i32, contrib: &mut [i32; 5], randv: i32) {
*accum = accum.wrapping_sub(contrib[I]);
contrib[I] = randv * COEFF_A[I];
*accum = accum.wrapping_add(contrib[I]);
}