use super::Pitch; use std::fmt; use std::time::Duration; use ::byteorder::{ByteOrder, NetworkEndian}; #[allow(dead_code)] const OBLIGATE_POLYPHONE: u32 = 0xffffffff; pub enum Command { KeepAlive, Ping { data: [u8; 32], }, Quit, Play { sec: u32, usec: u32, freq: u32, amp: f32, voice: u32, }, Caps { voices: u32, tp: [u8; 4], ident: [u8; 24], }, PCM { samples: [i16; 16], }, PCMSyn { buffered: u32, }, ArtParam { voice: Option, index: u32, value: f32, }, Unknown { data: [u8; Command::SIZE], }, } impl Command { pub const SIZE: usize = 36; pub fn duration(&self) -> Option { match *self { Command::Play { sec, usec, .. } => Some(Duration::new(sec as u64, usec * 1000)), _ => None, } } pub fn pitch(&self) -> Option { match *self { Command::Play { freq, .. } => Some(Pitch::Freq(freq as f32)), _ => None, } } pub fn write_into(&self, ret: &mut [u8]) -> bool { if ret.len() < Command::SIZE { return false; } match *self { Command::KeepAlive => NetworkEndian::write_u32(&mut ret[..4], 0), Command::Ping { data } => { NetworkEndian::write_u32(&mut ret[..4], 1); (&mut ret[4..]).copy_from_slice(&data); } Command::Quit => NetworkEndian::write_u32(&mut ret[..4], 2), Command::Play { sec, usec, freq, amp, voice, } => { NetworkEndian::write_u32_into(&[3u32, sec, usec, freq], &mut ret[..16]); NetworkEndian::write_f32(&mut ret[16..20], amp); NetworkEndian::write_u32(&mut ret[20..24], voice); } Command::Caps { voices, tp, ident } => { NetworkEndian::write_u32_into(&[4u32, voices], &mut ret[..8]); (&mut ret[8..12]).copy_from_slice(&tp); (&mut ret[12..]).copy_from_slice(&ident); } Command::PCM { samples } => { NetworkEndian::write_u32(&mut ret[..4], 5); NetworkEndian::write_i16_into(&samples, &mut ret[4..]); } Command::PCMSyn { buffered } => { NetworkEndian::write_u32_into(&[6u32, buffered], &mut ret[..8]); } Command::ArtParam { voice, index, value, } => { NetworkEndian::write_u32_into( &[7u32, voice.unwrap_or(OBLIGATE_POLYPHONE), index], &mut ret[..12], ); NetworkEndian::write_f32(&mut ret[12..16], value); } Command::Unknown { data } => { ret.copy_from_slice(&data); } }; true } } impl fmt::Debug for Command { fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> { f.write_str("Command::")?; match *self { Command::KeepAlive => f.write_str("KeepAlive"), Command::Ping { data } => f.debug_struct("Ping").field("data", &data).finish(), Command::Quit => f.write_str("Quit"), Command::Play { sec, usec, freq, amp, voice, } => f .debug_struct("Play") .field("sec", &sec) .field("usec", &usec) .field("freq", &freq) .field("amp", &) .field("voice", &voice) .finish(), Command::Caps { voices, tp, ident } => f .debug_struct("Caps") .field("voices", &voices) .field("tp", &tp) .field("ident", &ident) .finish(), Command::PCM { samples } => f.debug_struct("PCM").field("samples", &samples).finish(), Command::PCMSyn { buffered } => f .debug_struct("PCMSyn") .field("buffered", &buffered) .finish(), Command::ArtParam { voice, index, value, } => f .debug_struct("ArtParam") .field("voice", &voice) .field("index", &index) .field("value", &value) .finish(), Command::Unknown { data } => f .debug_struct("Unknown") .field("data", &(&data as &[u8])) .finish(), } } } impl<'a> From<&'a [u8; Command::SIZE]> for Command { fn from(packet: &'a [u8; Command::SIZE]) -> Command { let mut fields_u32: [u32; Command::SIZE / 4] = [0; Command::SIZE / 4]; let mut fields_f32: [f32; Command::SIZE / 4] = [0.0; Command::SIZE / 4]; NetworkEndian::read_u32_into(packet, &mut fields_u32); NetworkEndian::read_f32_into(packet, &mut fields_f32); match fields_u32[0] { 0 => Command::KeepAlive, 1 => { let mut data: [u8; 32] = [0; 32]; data.copy_from_slice(&packet[4..]); Command::Ping { data: data } } 2 => Command::Quit, 3 => Command::Play { sec: fields_u32[1], usec: fields_u32[2], freq: fields_u32[3], amp: fields_f32[4], voice: fields_u32[5], }, 4 => { let mut tp: [u8; 4] = [0; 4]; let mut ident: [u8; 24] = [0; 24]; tp.copy_from_slice(&packet[8..12]); ident.copy_from_slice(&packet[12..]); Command::Caps { voices: fields_u32[1], tp: tp, ident: ident, } } 5 => { let mut samples: [i16; 16] = [0; 16]; ::byteorder::LittleEndian::read_i16_into(&packet[4..], &mut samples); Command::PCM { samples: samples } } 6 => Command::PCMSyn { buffered: fields_u32[1], }, 7 => Command::ArtParam { voice: if fields_u32[1] == OBLIGATE_POLYPHONE { None } else { Some(fields_u32[1]) }, index: fields_u32[2], value: fields_f32[3], }, _ => { let mut data: [u8; Command::SIZE] = [0; Command::SIZE]; data.copy_from_slice(packet); Command::Unknown { data: data } } } } } impl<'a> From<&'a Command> for [u8; Command::SIZE] { fn from(cmd: &'a Command) -> [u8; Command::SIZE] { let mut ret: [u8; Command::SIZE] = [0u8; Command::SIZE]; cmd.write_into(&mut ret); ret } }