summaryrefslogtreecommitdiff
path: root/src/client.rs
diff options
context:
space:
mode:
authorGraham Northup <grissess@nexusg.org>2017-10-01 07:04:21 -0400
committerGraham Northup <grissess@nexusg.org>2017-10-01 07:04:21 -0400
commit145e2771c0d1ad30748da6e6ef1fabbd4cc2478c (patch)
treeb9e96a57df0d515383a7c09e3b2ea71edb31db96 /src/client.rs
parent629d2fa754dcd4bbdbd1e84ea9f7598806abc840 (diff)
It plays network noises now!
Diffstat (limited to 'src/client.rs')
-rw-r--r--src/client.rs116
1 files changed, 72 insertions, 44 deletions
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<GenBox>, env: Environment) -> io::Result<Client> {
- 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<u8>) -> 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<u8>) {
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;
}
}