1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
|
use std::mem;
use std::time::Duration;
use super::*;
use ::byteorder::{ByteOrder, NetworkEndian};
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]},
Unknown{data: [u8; Command::SIZE]},
}
impl Command {
const SIZE: usize = 36;
pub fn duration(&self) -> Option<Duration> {
match *self {
Command::Play{sec, usec, ..} => Some(Duration::new(sec as u64, usec * 1000)),
_ => None,
}
}
pub fn pitch(&self) -> Option<Pitch> {
match *self {
Command::Play{freq, ..} => Some(Pitch::Freq(freq as f32)),
_ => None,
}
}
}
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] = unsafe { mem::uninitialized() };
let mut fields_f32: [f32; Command::SIZE / 4] = unsafe { mem::uninitialized() };
NetworkEndian::read_u32_into(packet, &mut fields_u32);
unsafe { NetworkEndian::read_f32_into_unchecked(packet, &mut fields_f32); }
match fields_u32[0] {
0 => Command::KeepAlive,
1 => {
let mut data: [u8; 32] = unsafe { mem::uninitialized() };
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] = unsafe { mem::uninitialized() };
let mut ident: [u8; 24] = unsafe { mem::uninitialized() };
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] = unsafe { mem::uninitialized() };
::byteorder::LittleEndian::read_i16_into(&packet[4..], &mut samples);
Command::PCM{samples: samples}
},
_ => {
let mut data: [u8; Command::SIZE] = unsafe { mem::uninitialized() };
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];
match *cmd {
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::Unknown{data} => {
ret.copy_from_slice(&data);
},
};
ret
}
}
|