2017-07-01 22:03:21 +00:00
|
|
|
use alsa::card::Card;
|
|
|
|
use alsa::mixer::SelemChannelId::*;
|
2017-07-04 19:15:11 +00:00
|
|
|
use alsa::mixer::{Mixer, Selem, SelemId};
|
2017-07-01 22:03:21 +00:00
|
|
|
use alsa::poll::PollDescriptors;
|
|
|
|
use alsa_sys;
|
|
|
|
use errors::*;
|
|
|
|
use glib_sys;
|
|
|
|
use libc::c_uint;
|
|
|
|
use libc::pollfd;
|
|
|
|
use libc::size_t;
|
2017-07-01 23:35:12 +00:00
|
|
|
use std::cell::Cell;
|
2017-07-01 22:03:21 +00:00
|
|
|
use std::mem;
|
|
|
|
use std::ptr;
|
|
|
|
use std::rc::Rc;
|
|
|
|
use std::u8;
|
2017-07-04 19:15:11 +00:00
|
|
|
use support_alsa::*;
|
2017-07-01 22:03:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
|
|
|
pub enum AlsaEvent {
|
|
|
|
AlsaCardError,
|
|
|
|
AlsaCardDiconnected,
|
|
|
|
AlsaCardValuesChanged,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pub struct AlsaCard {
|
|
|
|
_cannot_construct: (),
|
|
|
|
pub card: Card,
|
2017-07-01 23:35:12 +00:00
|
|
|
pub mixer: Mixer,
|
2017-07-01 22:03:21 +00:00
|
|
|
pub selem_id: SelemId,
|
2017-07-01 23:35:12 +00:00
|
|
|
pub watch_ids: Cell<Vec<u32>>,
|
2017-07-01 22:03:21 +00:00
|
|
|
pub cb: Rc<Fn(AlsaEvent)>,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
impl AlsaCard {
|
2017-07-07 20:10:45 +00:00
|
|
|
pub fn new(card_name: Option<String>,
|
|
|
|
elem_name: Option<String>,
|
|
|
|
cb: Rc<Fn(AlsaEvent)>)
|
|
|
|
-> Result<Box<AlsaCard>> {
|
2017-07-01 22:03:21 +00:00
|
|
|
let card = {
|
|
|
|
match card_name {
|
2017-07-07 20:10:45 +00:00
|
|
|
Some(name) => {
|
|
|
|
if name == "(default)" {
|
2017-07-07 23:35:13 +00:00
|
|
|
let default = get_default_alsa_card();
|
|
|
|
if alsa_card_has_playable_selem(&default) {
|
|
|
|
default
|
|
|
|
} else {
|
|
|
|
warn!("Default alsa card not playabla, trying others");
|
|
|
|
get_first_playable_alsa_card()?
|
|
|
|
}
|
2017-07-07 20:10:45 +00:00
|
|
|
} else {
|
2017-07-07 23:35:13 +00:00
|
|
|
let mycard = get_alsa_card_by_name(name.clone());
|
|
|
|
match mycard {
|
|
|
|
Ok(card) => card,
|
|
|
|
Err(_) => {
|
|
|
|
warn!("Card {} not playable, trying others", name);
|
|
|
|
get_first_playable_alsa_card()?
|
|
|
|
}
|
|
|
|
}
|
2017-07-07 20:10:45 +00:00
|
|
|
}
|
|
|
|
},
|
2017-07-07 23:35:13 +00:00
|
|
|
None => get_first_playable_alsa_card()?,
|
2017-07-01 22:03:21 +00:00
|
|
|
}
|
|
|
|
};
|
2017-07-01 23:35:12 +00:00
|
|
|
let mixer = get_mixer(&card)?;
|
|
|
|
|
2017-07-07 20:10:45 +00:00
|
|
|
let selem_id =
|
2017-07-07 23:35:13 +00:00
|
|
|
get_playable_selem_by_name(&mixer,
|
|
|
|
elem_name.unwrap_or(String::from("Master")))?
|
2017-07-07 20:10:45 +00:00
|
|
|
.get_id();
|
2017-07-01 22:03:21 +00:00
|
|
|
|
2017-07-07 23:35:13 +00:00
|
|
|
let vec_pollfd = PollDescriptors::get(&mixer)?;
|
|
|
|
|
2017-07-01 22:03:21 +00:00
|
|
|
let acard = Box::new(AlsaCard {
|
2017-07-07 20:10:45 +00:00
|
|
|
_cannot_construct: (),
|
|
|
|
card: card,
|
|
|
|
mixer: mixer,
|
|
|
|
selem_id: selem_id,
|
|
|
|
watch_ids: Cell::new(vec![]),
|
|
|
|
cb: cb,
|
|
|
|
});
|
2017-07-01 22:03:21 +00:00
|
|
|
|
|
|
|
/* TODO: callback is registered here, which must be unregistered
|
|
|
|
* when the card is destroyed!!
|
|
|
|
* poll descriptors must be unwatched too */
|
2017-07-07 20:10:45 +00:00
|
|
|
let watch_ids = AlsaCard::watch_poll_descriptors(vec_pollfd,
|
|
|
|
acard.as_ref());
|
2017-07-01 23:35:12 +00:00
|
|
|
acard.watch_ids.set(watch_ids);
|
2017-07-01 22:03:21 +00:00
|
|
|
|
|
|
|
return Ok(acard);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-07-02 16:11:56 +00:00
|
|
|
pub fn card_name(&self) -> Result<String> {
|
|
|
|
return self.card.get_name().from_err();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pub fn chan_name(&self) -> Result<String> {
|
2017-07-07 20:10:45 +00:00
|
|
|
let n = self.selem_id
|
|
|
|
.get_name()
|
|
|
|
.map(|y| String::from(y))?;
|
2017-07-02 16:11:56 +00:00
|
|
|
return Ok(n);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-07-01 22:03:21 +00:00
|
|
|
pub fn selem(&self) -> Selem {
|
2017-07-02 16:11:56 +00:00
|
|
|
return self.mixer.find_selem(&self.selem_id).unwrap();
|
2017-07-01 22:03:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pub fn get_vol(&self) -> Result<f64> {
|
|
|
|
let selem = self.selem();
|
|
|
|
let range = selem.get_playback_volume_range();
|
|
|
|
let volume = selem.get_playback_volume(FrontRight).map(|v| {
|
|
|
|
return vol_to_percent(v, range);
|
|
|
|
});
|
|
|
|
|
|
|
|
return volume.from_err();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pub fn set_vol(&self, new_vol: f64) -> Result<()> {
|
|
|
|
let selem = self.selem();
|
|
|
|
/* auto-unmute */
|
2017-07-07 23:35:13 +00:00
|
|
|
if self.has_mute() && self.get_mute()? {
|
2017-07-01 22:03:21 +00:00
|
|
|
self.set_mute(false)?;
|
|
|
|
}
|
|
|
|
|
|
|
|
let range = selem.get_playback_volume_range();
|
2017-07-07 20:10:45 +00:00
|
|
|
selem.set_playback_volume_all(percent_to_vol(new_vol, range))?;
|
2017-07-01 22:03:21 +00:00
|
|
|
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pub fn has_mute(&self) -> bool {
|
|
|
|
let selem = self.selem();
|
|
|
|
return selem.has_playback_switch();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pub fn get_mute(&self) -> Result<bool> {
|
|
|
|
let selem = self.selem();
|
|
|
|
let val = selem.get_playback_switch(FrontRight)?;
|
|
|
|
return Ok(val == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pub fn set_mute(&self, mute: bool) -> Result<()> {
|
|
|
|
let selem = self.selem();
|
|
|
|
/* true -> mute, false -> unmute */
|
|
|
|
let _ = selem.set_playback_switch_all(!mute as i32)?;
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-07-07 20:10:45 +00:00
|
|
|
fn watch_poll_descriptors(polls: Vec<pollfd>,
|
|
|
|
acard: &AlsaCard)
|
|
|
|
-> Vec<c_uint> {
|
2017-07-01 22:03:21 +00:00
|
|
|
let mut watch_ids: Vec<c_uint> = vec![];
|
|
|
|
let acard_ptr =
|
|
|
|
unsafe { mem::transmute::<&AlsaCard, glib_sys::gpointer>(acard) };
|
|
|
|
for poll in polls {
|
|
|
|
let gioc: *mut glib_sys::GIOChannel =
|
|
|
|
unsafe { glib_sys::g_io_channel_unix_new(poll.fd) };
|
|
|
|
let id = unsafe {
|
|
|
|
glib_sys::g_io_add_watch(
|
|
|
|
gioc,
|
|
|
|
glib_sys::GIOCondition::from_bits(
|
|
|
|
glib_sys::G_IO_IN.bits() | glib_sys::G_IO_ERR.bits(),
|
|
|
|
).unwrap(),
|
|
|
|
Some(watch_cb),
|
|
|
|
acard_ptr,
|
|
|
|
)
|
|
|
|
};
|
|
|
|
watch_ids.push(id);
|
|
|
|
unsafe { glib_sys::g_io_channel_unref(gioc) }
|
|
|
|
}
|
|
|
|
|
|
|
|
return watch_ids;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn unwatch_poll_descriptors(watch_ids: &Vec<u32>) {
|
|
|
|
for watch_id in watch_ids {
|
|
|
|
unsafe {
|
|
|
|
glib_sys::g_source_remove(*watch_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-07-01 23:35:12 +00:00
|
|
|
// TODO: test that this is actually triggered when switching cards
|
2017-07-01 22:03:21 +00:00
|
|
|
impl Drop for AlsaCard {
|
|
|
|
// call Box::new(x), transmute the Box into a raw pointer, and then
|
|
|
|
// std::mem::forget
|
|
|
|
//
|
|
|
|
// if you unregister the callback, you should keep a raw pointer to the
|
|
|
|
// box
|
|
|
|
//
|
|
|
|
// For instance, `register` could return a raw pointer to the
|
|
|
|
// Box + a std::marker::PhantomData with the appropriate
|
|
|
|
// lifetime (if applicable)
|
|
|
|
//
|
|
|
|
// The struct could implement Drop, which unregisters the
|
|
|
|
// callback and frees the Box, by simply transmuting the
|
|
|
|
// raw pointer to a Box<T>
|
|
|
|
fn drop(&mut self) {
|
2017-07-01 23:35:12 +00:00
|
|
|
debug!("Destructing watch_ids: {:?}", self.watch_ids.get_mut());
|
|
|
|
AlsaCard::unwatch_poll_descriptors(&self.watch_ids.get_mut());
|
2017-07-01 22:03:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-07-07 20:10:45 +00:00
|
|
|
extern "C" fn watch_cb(chan: *mut glib_sys::GIOChannel,
|
|
|
|
cond: glib_sys::GIOCondition,
|
|
|
|
data: glib_sys::gpointer)
|
|
|
|
-> glib_sys::gboolean {
|
2017-07-01 22:03:21 +00:00
|
|
|
|
|
|
|
let acard =
|
|
|
|
unsafe { mem::transmute::<glib_sys::gpointer, &AlsaCard>(data) };
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
let mixer_ptr =
|
|
|
|
mem::transmute::<&Mixer, &*mut alsa_sys::snd_mixer_t>(&acard.mixer);
|
|
|
|
alsa_sys::snd_mixer_handle_events(*mixer_ptr);
|
|
|
|
};
|
|
|
|
|
|
|
|
if cond == glib_sys::G_IO_ERR {
|
|
|
|
return false as glib_sys::gboolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut sread: size_t = 1;
|
|
|
|
let mut buf: Vec<u8> = vec![0; 256];
|
|
|
|
|
|
|
|
while sread > 0 {
|
2017-07-07 20:10:45 +00:00
|
|
|
let stat: glib_sys::GIOStatus =
|
|
|
|
unsafe {
|
|
|
|
glib_sys::g_io_channel_read_chars(chan,
|
|
|
|
buf.as_mut_ptr() as *mut u8,
|
|
|
|
256,
|
|
|
|
&mut sread as *mut size_t,
|
|
|
|
ptr::null_mut())
|
|
|
|
};
|
2017-07-01 22:03:21 +00:00
|
|
|
|
|
|
|
match stat {
|
|
|
|
glib_sys::G_IO_STATUS_AGAIN => {
|
|
|
|
debug!("G_IO_STATUS_AGAIN");
|
|
|
|
continue;
|
|
|
|
}
|
2017-07-07 23:35:13 +00:00
|
|
|
// TODO: handle these failure cases
|
2017-07-01 22:03:21 +00:00
|
|
|
glib_sys::G_IO_STATUS_NORMAL => debug!("G_IO_STATUS_NORMAL"),
|
|
|
|
glib_sys::G_IO_STATUS_ERROR => debug!("G_IO_STATUS_ERROR"),
|
|
|
|
glib_sys::G_IO_STATUS_EOF => debug!("G_IO_STATUS_EOF"),
|
|
|
|
}
|
|
|
|
return true as glib_sys::gboolean;
|
|
|
|
}
|
|
|
|
let cb = &acard.cb;
|
|
|
|
cb(AlsaEvent::AlsaCardValuesChanged);
|
|
|
|
|
|
|
|
return true as glib_sys::gboolean;
|
|
|
|
}
|