adding outputs

This commit is contained in:
Tom Gowan 2018-10-30 16:50:20 +11:00 committed by mitchmindtree
parent ff7fd62a2d
commit ed50bb9923
3 changed files with 77 additions and 25 deletions

View File

@ -19,7 +19,7 @@ use std::os::raw::c_char;
use std::os::raw::c_double; use std::os::raw::c_double;
use std::os::raw::c_long; use std::os::raw::c_long;
use std::os::raw::c_void; use std::os::raw::c_void;
use std::sync::atomic::{AtomicUsize, Ordering}; use std::sync::atomic::{AtomicUsize, AtomicBool, Ordering};
use std::sync::{Arc, Mutex}; use std::sync::{Arc, Mutex};
use std::sync::MutexGuard; use std::sync::MutexGuard;
@ -46,6 +46,8 @@ lazy_static! {
} }
static STREAM_DRIVER_COUNT: AtomicUsize = AtomicUsize::new(0); static STREAM_DRIVER_COUNT: AtomicUsize = AtomicUsize::new(0);
pub static SILENCE_FIRST: AtomicBool = AtomicBool::new(false);
pub static SILENCE_SECOND: AtomicBool = AtomicBool::new(false);
#[derive(Debug)] #[derive(Debug)]
pub struct Channel { pub struct Channel {

View File

@ -45,7 +45,7 @@ impl Device {
FormatsEnumerationError> { FormatsEnumerationError> {
match self.default_input_format() { match self.default_input_format() {
Ok(f) => Ok(vec![SupportedFormat::from(f)].into_iter()), Ok(f) => Ok(vec![SupportedFormat::from(f)].into_iter()),
Err(e) => Err(FormatsEnumerationError::DeviceNotAvailable), Err(_) => Err(FormatsEnumerationError::DeviceNotAvailable),
} }
} }
@ -53,7 +53,7 @@ impl Device {
FormatsEnumerationError> { FormatsEnumerationError> {
match self.default_output_format() { match self.default_output_format() {
Ok(f) => Ok(vec![SupportedFormat::from(f)].into_iter()), Ok(f) => Ok(vec![SupportedFormat::from(f)].into_iter()),
Err(e) => Err(FormatsEnumerationError::DeviceNotAvailable), Err(_) => Err(FormatsEnumerationError::DeviceNotAvailable),
} }
} }
@ -127,7 +127,7 @@ impl Iterator for Devices {
// Asio doesn't have a concept of default // Asio doesn't have a concept of default
// so returning first in list as default // so returning first in list as default
pub fn default_input_device() -> Option<Device> { pub fn default_input_device() -> Option<Device> {
let mut driver_list = sys::get_driver_list(); let driver_list = sys::get_driver_list();
// Remove // Remove
let d_name = driver_list.into_iter() let d_name = driver_list.into_iter()
.filter(|d| d == "ASIO4ALL v2") .filter(|d| d == "ASIO4ALL v2")
@ -151,7 +151,7 @@ pub fn default_input_device() -> Option<Device> {
} }
pub fn default_output_device() -> Option<Device> { pub fn default_output_device() -> Option<Device> {
let mut driver_list = sys::get_driver_list(); let driver_list = sys::get_driver_list();
// Remove // Remove
let d_name = driver_list.into_iter() let d_name = driver_list.into_iter()
.filter(|d| d == "ASIO4ALL v2") .filter(|d| d == "ASIO4ALL v2")

View File

@ -5,11 +5,9 @@ use std;
use Format; use Format;
use CreationError; use CreationError;
use StreamData; use StreamData;
use std::marker::PhantomData;
use super::Device; use super::Device;
use std::cell::Cell;
use UnknownTypeOutputBuffer;
use UnknownTypeInputBuffer; use UnknownTypeInputBuffer;
use UnknownTypeOutputBuffer;
use std::sync::{Arc, Mutex}; use std::sync::{Arc, Mutex};
use std::mem; use std::mem;
use std::sync::atomic::{AtomicUsize, Ordering}; use std::sync::atomic::{AtomicUsize, Ordering};
@ -32,9 +30,8 @@ pub struct OutputBuffer<'a, T: 'a> {
buffer: &'a mut [T], buffer: &'a mut [T],
} }
enum Stream{ struct Stream{
Input, playing: bool,
Output,
} }
#[derive(Default)] #[derive(Default)]
@ -68,6 +65,9 @@ impl EventLoop {
} }
} }
/// Create a new CPAL Input Stream
/// If there is no ASIO Input Stream
/// it will be created
fn get_input_stream(&self, drivers: &sys::Drivers, num_channels: usize) -> Result<usize, CreationError> { fn get_input_stream(&self, drivers: &sys::Drivers, num_channels: usize) -> Result<usize, CreationError> {
let ref mut streams = *self.asio_streams.lock().unwrap(); let ref mut streams = *self.asio_streams.lock().unwrap();
match streams.input { match streams.input {
@ -130,8 +130,8 @@ impl EventLoop {
let count = self.stream_count.load(Ordering::SeqCst); let count = self.stream_count.load(Ordering::SeqCst);
self.stream_count.store(count + 1, Ordering::SeqCst); self.stream_count.store(count + 1, Ordering::SeqCst);
let asio_streams = self.asio_streams.clone(); let asio_streams = self.asio_streams.clone();
let cpal_streams = self.cpal_streams.clone();
let callbacks = self.callbacks.clone(); let callbacks = self.callbacks.clone();
let bytes_per_channel = format.data_type.sample_size();
// Create buffers // Create buffers
let channel_len = cpal_num_samples let channel_len = cpal_num_samples
@ -175,11 +175,19 @@ impl EventLoop {
}; };
sys::set_callback(move |index| unsafe { sys::set_callback(move |index| unsafe {
//if not playing return early
{
if let Some(s) = cpal_streams.lock().unwrap().get(count - 1){
if let Some(s) = s{
if !s.playing { return (); }
}
}
}
if let Some(ref asio_stream) = asio_streams.lock().unwrap().input { if let Some(ref asio_stream) = asio_streams.lock().unwrap().input {
// Number of samples needed total // Number of samples needed total
let mut callbacks = callbacks.lock().unwrap(); let mut callbacks = callbacks.lock().unwrap();
// Assuming only one callback, probably needs to change // Theres only a single callback because theres only one event loop
match callbacks.first_mut() { match callbacks.first_mut() {
Some(callback) => { Some(callback) => {
macro_rules! try_callback { macro_rules! try_callback {
@ -266,7 +274,7 @@ impl EventLoop {
} }
} }
}); });
self.cpal_streams.lock().unwrap().push(Some(Stream::Input)); self.cpal_streams.lock().unwrap().push(Some(Stream{ playing: false }));
StreamId(count) StreamId(count)
}) })
} }
@ -287,8 +295,8 @@ pub fn build_output_stream(
let count = self.stream_count.load(Ordering::SeqCst); let count = self.stream_count.load(Ordering::SeqCst);
self.stream_count.store(count + 1, Ordering::SeqCst); self.stream_count.store(count + 1, Ordering::SeqCst);
let asio_streams = self.asio_streams.clone(); let asio_streams = self.asio_streams.clone();
let cpal_streams = self.cpal_streams.clone();
let callbacks = self.callbacks.clone(); let callbacks = self.callbacks.clone();
let bytes_per_channel = format.data_type.sample_size();
// Create buffers // Create buffers
let channel_len = cpal_num_samples let channel_len = cpal_num_samples
/ num_channels as usize; / num_channels as usize;
@ -331,11 +339,19 @@ pub fn build_output_stream(
}; };
sys::set_callback(move |index| unsafe { sys::set_callback(move |index| unsafe {
//if not playing return early
{
if let Some(s) = cpal_streams.lock().unwrap().get(count - 1){
if let Some(s) = s{
if !s.playing { return (); }
}
}
}
if let Some(ref asio_stream) = asio_streams.lock().unwrap().output { if let Some(ref asio_stream) = asio_streams.lock().unwrap().output {
// Number of samples needed total // Number of samples needed total
let mut callbacks = callbacks.lock().unwrap(); let mut callbacks = callbacks.lock().unwrap();
// Assuming only one callback, probably needs to change // Theres only a single callback because theres only one event loop
match callbacks.first_mut() { match callbacks.first_mut() {
Some(callback) => { Some(callback) => {
macro_rules! try_callback { macro_rules! try_callback {
@ -349,9 +365,6 @@ pub fn build_output_stream(
$BuffersTypeIdent:ident $BuffersTypeIdent:ident
) => { ) => {
let mut my_buffers = $Buffers; let mut my_buffers = $Buffers;
// Buffer that is filled by cpal.
//let mut cpal_buffer: Vec<$SampleType> = vec![0 as $SampleType; cpal_num_samples];
// Call in block because of mut borrow
{ {
let buff = OutputBuffer{ let buff = OutputBuffer{
buffer: &mut my_buffers.cpal buffer: &mut my_buffers.cpal
@ -375,6 +388,25 @@ pub fn build_output_stream(
au::deinterleave(&c_buffer[..], channels); au::deinterleave(&c_buffer[..], channels);
} }
let silence = match index {
0 =>{
if !sys::SILENCE_FIRST.load(Ordering::SeqCst) {
sys::SILENCE_FIRST.store(true, Ordering::SeqCst);
sys::SILENCE_SECOND.store(false, Ordering::SeqCst);
true
}else{false}
},
1 =>{
if !sys::SILENCE_SECOND.load(Ordering::SeqCst) {
sys::SILENCE_SECOND.store(true, Ordering::SeqCst);
sys::SILENCE_FIRST.store(false, Ordering::SeqCst);
true
}else{false}
},
_ => unreachable!(),
};
// For each channel write the cpal data to // For each channel write the cpal data to
// the asio buffer // the asio buffer
// TODO need to check for Endian // TODO need to check for Endian
@ -388,7 +420,8 @@ pub fn build_output_stream(
asio_stream.buffer_size as usize); asio_stream.buffer_size as usize);
for (asio_s, cpal_s) in asio_buffer.iter_mut() for (asio_s, cpal_s) in asio_buffer.iter_mut()
.zip(channel){ .zip(channel){
*asio_s = (*cpal_s as i64 * if silence { *asio_s = 0.0 as $AsioType; }
*asio_s += (*cpal_s as i64 *
::std::$AsioTypeIdent::MAX as i64 / ::std::$AsioTypeIdent::MAX as i64 /
::std::$SampleTypeIdent::MAX as i64) as $AsioType; ::std::$SampleTypeIdent::MAX as i64) as $AsioType;
} }
@ -422,23 +455,40 @@ pub fn build_output_stream(
} }
} }
}); });
self.cpal_streams.lock().unwrap().push(Some(Stream::Output)); self.cpal_streams.lock().unwrap().push(Some(Stream{ playing: false }));
StreamId(count) StreamId(count)
}) })
} }
pub fn play_stream(&self, stream: StreamId) { pub fn play_stream(&self, stream_id: StreamId) {
let mut streams = self.cpal_streams.lock().unwrap();
if let Some(s) = streams.get_mut(stream_id.0).expect("Bad play stream index") {
s.playing = true;
}
// Calling play when already playing is a no-op
sys::play(); sys::play();
} }
pub fn pause_stream(&self, stream: StreamId) { pub fn pause_stream(&self, stream_id: StreamId) {
let mut streams = self.cpal_streams.lock().unwrap();
if let Some(s) = streams.get_mut(stream_id.0).expect("Bad pause stream index") {
s.playing = false;
}
let any_playing = streams
.iter()
.filter(|s| if let Some(s) = s {
s.playing
} else {false} )
.next();
if let None = any_playing {
sys::stop(); sys::stop();
} }
}
pub fn destroy_stream(&self, stream_id: StreamId) { pub fn destroy_stream(&self, stream_id: StreamId) {
let mut streams = self.cpal_streams.lock().unwrap(); let mut streams = self.cpal_streams.lock().unwrap();
streams.get_mut(stream_id.0 - 1).take(); streams.get_mut(stream_id.0).take();
let count = self.stream_count.load(Ordering::SeqCst); let count = self.stream_count.load(Ordering::SeqCst);
self.stream_count.store(count - 1, Ordering::SeqCst); self.stream_count.store(count - 1, Ordering::SeqCst);
if count == 1 { if count == 1 {