2017-08-08 22:29:49 +00:00
|
|
|
// Copyright (C) 2017 Sebastian Dröge <sebastian@centricular.com>
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2020-06-05 09:57:29 +00:00
|
|
|
use std::cmp::Ordering;
|
2017-08-08 22:29:49 +00:00
|
|
|
use std::ffi::CStr;
|
|
|
|
use std::fmt;
|
|
|
|
use std::str;
|
|
|
|
|
2017-08-09 22:13:47 +00:00
|
|
|
use glib::translate::{from_glib, FromGlib, FromGlibPtrNone, ToGlib, ToGlibPtr, ToGlibPtrMut};
|
2017-08-08 22:29:49 +00:00
|
|
|
|
2017-11-26 21:50:39 +00:00
|
|
|
#[derive(PartialEq, Eq, Copy, Clone, Debug, Hash)]
|
2017-08-08 22:29:49 +00:00
|
|
|
pub enum AudioEndianness {
|
|
|
|
Unknown,
|
|
|
|
LittleEndian = 1234,
|
|
|
|
BigEndian = 4321,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromGlib<i32> for AudioEndianness {
|
|
|
|
fn from_glib(value: i32) -> Self {
|
2017-08-30 11:39:09 +00:00
|
|
|
assert_initialized_main_thread!();
|
|
|
|
|
2017-08-08 22:29:49 +00:00
|
|
|
match value {
|
|
|
|
1234 => AudioEndianness::LittleEndian,
|
|
|
|
4321 => AudioEndianness::BigEndian,
|
|
|
|
_ => AudioEndianness::Unknown,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ToGlib for AudioEndianness {
|
|
|
|
type GlibType = i32;
|
|
|
|
|
|
|
|
fn to_glib(&self) -> i32 {
|
|
|
|
match *self {
|
|
|
|
AudioEndianness::LittleEndian => 1234,
|
|
|
|
AudioEndianness::BigEndian => 4321,
|
|
|
|
_ => 0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
pub struct AudioFormatInfo(&'static ffi::GstAudioFormatInfo);
|
2017-08-08 22:29:49 +00:00
|
|
|
|
|
|
|
impl AudioFormatInfo {
|
2020-11-21 18:17:20 +00:00
|
|
|
pub fn from_format(format: crate::AudioFormat) -> AudioFormatInfo {
|
2017-08-30 11:39:09 +00:00
|
|
|
assert_initialized_main_thread!();
|
|
|
|
|
2017-08-08 22:29:49 +00:00
|
|
|
unsafe {
|
2020-11-21 18:17:20 +00:00
|
|
|
let info = ffi::gst_audio_format_get_info(format.to_glib());
|
2017-08-08 22:29:49 +00:00
|
|
|
assert!(!info.is_null());
|
|
|
|
|
|
|
|
AudioFormatInfo(&*info)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
pub fn format(&self) -> crate::AudioFormat {
|
2017-08-08 22:29:49 +00:00
|
|
|
from_glib(self.0.format)
|
|
|
|
}
|
|
|
|
|
2017-09-01 08:40:32 +00:00
|
|
|
pub fn name<'a>(&self) -> &'a str {
|
2017-08-08 22:29:49 +00:00
|
|
|
unsafe { CStr::from_ptr(self.0.name).to_str().unwrap() }
|
|
|
|
}
|
|
|
|
|
2017-09-01 08:40:32 +00:00
|
|
|
pub fn description<'a>(&self) -> &'a str {
|
2017-08-08 22:29:49 +00:00
|
|
|
unsafe { CStr::from_ptr(self.0.description).to_str().unwrap() }
|
|
|
|
}
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
pub fn flags(&self) -> crate::AudioFormatFlags {
|
2017-08-08 22:29:49 +00:00
|
|
|
from_glib(self.0.flags)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn endianness(&self) -> AudioEndianness {
|
|
|
|
from_glib(self.0.endianness)
|
|
|
|
}
|
|
|
|
|
2017-08-10 21:41:36 +00:00
|
|
|
pub fn width(&self) -> u32 {
|
|
|
|
self.0.width as u32
|
2017-08-08 22:29:49 +00:00
|
|
|
}
|
|
|
|
|
2017-08-10 21:41:36 +00:00
|
|
|
pub fn depth(&self) -> u32 {
|
|
|
|
self.0.depth as u32
|
2017-08-08 22:29:49 +00:00
|
|
|
}
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
pub fn unpack_format(&self) -> crate::AudioFormat {
|
2017-08-08 22:29:49 +00:00
|
|
|
from_glib(self.0.unpack_format)
|
|
|
|
}
|
|
|
|
|
2017-09-01 08:40:32 +00:00
|
|
|
pub fn silence<'a>(&self) -> &'a [u8] {
|
2017-08-08 22:29:49 +00:00
|
|
|
&self.0.silence
|
|
|
|
}
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
pub fn unpack(&self, flags: crate::AudioPackFlags, dest: &mut [u8], src: &[u8]) {
|
2017-08-08 22:29:49 +00:00
|
|
|
let unpack_format = Self::from_format(self.unpack_format());
|
|
|
|
let unpack_width = unpack_format.width() as usize;
|
|
|
|
|
|
|
|
if unpack_width == 0 || self.0.unpack_func.is_none() {
|
|
|
|
panic!("No unpack format for {:?}", self);
|
|
|
|
}
|
|
|
|
|
|
|
|
let self_width = self.width() as usize;
|
|
|
|
if self_width == 0 {
|
|
|
|
panic!("No width for {:?}", self);
|
|
|
|
}
|
|
|
|
|
|
|
|
if src.len() % (self_width / 8) != 0 {
|
|
|
|
panic!("Incomplete number of samples in src");
|
|
|
|
}
|
|
|
|
|
|
|
|
let nsamples = src.len() / (self_width / 8);
|
|
|
|
|
|
|
|
if dest.len() != nsamples * (unpack_width / 8) {
|
|
|
|
panic!("Invalid dest length");
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
(self.0.unpack_func.as_ref().unwrap())(
|
|
|
|
self.0,
|
|
|
|
flags.to_glib(),
|
|
|
|
dest.as_mut_ptr() as *mut _,
|
2019-05-26 16:31:21 +00:00
|
|
|
src.as_ptr() as *const _,
|
2017-08-08 22:29:49 +00:00
|
|
|
nsamples as i32,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
pub fn pack(&self, flags: crate::AudioPackFlags, dest: &mut [u8], src: &[u8]) {
|
2017-08-08 22:29:49 +00:00
|
|
|
let unpack_format = Self::from_format(self.unpack_format());
|
|
|
|
let unpack_width = unpack_format.width() as usize;
|
|
|
|
|
|
|
|
if unpack_width == 0 || self.0.pack_func.is_none() {
|
|
|
|
panic!("No unpack format for {:?}", self);
|
|
|
|
}
|
|
|
|
|
|
|
|
let self_width = self.width() as usize;
|
|
|
|
if self_width == 0 {
|
|
|
|
panic!("No width for {:?}", self);
|
|
|
|
}
|
|
|
|
|
|
|
|
if src.len() % (unpack_width / 8) != 0 {
|
|
|
|
panic!("Incomplete number of samples in src");
|
|
|
|
}
|
|
|
|
|
|
|
|
let nsamples = src.len() / (unpack_width / 8);
|
|
|
|
|
|
|
|
if dest.len() != nsamples * (self_width / 8) {
|
|
|
|
panic!("Invalid dest length");
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
(self.0.pack_func.as_ref().unwrap())(
|
|
|
|
self.0,
|
|
|
|
flags.to_glib(),
|
2019-05-26 16:31:21 +00:00
|
|
|
src.as_ptr() as *const _,
|
2017-08-08 22:29:49 +00:00
|
|
|
dest.as_mut_ptr() as *mut _,
|
|
|
|
nsamples as i32,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn fill_silence(&self, dest: &mut [u8]) {
|
|
|
|
let self_width = self.width() as usize;
|
|
|
|
|
|
|
|
if self_width == 0 {
|
|
|
|
panic!("Filling with silence unsupported");
|
|
|
|
}
|
|
|
|
|
|
|
|
if dest.len() % (self_width / 8) != 0 {
|
|
|
|
panic!("Incomplete number of samples in dest");
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe {
|
2020-11-21 18:17:20 +00:00
|
|
|
ffi::gst_audio_format_fill_silence(self.0, dest.as_mut_ptr() as *mut _, dest.len())
|
2017-08-08 22:29:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_float(&self) -> bool {
|
2020-11-21 18:17:20 +00:00
|
|
|
self.flags().contains(crate::AudioFormatFlags::FLOAT)
|
2017-08-08 22:29:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_integer(&self) -> bool {
|
2020-11-21 18:17:20 +00:00
|
|
|
self.flags().contains(crate::AudioFormatFlags::INTEGER)
|
2017-08-08 22:29:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_signed(&self) -> bool {
|
2020-11-21 18:17:20 +00:00
|
|
|
self.flags().contains(crate::AudioFormatFlags::SIGNED)
|
2017-08-08 22:29:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_little_endian(&self) -> bool {
|
|
|
|
self.endianness() == AudioEndianness::LittleEndian
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_big_endian(&self) -> bool {
|
|
|
|
self.endianness() == AudioEndianness::BigEndian
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe impl Sync for AudioFormatInfo {}
|
|
|
|
unsafe impl Send for AudioFormatInfo {}
|
|
|
|
|
|
|
|
impl PartialEq for AudioFormatInfo {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
self.format() == other.format()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Eq for AudioFormatInfo {}
|
|
|
|
|
2020-06-05 09:57:29 +00:00
|
|
|
impl PartialOrd for AudioFormatInfo {
|
|
|
|
fn partial_cmp(&self, other: &AudioFormatInfo) -> Option<Ordering> {
|
|
|
|
Some(self.cmp(other))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Ord for AudioFormatInfo {
|
|
|
|
// See GST_AUDIO_FORMATS_ALL for the sorting algorithm
|
|
|
|
fn cmp(&self, other: &AudioFormatInfo) -> Ordering {
|
|
|
|
self.depth()
|
|
|
|
.cmp(&other.depth())
|
|
|
|
.then_with(|| self.width().cmp(&other.width()))
|
|
|
|
.then_with(|| {
|
|
|
|
match (
|
2020-11-21 18:17:20 +00:00
|
|
|
self.flags().contains(crate::AudioFormatFlags::FLOAT),
|
|
|
|
other.flags().contains(crate::AudioFormatFlags::FLOAT),
|
2020-06-05 09:57:29 +00:00
|
|
|
) {
|
|
|
|
(true, false) => Ordering::Greater,
|
|
|
|
(false, true) => Ordering::Less,
|
|
|
|
_ => Ordering::Equal,
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.then_with(|| {
|
|
|
|
match (
|
2020-11-21 18:17:20 +00:00
|
|
|
self.flags().contains(crate::AudioFormatFlags::SIGNED),
|
|
|
|
other.flags().contains(crate::AudioFormatFlags::SIGNED),
|
2020-06-05 09:57:29 +00:00
|
|
|
) {
|
|
|
|
(true, false) => Ordering::Greater,
|
|
|
|
(false, true) => Ordering::Less,
|
|
|
|
_ => Ordering::Equal,
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.then_with(|| match (self.endianness(), other.endianness()) {
|
2020-11-21 18:17:20 +00:00
|
|
|
(crate::AudioEndianness::LittleEndian, crate::AudioEndianness::BigEndian) => {
|
2020-06-05 09:57:29 +00:00
|
|
|
#[cfg(target_endian = "little")]
|
|
|
|
{
|
|
|
|
Ordering::Greater
|
|
|
|
}
|
|
|
|
#[cfg(target_endian = "big")]
|
|
|
|
{
|
|
|
|
Ordering::Less
|
|
|
|
}
|
|
|
|
}
|
2020-11-21 18:17:20 +00:00
|
|
|
(crate::AudioEndianness::BigEndian, crate::AudioEndianness::LittleEndian) => {
|
2020-06-05 09:57:29 +00:00
|
|
|
#[cfg(target_endian = "little")]
|
|
|
|
{
|
|
|
|
Ordering::Less
|
|
|
|
}
|
|
|
|
#[cfg(target_endian = "big")]
|
|
|
|
{
|
|
|
|
Ordering::Greater
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => Ordering::Equal,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-08 22:29:49 +00:00
|
|
|
impl fmt::Debug for AudioFormatInfo {
|
2020-11-28 11:33:46 +00:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2019-07-03 12:23:09 +00:00
|
|
|
f.debug_struct("AudioFormatInfo")
|
|
|
|
.field("format", &self.format())
|
|
|
|
.field("name", &self.name())
|
|
|
|
.field("description", &self.description())
|
|
|
|
.field("flags", &self.flags())
|
|
|
|
.field("endianness", &self.endianness())
|
|
|
|
.field("width", &self.width())
|
|
|
|
.field("depth", &self.depth())
|
|
|
|
.field("silence", &self.silence())
|
|
|
|
.finish()
|
2017-08-08 22:29:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-10 08:14:08 +00:00
|
|
|
impl fmt::Display for AudioFormatInfo {
|
2020-11-28 11:33:46 +00:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2017-08-10 08:14:08 +00:00
|
|
|
f.write_str(self.name())
|
2017-08-08 22:29:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
impl str::FromStr for crate::AudioFormatInfo {
|
2019-12-17 19:00:42 +00:00
|
|
|
type Err = glib::BoolError;
|
2017-08-08 22:29:49 +00:00
|
|
|
|
2019-12-17 19:00:42 +00:00
|
|
|
fn from_str(s: &str) -> Result<Self, glib::BoolError> {
|
2017-08-30 11:39:09 +00:00
|
|
|
skip_assert_initialized!();
|
2017-08-10 08:14:08 +00:00
|
|
|
let format = s.parse()?;
|
|
|
|
Ok(AudioFormatInfo::from_format(format))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
impl From<crate::AudioFormat> for AudioFormatInfo {
|
|
|
|
fn from(f: crate::AudioFormat) -> Self {
|
2017-08-30 11:39:09 +00:00
|
|
|
skip_assert_initialized!();
|
2017-08-10 08:14:08 +00:00
|
|
|
Self::from_format(f)
|
2017-08-08 22:29:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-09 22:13:47 +00:00
|
|
|
impl glib::types::StaticType for AudioFormatInfo {
|
|
|
|
fn static_type() -> glib::types::Type {
|
2020-11-21 18:17:20 +00:00
|
|
|
unsafe { glib::translate::from_glib(ffi::gst_audio_format_info_get_type()) }
|
2017-08-09 22:13:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
impl<'a> glib::value::FromValueOptional<'a> for AudioFormatInfo {
|
|
|
|
unsafe fn from_value_optional(value: &glib::Value) -> Option<Self> {
|
2020-11-21 18:17:20 +00:00
|
|
|
Option::<AudioFormatInfo>::from_glib_none(glib::gobject_ffi::g_value_get_boxed(
|
2017-12-20 17:30:14 +00:00
|
|
|
value.to_glib_none().0,
|
2020-11-21 18:17:20 +00:00
|
|
|
) as *mut ffi::GstAudioFormatInfo)
|
2017-08-09 22:13:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
impl glib::value::SetValue for AudioFormatInfo {
|
|
|
|
unsafe fn set_value(value: &mut glib::Value, this: &Self) {
|
2020-11-21 18:17:20 +00:00
|
|
|
glib::gobject_ffi::g_value_set_boxed(
|
2017-08-09 22:13:47 +00:00
|
|
|
value.to_glib_none_mut().0,
|
2020-11-21 18:17:20 +00:00
|
|
|
glib::translate::ToGlibPtr::<*const ffi::GstAudioFormatInfo>::to_glib_none(this).0
|
|
|
|
as glib::ffi::gpointer,
|
2017-08-09 22:13:47 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
impl glib::value::SetValueOptional for AudioFormatInfo {
|
|
|
|
unsafe fn set_value_optional(value: &mut glib::Value, this: Option<&Self>) {
|
2020-11-21 18:17:20 +00:00
|
|
|
glib::gobject_ffi::g_value_set_boxed(
|
2017-08-09 22:13:47 +00:00
|
|
|
value.to_glib_none_mut().0,
|
2020-11-21 18:17:20 +00:00
|
|
|
glib::translate::ToGlibPtr::<*const ffi::GstAudioFormatInfo>::to_glib_none(&this).0
|
|
|
|
as glib::ffi::gpointer,
|
2017-08-09 22:13:47 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
impl glib::translate::GlibPtrDefault for AudioFormatInfo {
|
2020-11-21 18:17:20 +00:00
|
|
|
type GlibType = *mut ffi::GstAudioFormatInfo;
|
2017-08-09 22:13:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
2020-11-21 18:17:20 +00:00
|
|
|
impl<'a> glib::translate::ToGlibPtr<'a, *const ffi::GstAudioFormatInfo> for AudioFormatInfo {
|
2017-08-09 22:13:47 +00:00
|
|
|
type Storage = &'a AudioFormatInfo;
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
fn to_glib_none(&'a self) -> glib::translate::Stash<'a, *const ffi::GstAudioFormatInfo, Self> {
|
2017-08-09 22:13:47 +00:00
|
|
|
glib::translate::Stash(self.0, self)
|
|
|
|
}
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
fn to_glib_full(&self) -> *const ffi::GstAudioFormatInfo {
|
2017-08-09 22:13:47 +00:00
|
|
|
unimplemented!()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
2020-11-21 18:17:20 +00:00
|
|
|
impl glib::translate::FromGlibPtrNone<*mut ffi::GstAudioFormatInfo> for AudioFormatInfo {
|
2017-08-09 22:13:47 +00:00
|
|
|
#[inline]
|
2020-11-21 18:17:20 +00:00
|
|
|
unsafe fn from_glib_none(ptr: *mut ffi::GstAudioFormatInfo) -> Self {
|
2017-08-09 22:13:47 +00:00
|
|
|
AudioFormatInfo(&*ptr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-08 22:29:49 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_get() {
|
|
|
|
gst::init().unwrap();
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
let info = AudioFormatInfo::from_format(crate::AudioFormat::S16le);
|
2017-08-08 22:29:49 +00:00
|
|
|
assert_eq!(info.name(), "S16LE");
|
|
|
|
|
|
|
|
let other_info = "S16LE".parse().unwrap();
|
|
|
|
assert_eq!(info, other_info);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn pack_unpack() {
|
|
|
|
gst::init().unwrap();
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
let info = AudioFormatInfo::from_format(crate::AudioFormat::S16le);
|
2017-08-08 22:29:49 +00:00
|
|
|
let unpack_info = AudioFormatInfo::from_format(info.unpack_format());
|
|
|
|
|
|
|
|
assert!(unpack_info.width() > 0);
|
|
|
|
|
|
|
|
let input = [0, 0, 255, 255, 128, 128, 64, 64];
|
|
|
|
let mut unpacked = [0; 16];
|
|
|
|
let mut output = [0; 8];
|
|
|
|
|
2020-11-21 18:17:20 +00:00
|
|
|
info.unpack(crate::AudioPackFlags::empty(), &mut unpacked, &input);
|
|
|
|
info.pack(crate::AudioPackFlags::empty(), &mut output, &unpacked);
|
2017-08-08 22:29:49 +00:00
|
|
|
|
|
|
|
assert_eq!(input, output);
|
|
|
|
}
|
|
|
|
}
|