fuchsia_audio_device/
types.rsuse fidl_fuchsia_hardware_audio::{PcmFormat, SampleFormat};
use std::result;
use thiserror::Error;
pub type Result<T> = result::Result<T, Error>;
#[derive(Error, Debug)]
pub enum Error {
#[error("Value was out of range")]
OutOfRange,
#[error("Invalid Header for a message")]
InvalidHeader,
#[error("Encoding error")]
Encoding,
#[error("Encountered an IO error reading from the channel: {}", _0)]
PeerRead(zx::Status),
#[error("Encountered an IO error writing to the channel: {}", _0)]
PeerWrite(zx::Status),
#[error("Encountered an IO error: {}", _0)]
IOError(zx::Status),
#[error("Encountered an error on a RequestStream: {}", _0)]
RequestStreamError(#[from] fidl::Error),
#[error("Peer performed an invalid action: {}", _0)]
PeerError(String),
#[error("Tried to do an action in an invalid state")]
InvalidState,
#[error("No channel found for reply")]
NoChannel,
#[error("Message has not been implemented yet")]
UnimplementedMessage,
#[error("Invalid argument")]
InvalidArgs,
#[doc(hidden)]
#[error("__Nonexhaustive error should never be created.")]
__Nonexhaustive,
}
#[derive(Debug, PartialEq, Clone)]
pub enum AudioSampleFormat {
Eight { unsigned: bool },
Sixteen { unsigned: bool, invert_endian: bool },
TwentyFourPacked { unsigned: bool, invert_endian: bool },
TwentyIn32 { unsigned: bool, invert_endian: bool },
TwentyFourIn32 { unsigned: bool, invert_endian: bool },
ThirtyTwo { unsigned: bool, invert_endian: bool },
Float { invert_endian: bool },
}
impl AudioSampleFormat {
fn is_unsigned(&self) -> bool {
use AudioSampleFormat::*;
match self {
Eight { unsigned }
| Sixteen { unsigned, .. }
| TwentyFourPacked { unsigned, .. }
| TwentyIn32 { unsigned, .. }
| TwentyFourIn32 { unsigned, .. }
| ThirtyTwo { unsigned, .. } => *unsigned,
Float { .. } => false,
}
}
}
impl From<PcmFormat> for AudioSampleFormat {
fn from(v: PcmFormat) -> Self {
if let SampleFormat::PcmFloat = v.sample_format {
if v.bytes_per_sample == 32 && v.valid_bits_per_sample == 32 {
AudioSampleFormat::Float { invert_endian: false }
} else {
panic!("audio sample format not supported");
}
} else {
let is_unsigned = v.sample_format == SampleFormat::PcmUnsigned;
match v.bytes_per_sample {
1u8 => {
assert_eq!(v.valid_bits_per_sample, 8u8);
AudioSampleFormat::Eight { unsigned: is_unsigned }
}
2u8 => {
assert_eq!(v.valid_bits_per_sample, 16u8);
AudioSampleFormat::Sixteen { unsigned: is_unsigned, invert_endian: false }
}
3u8 => {
assert_eq!(v.valid_bits_per_sample, 24u8);
AudioSampleFormat::TwentyFourPacked {
unsigned: is_unsigned,
invert_endian: false,
}
}
4u8 => match v.valid_bits_per_sample {
20u8 => AudioSampleFormat::TwentyIn32 {
unsigned: is_unsigned,
invert_endian: false,
},
24u8 => AudioSampleFormat::TwentyFourIn32 {
unsigned: is_unsigned,
invert_endian: false,
},
32u8 => {
AudioSampleFormat::ThirtyTwo { unsigned: is_unsigned, invert_endian: false }
}
_ => panic!(
"audio valid bits per sample {:?} not supported",
v.valid_bits_per_sample
),
},
_ => panic!("audio bytes per samples {:?} not supported", v.bytes_per_sample),
}
}
}
}
impl AudioSampleFormat {
pub fn compute_frame_size(&self, channels: usize) -> Result<usize> {
let bytes_per_channel = match self {
AudioSampleFormat::Eight { .. } => 1,
AudioSampleFormat::Sixteen { .. } => 2,
AudioSampleFormat::TwentyFourPacked { .. } => 3,
AudioSampleFormat::TwentyIn32 { .. }
| AudioSampleFormat::TwentyFourIn32 { .. }
| AudioSampleFormat::ThirtyTwo { .. }
| AudioSampleFormat::Float { .. } => 4,
};
Ok(channels * bytes_per_channel)
}
}
impl std::fmt::Display for AudioSampleFormat {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
if self.is_unsigned() {
f.write_str("u")?;
} else {
f.write_str("i")?;
}
use AudioSampleFormat::*;
match self {
Eight { .. } => f.write_str("8"),
Sixteen { .. } => f.write_str("16"),
TwentyFourPacked { .. } => f.write_str("24p"),
TwentyIn32 { .. } => f.write_str("20(32)"),
TwentyFourIn32 { .. } => f.write_str("24(32)"),
ThirtyTwo { .. } => f.write_str("32"),
Float { .. } => f.write_str("float"),
}
}
}