From 145e2771c0d1ad30748da6e6ef1fabbd4cc2478c Mon Sep 17 00:00:00 2001 From: Graham Northup Date: Sun, 1 Oct 2017 07:04:21 -0400 Subject: It plays network noises now! --- src/client.rs | 116 ++++++++++++++++++++++++++++++++++++---------------------- 1 file changed, 72 insertions(+), 44 deletions(-) (limited to 'src/client.rs') diff --git a/src/client.rs b/src/client.rs index 1a0c69f..4615edd 100644 --- a/src/client.rs +++ b/src/client.rs @@ -1,4 +1,4 @@ -use std::net::{UdpSocket}; +use std::net::{UdpSocket, SocketAddr}; use std::{io, mem}; use synth::*; @@ -15,6 +15,7 @@ pub struct Client { pub env: Environment, pub frames: usize, pub buf: SampleBuffer, + norm: SampleBuffer } macro_rules! dprintln { @@ -23,7 +24,6 @@ macro_rules! dprintln { impl Client { pub fn new(socket: UdpSocket, gens: Vec, env: Environment) -> io::Result { - socket.set_nonblocking(true)?; let buf = SampleBuffer::new(env.default_buffer_size); let voices = gens.into_iter().map(|g| Voice { gen: g, params: Parameters { env: env.clone(), ..Default::default() } }).collect(); Ok(Client { @@ -32,10 +32,13 @@ impl Client { env: env, frames: 0, buf: buf, + norm: SampleBuffer::new(1), }) } - pub fn pump(&mut self, out_buffer: &mut Vec) -> bool { + // NB: Loops indefinitely (until timeout, quit, or error) iff self.socket blocks + /* + pub fn process_packets(&mut self) -> bool { if self.voices.len() == 0 { return false; } @@ -51,45 +54,8 @@ impl Client { } let cmd = Command::from(&buffer); - dprintln!("Packet {:?} from {:?}", (&buffer as &[u8]), sender); - match cmd { - Command::KeepAlive => {}, - Command::Ping{..} => { - self.socket.send_to(&buffer, sender); - }, - Command::Quit => { - return false; - }, - Command::Play{voice, freq, amp, ..} => { - if (voice as usize) >= self.voices.len() { - dprintln!("Dropping packet: tried to send to voice {} >= number of voices {}", voice, self.voices.len()); - continue; - } - let dur = cmd.duration().unwrap(); - let frac_secs = (dur.as_secs() as f32) + (dur.subsec_nanos() as f32) / 1.0e9; - let frames = frac_secs * (self.env.sample_rate as f32); - - dprintln!("Playing on voice {} freq {} amp {} from frame {} until frame {}", voice, freq, amp, self.frames, (self.frames as f32) + frames); - - let mut vars = &mut self.voices[voice as usize].params.vars; - *vars.entry("v_deadline".to_string()).or_insert_with(Default::default) = (self.frames as f32) + frames; - *vars.entry("v_freq".to_string()).or_insert_with(Default::default) = freq as f32; - *vars.entry("v_amp".to_string()).or_insert_with(Default::default) = amp; - }, - Command::Caps{..} => { - let reply = Command::Caps { - voices: self.voices.len() as u32, - tp: ['S' as u8, 'Y' as u8, 'N' as u8, 'F' as u8], - ident: [0u8; 24], - }; - let mut reply_buffer: [u8; Command::SIZE] = [0u8; Command::SIZE]; - reply.write_into(&mut reply_buffer); - self.socket.send_to(&reply_buffer, sender); - }, - Command::PCM{..} => { /* TODO */ }, - Command::Unknown{data} => { - dprintln!("Dropping packet: unknown data {:?}", (&data as &[u8])); - }, + if !self.handle_command(cmd, sender) { + return false; } }, Err(err) => { @@ -101,6 +67,60 @@ impl Client { } } + true + } + */ + + pub fn handle_command(&mut self, cmd: Command, sender: SocketAddr) -> bool { + dprintln!("Packet {:?} from {:?}", cmd, sender); + match cmd { + Command::KeepAlive => {}, + Command::Ping{..} => { + let mut reply_buffer: [u8; Command::SIZE] = [0u8; Command::SIZE]; + cmd.write_into(&mut reply_buffer); + self.socket.send_to(&reply_buffer, sender); + }, + Command::Quit => { + return false; + }, + Command::Play{voice, freq, amp, ..} => { + if (voice as usize) >= self.voices.len() { + dprintln!("Dropping packet: tried to send to voice {} >= number of voices {}", voice, self.voices.len()); + return true; + } + let dur = cmd.duration().unwrap(); + let frac_secs = (dur.as_secs() as f32) + (dur.subsec_nanos() as f32) / 1.0e9; + let frames = frac_secs * (self.env.sample_rate as f32); + + dprintln!("Playing on voice {} freq {} amp {} from frame {} until frame {}", voice, freq, amp, self.frames, (self.frames as f32) + frames); + + let mut vars = &mut self.voices[voice as usize].params.vars; + *vars.entry("v_deadline".to_string()).or_insert_with(Default::default) = (self.frames as f32) + frames; + *vars.entry("v_freq".to_string()).or_insert_with(Default::default) = freq as f32; + *vars.entry("v_amp".to_string()).or_insert_with(Default::default) = amp; + }, + Command::Caps{..} => { + let reply = Command::Caps { + voices: self.voices.len() as u32, + tp: ['S' as u8, 'Y' as u8, 'N' as u8, 'F' as u8], + ident: [0u8; 24], + }; + let mut reply_buffer: [u8; Command::SIZE] = [0u8; Command::SIZE]; + reply.write_into(&mut reply_buffer); + self.socket.send_to(&reply_buffer, sender); + }, + Command::PCM{..} => { /* TODO */ }, + Command::Unknown{data} => { + dprintln!("Dropping packet: unknown data {:?}", (&data as &[u8])); + }, + } + + true + } + + pub fn next_frames(&mut self) { + let len = self.voices.len(); + for voice in self.voices.iter_mut() { *voice.params.vars.entry("v_frame".to_string()).or_insert_with(Default::default) = self.frames as f32; } @@ -112,11 +132,19 @@ impl Client { self.buf.sum_into(voice.gen.eval(&voice.params)); } + self.norm.set(1.0 / (len as f32)); + self.buf.mul_into(&self.norm); + self.frames += self.buf.len(); + } + + pub fn buffer(&self) -> &SampleBuffer { + &self.buf + } + + pub fn write_frames_bytes(&self, out_buffer: &mut Vec) { let current = out_buffer.len(); out_buffer.reserve_exact(self.buf.size() - current); unsafe { out_buffer.set_len(self.buf.size()); } self.buf.write_bytes(out_buffer); - self.frames += self.buf.len(); - return true; } } -- cgit v1.2.3-70-g09d2