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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
use crate::{sys, mem, ErrorCode, Channels, SampleRate, Bandwidth};
use core::{ptr, num};
use core::convert::TryInto;
use mem::alloc::vec::Vec;
///OPUS Decoder
pub struct Decoder {
pub(crate) inner: mem::Unique<sys::OpusDecoder>,
channels: Channels,
}
impl Decoder {
///Creates new decoder instance
pub fn new(channels: Channels, rate: SampleRate) -> Result<Self, ErrorCode> {
let size = unsafe {
sys::opus_decoder_get_size(channels as _)
};
if size == 0 {
return Err(ErrorCode::Internal);
}
let mut decoder = match mem::Unique::new(size as _) {
Some(inner) => Decoder {
inner,
channels,
},
None => return Err(ErrorCode::AllocFail)
};
let result = unsafe {
sys::opus_decoder_init(decoder.inner.as_mut(), rate as _, channels as _)
};
map_sys_error!(result => decoder)
}
#[inline(always)]
///Returns channels number
///
///When decoding, it is used to determine frame size as `output.len() / channels`
pub fn channels(&self) -> Channels {
self.channels
}
///Decodes input packet, returning number of decoded samples.
///
///If more than 1 channel is configured, then input must be interleaved.
///
///Output size must correspond to sampling rate.
///For example, at 48 kHz allowed frame sizes are 120, 240, 480, 960, 1920, and 2880.
///
///Maximum packet duration is 120ms therefore maximum `frame size` must be
///`frame_bytes_size(SampleRate::Hz48000, Channels::Stereo, 120)`
///
///When `input` size is 0, libopus shall treat it as packet loss, in which case `output` size must
///match expected output of next packet to know how much frames is skipped
///
///When `decode_fec` is `true`, requests that any in-band forward error correction data be decoded.
///If no such data is available, the frame is decoded as if it were lost.
pub fn decode_to(&mut self, input: &[u8], output: &mut [mem::MaybeUninit<u16>], decode_fec: bool) -> Result<usize, ErrorCode> {
let (input_ptr, input_len) = match input.len() {
0 => (ptr::null(), 0),
len => (input.as_ptr(), len as _)
};
let fec = match decode_fec {
true => 1,
false => 0,
};
let result = unsafe {
sys::opus_decode(self.inner.as_mut(),
input_ptr, input_len,
output.as_mut_ptr() as _, (output.len() / self.channels as usize) as _,
fec)
};
map_sys_error!(result => result as _)
}
#[inline(always)]
///Decodes input packet, returning number of decoded samples.
///
///Refer to `decode_to` for details
pub fn decode_to_slice(&mut self, input: &[u8], output: &mut [u16], decode_fec: bool) -> Result<usize, ErrorCode> {
self.decode_to(input, unsafe { mem::transmute(output) }, decode_fec)
}
#[inline(always)]
///Decodes input packet, returning number of decoded samples.
///
///Vector will be written into spare capacity, modifying its length on success.
///
///`decode_len` is used to reserve additional memory and will be passed exactly with this size to `decode_to`
///
///Refer to `decode_to` for details
pub fn decode_to_vec(&mut self, input: &[u8], output: &mut Vec<u16>, decode_len: usize, decode_fec: bool) -> Result<usize, ErrorCode> {
let initial_len = output.len();
if output.try_reserve(decode_len).is_err() {
return Err(ErrorCode::alloc_fail())
}
let result = self.decode_to(input, &mut output.spare_capacity_mut()[..decode_len], decode_fec)?;
unsafe {
output.set_len(initial_len + result);
}
Ok(result)
}
///Decodes input packet, returning number of decoded samples.
///
///If more than 1 channel is configured, then input must be interleaved.
///
///Output size must correspond to sampling rate.
///For example, at 48 kHz allowed frame sizes are 120, 240, 480, 960, 1920, and 2880.
///
///Maximum packet duration is 120ms therefore maximum `frame size` must be
///`frame_bytes_size(SampleRate::Hz48000, Channels::Stereo, 120)`
///
///When `input` size is 0, libopus shall treat it as packet loss, in which case `output` size must
///match expected output of next packet to know how much frames is skipped
///
///When `decode_fec` is `true`, requests that any in-band forward error correction data be decoded.
///If no such data is available, the frame is decoded as if it were lost.
pub fn decode_float_to(&mut self, input: &[u8], output: &mut [mem::MaybeUninit<f32>], decode_fec: bool) -> Result<usize, ErrorCode> {
let (input_ptr, input_len) = match input.len() {
0 => (ptr::null(), 0),
len => (input.as_ptr(), len as _)
};
let fec = match decode_fec {
true => 1,
false => 0,
};
let result = unsafe {
sys::opus_decode_float(self.inner.as_mut(),
input_ptr, input_len,
output.as_mut_ptr() as _, (output.len() / self.channels as usize) as _,
fec)
};
map_sys_error!(result => result as _)
}
#[inline(always)]
///Decodes input packet, returning number of decoded samples.
///
///Refer to `decode_to` for details
pub fn decode_float_to_slice(&mut self, input: &[u8], output: &mut [f32], decode_fec: bool) -> Result<usize, ErrorCode> {
self.decode_float_to(input, unsafe { mem::transmute(output) }, decode_fec)
}
#[inline(always)]
///Decodes input packet, returning number of decoded samples.
///
///Vector will be written into spare capacity, modifying its length on success.
///
///`decode_len` is used to reserve additional memory and will be passed exactly with this size to `decode_to`
///
///Refer to `decode_to` for details
pub fn decode_float_to_vec(&mut self, input: &[u8], output: &mut Vec<f32>, decode_len: usize, decode_fec: bool) -> Result<usize, ErrorCode> {
let initial_len = output.len();
if output.try_reserve(decode_len).is_err() {
return Err(ErrorCode::alloc_fail())
}
let result = self.decode_float_to(input, &mut output.spare_capacity_mut()[..decode_len], decode_fec)?;
unsafe {
output.set_len(initial_len + result);
}
Ok(result)
}
///Gets the number of samples of an Opus packet.
pub fn get_nb_samples(&self, input: &[u8]) -> Result<usize, ErrorCode> {
let len = match input.len().try_into() {
Ok(len) => len,
//This is very unlikely if user gets valid packets fed
Err(_) => return Err(ErrorCode::invalid_packet()),
};
let result = unsafe {
sys::opus_decoder_get_nb_samples(self.inner.as_ptr(), input.as_ptr() as *const _, len)
};
map_sys_error!(result => result as _)
}
#[inline]
///Resets state to initial
pub fn reset(&mut self) -> Result<(), ErrorCode> {
let result = unsafe {
sys::opus_decoder_ctl(self.inner.as_mut(), sys::OPUS_RESET_STATE)
};
map_sys_error!(result => ())
}
#[inline]
///Gets the pitch of the last decoded frame, if available.
///
///This can be used for any post-processing algorithm requiring the use of pitch, e.g. time
///stretching/shortening. If the last frame was not voiced, or if the pitch was not coded in
///the frame, then zero is returned.
pub fn get_pitch(&mut self) -> Result<Option<num::NonZeroU32>, ErrorCode> {
let mut value: i32 = 0;
let result = unsafe {
sys::opus_decoder_ctl(self.inner.as_mut(), sys::OPUS_GET_PITCH_REQUEST, &mut value)
};
map_sys_error!(result => num::NonZeroU32::new(result as _))
}
#[inline]
///Gets the duration (in samples) of the last packet successfully decoded or concealed.
pub fn get_last_packet_duration(&mut self) -> Result<u32, ErrorCode> {
let mut value: i32 = 0;
let result = unsafe {
sys::opus_decoder_ctl(self.inner.as_mut(), sys::OPUS_GET_LAST_PACKET_DURATION_REQUEST, &mut value)
};
map_sys_error!(result => value as _)
}
#[inline]
///Gets the decoder's gain configuration
pub fn get_gain(&mut self) -> Result<i32, ErrorCode> {
let mut value: i32 = 0;
let result = unsafe {
sys::opus_decoder_ctl(self.inner.as_mut(), sys::OPUS_GET_GAIN_REQUEST, &mut value)
};
map_sys_error!(result => value)
}
#[inline]
///Configures decoder gain adjustment.
///
///Scales the decoded output by a factor specified in Q8 dB units.
///This has a maximum range of -32768 to 32767 inclusive, and returns `BadArg` otherwise.
///
///The default is zero indicating no adjustment.
///
///_This setting survives decoder reset_.
///
///Formula:
///
///`gain = pow(10, x/(20.0*256))`
pub fn set_gain(&mut self, value: i32) -> Result<(), ErrorCode> {
let result = unsafe {
sys::opus_decoder_ctl(self.inner.as_mut(), sys::OPUS_SET_GAIN_REQUEST, value)
};
map_sys_error!(result => ())
}
#[cfg(feature = "dred")]
#[inline]
///Configures the decoder's computational complexity.
///
///The supported range is 0-10 inclusive with 10 representing the highest complexity.
///Values of 5 or above will use deep packet loss concealment.
pub fn set_complexity(&mut self, value: u8) -> Result<(), ErrorCode> {
let result = unsafe {
sys::opus_decoder_ctl(
self.inner.as_mut(),
sys::OPUS_SET_COMPLEXITY_REQUEST,
value as i32,
)
};
map_sys_error!(result => ())
}
#[cfg(feature = "dred")]
#[inline]
///Gets the decoder's complexity configuration.
pub fn get_complexity(&mut self) -> Result<u8, ErrorCode> {
let mut value: i32 = 0;
let result = unsafe {
sys::opus_decoder_ctl(self.inner.as_mut(), sys::OPUS_GET_COMPLEXITY_REQUEST, &mut value)
};
map_sys_error!(result => value as u8)
}
#[inline]
///Gets the decoder's last bandpass
pub fn get_bandwidth(&mut self) -> Result<Bandwidth, ErrorCode> {
let mut value: i32 = 0;
let result = unsafe {
sys::opus_decoder_ctl(self.inner.as_mut(), sys::OPUS_GET_BANDWIDTH_REQUEST, &mut value)
};
map_sys_error!(result => value.into())
}
#[inline]
///Gets configured sample rate of this instance
pub fn get_sample_rate(&mut self) -> Result<SampleRate, ErrorCode> {
let mut value: i32 = 0;
let result = unsafe {
sys::opus_decoder_ctl(self.inner.as_mut(), sys::OPUS_GET_SAMPLE_RATE_REQUEST, &mut value)
};
map_sys_error!(result => match value {
8000 => SampleRate::Hz8000,
12000 => SampleRate::Hz12000,
16000 => SampleRate::Hz16000,
24000 => SampleRate::Hz24000,
48000 => SampleRate::Hz48000,
_ => return Err(ErrorCode::unknown())
})
}
#[inline]
///Gets the decoder's configured phase inversion status.
pub fn get_phase_inversion_disabled(&mut self) -> Result<bool, ErrorCode> {
let mut value: i32 = 0;
let result = unsafe {
sys::opus_decoder_ctl(self.inner.as_mut(), sys::OPUS_GET_PHASE_INVERSION_DISABLED_REQUEST, &mut value)
};
map_sys_error!(result => value == 1)
}
#[inline]
///Configures phase inversion.
///
///If set to `true`, disables the use of phase inversion for intensity stereo, improving the quality
///of mono downmixes, but slightly reducing normal stereo quality.
///
///Disabling phase inversion in the decoder does not comply with RFC 6716, although it does not
///cause any interoperability issue and is expected to become part of the Opus standard once
///RFC 6716 is updated by draft-ietf-codec-opus-update.
pub fn set_phase_inversion_disabled(&mut self, value: bool) -> Result<(), ErrorCode> {
let value: i32 = match value {
true => 1,
false => 0,
};
let result = unsafe {
sys::opus_decoder_ctl(self.inner.as_mut(), sys::OPUS_SET_PHASE_INVERSION_DISABLED_REQUEST, value)
};
map_sys_error!(result => ())
}
}
unsafe impl Send for Decoder {}