callback based playing. will prolly do it on record too. will want a thread.

This commit is contained in:
Adam D. Ruppe 2014-12-26 12:04:10 -05:00
parent 780ee597eb
commit 688b42a785
1 changed files with 167 additions and 30 deletions

View File

@ -16,6 +16,18 @@
being able to get/set the volume. being able to get/set the volume.
HOW IT WORKS:
You make a callback which feeds data to the device. Make an
AudioOutput struct then feed your callback to it. Then play.
Then call loop? Or that could be in play?
Methods:
setCallback
play
pause
TODO: TODO:
* play audio high level with options to wait until completion or return immediately * play audio high level with options to wait until completion or return immediately
* midi mid-level stuff * midi mid-level stuff
@ -30,6 +42,9 @@
*/ */
module arsd.simpleaudio; module arsd.simpleaudio;
enum BUFFER_SIZE_FRAMES = 2048;
enum BUFFER_SIZE_SHORT = BUFFER_SIZE_FRAMES * 2;
version(Demo) version(Demo)
void main() { void main() {
/* /*
@ -39,6 +54,7 @@ void main() {
writeln(aio.muteMaster); writeln(aio.muteMaster);
*/ */
/*
mciSendStringA("play test.wav", null, 0, null); mciSendStringA("play test.wav", null, 0, null);
Sleep(3000); Sleep(3000);
import std.stdio; import std.stdio;
@ -46,16 +62,33 @@ void main() {
writeln(err); writeln(err);
Sleep(6000); Sleep(6000);
return; return;
*/
// output about a second of random noise to demo PCM // output about a second of random noise to demo PCM
auto ao = AudioOutput(0); auto ao = AudioOutput(0);
short[4046] randomSpam = void; short[BUFFER_SIZE_SHORT] randomSpam = void;
import core.stdc.stdlib; import core.stdc.stdlib;
foreach(ref s; randomSpam) foreach(ref s; randomSpam)
s = cast(short)((cast(short) rand()) - short.max / 2); s = cast(short)((cast(short) rand()) - short.max / 2);
foreach(i; 0 .. 50) {
ao.write(randomSpam[]); int loopCount = 40;
}
//import std.stdio;
//writeln("Should be about ", loopCount * BUFFER_SIZE_FRAMES * 1000 / 44100, " microseconds");
int loops = 0;
// only do simple stuff in here like fill the data, set simple
// variables, or call stop anything else might cause deadlock
ao.fillData = (short[] buffer) {
buffer[] = randomSpam[0 .. buffer.length];
loops++;
if(loops == loopCount)
ao.stop();
};
ao.play();
return;
// Play a C major scale on the piano to demonstrate midi // Play a C major scale on the piano to demonstrate midi
auto midi = MidiOutput(0); auto midi = MidiOutput(0);
@ -197,54 +230,124 @@ struct AudioOutput {
} else static assert(0); } else static assert(0);
} }
/// passes a buffer of data to fill
///
/// Data is assumed to be interleaved stereo, LE 16 bit, 44.1 kHz /// Data is assumed to be interleaved stereo, LE 16 bit, 44.1 kHz
/// Each item in the array thus alternates between left and right channel /// Each item in the array thus alternates between left and right channel
/// and it takes a total of 88,200 items to make one second of sound. /// and it takes a total of 88,200 items to make one second of sound.
void write(scope const(short)[] data) { void delegate(short[]) fillData;
version(ALSA) {
snd_pcm_sframes_t written;
while(data.length) { shared(bool) playing = false; // considered to be volatile
written = snd_pcm_writei(handle, data.ptr, data.length / 2);
if(written < 0) /// Starts playing, loops until stop is called
throw new AlsaException("pcm write", written); void play() {
data = data[written * 2 .. $]; assert(fillData !is null);
playing = true;
version(ALSA) {
short[BUFFER_SIZE_SHORT] buffer;
while(playing) {
auto err = snd_pcm_wait(handle, 500);
if(err < 0)
throw new AlsaException("uh oh", err);
// err == 0 means timeout
// err == 1 means ready
auto ready = snd_pcm_avail_update(handle);
if(ready < 0)
throw new AlsaException("avail", ready);
if(ready > BUFFER_SIZE_FRAMES)
ready = BUFFER_SIZE_FRAMES;
fillData(buffer[0 .. ready * 2]);
if(playing) {
snd_pcm_sframes_t written;
auto data = buffer[];
while(data.length) {
written = snd_pcm_writei(handle, data.ptr, data.length / 2);
if(written < 0)
throw new AlsaException("pcm write", written);
data = data[written * 2 .. $];
}
}
} }
} else version(WinMM) { } else version(WinMM) {
// This is probably suboptimal but I need to change the API to fix it and not sure what is best yet
WAVEHDR header; enum numBuffers = 4; // use a lot of buffers to get smooth output with Sleep, see below
short[BUFFER_SIZE_SHORT][numBuffers] buffers;
header.lpData = cast(void*) data.ptr; // since this is wave out, it promises not to write... WAVEHDR[numBuffers] headers;
header.dwBufferLength = data.length * short.sizeof;
header.dwFlags = WHDR_BEGINLOOP | WHDR_ENDLOOP;
header.dwLoops = 1;
if(auto err = waveOutPrepareHeader(handle, &header, header.sizeof)) foreach(i, ref header; headers) {
throw new WinMMException("prepare header", err); // since this is wave out, it promises not to write...
auto buffer = buffers[i];
header.lpData = cast(void*) buffer.ptr;
header.dwBufferLength = buffer.length * short.sizeof;
header.dwFlags = WHDR_BEGINLOOP | WHDR_ENDLOOP;
header.dwLoops = 1;
playing = true; if(auto err = waveOutPrepareHeader(handle, &header, header.sizeof))
throw new WinMMException("prepare header", err);
if(auto err = waveOutWrite(handle, &header, header.sizeof)) // prime it
throw new WinMMException("wave out write", err); fillData(buffer[]);
while(playing) // indicate that they are filled and good to go
header.dwUser = 1;
}
while(playing) {
// and queue both to be played, if they are ready
foreach(ref header; headers)
if(header.dwUser) {
if(auto err = waveOutWrite(handle, &header, header.sizeof))
throw new WinMMException("wave out write", err);
header.dwUser = 0;
}
Sleep(1); Sleep(1);
// the system resolution may be lower than this sleep. To avoid gaps
// in output, we use multiple buffers. Might introduce latency, not
// sure how best to fix. I don't want to busy loop...
}
if(auto err = waveOutUnprepareHeader(handle, &header, header.sizeof)) // wait for the system to finish with our buffers
throw new WinMMException("unprepare", err); bool anyInUse = true;
while(anyInUse) {
anyInUse = false;
foreach(header; headers) {
if(!header.dwUser) {
anyInUse = true;
break;
}
}
if(anyInUse)
Sleep(1);
}
foreach(ref header; headers)
if(auto err = waveOutUnprepareHeader(handle, &header, header.sizeof))
throw new WinMMException("unprepare", err);
} else static assert(0); } else static assert(0);
} }
version(WinMM) { /// Breaks the play loop
// volatile void stop() {
shared(bool) playing = false; playing = false;
}
version(WinMM) {
extern(Windows) extern(Windows)
static void mmCallback(HWAVEOUT handle, UINT msg, void* userData, DWORD param1, DWORD param2) { static void mmCallback(HWAVEOUT handle, UINT msg, void* userData, DWORD param1, DWORD param2) {
AudioOutput* ao = cast(AudioOutput*) userData; AudioOutput* ao = cast(AudioOutput*) userData;
if(msg == WOM_DONE) { if(msg == WOM_DONE) {
ao.playing = false; auto header = cast(WAVEHDR*) param1;
// we want to bounce back and forth between two buffers
// to keep the sound going all the time
if(ao.playing) {
ao.fillData((cast(short*) header.lpData)[0 .. header.dwBufferLength / short.sizeof]);
}
header.dwUser = 1;
} }
} }
} }
@ -668,6 +771,8 @@ snd_pcm_t* openAlsaPcm(snd_pcm_stream_t direction) {
snd_pcm_t* handle; snd_pcm_t* handle;
snd_pcm_hw_params_t* hwParams; snd_pcm_hw_params_t* hwParams;
/* Open PCM and initialize hardware */
if (auto err = snd_pcm_open(&handle, cardName, direction, 0)) if (auto err = snd_pcm_open(&handle, cardName, direction, 0))
throw new AlsaException("open device", err); throw new AlsaException("open device", err);
scope(failure) scope(failure)
@ -698,6 +803,24 @@ snd_pcm_t* openAlsaPcm(snd_pcm_stream_t direction) {
if (auto err = snd_pcm_hw_params(handle, hwParams)) if (auto err = snd_pcm_hw_params(handle, hwParams))
throw new AlsaException("params install", err); throw new AlsaException("params install", err);
/* Setting up the callbacks */
snd_pcm_sw_params_t* swparams;
if(auto err = snd_pcm_sw_params_malloc(&swparams))
throw new AlsaException("sw malloc", err);
scope(exit)
snd_pcm_sw_params_free(swparams);
if(auto err = snd_pcm_sw_params_current(handle, swparams))
throw new AlsaException("sw set", err);
if(auto err = snd_pcm_sw_params_set_avail_min(handle, swparams, BUFFER_SIZE_FRAMES))
throw new AlsaException("sw min", err);
if(auto err = snd_pcm_sw_params_set_start_threshold(handle, swparams, 0))
throw new AlsaException("sw threshold", err);
if(auto err = snd_pcm_sw_params(handle, swparams))
throw new AlsaException("sw params", err);
/* finish setup */
if (auto err = snd_pcm_prepare(handle)) if (auto err = snd_pcm_prepare(handle))
throw new AlsaException("prepare", err); throw new AlsaException("prepare", err);
@ -853,6 +976,7 @@ extern(C):
struct snd_pcm_t {} struct snd_pcm_t {}
struct snd_pcm_hw_params_t {} struct snd_pcm_hw_params_t {}
struct snd_pcm_sw_params_t {}
int snd_pcm_open(snd_pcm_t**, const char*, snd_pcm_stream_t, int); int snd_pcm_open(snd_pcm_t**, const char*, snd_pcm_stream_t, int);
int snd_pcm_close(snd_pcm_t*); int snd_pcm_close(snd_pcm_t*);
@ -866,11 +990,24 @@ extern(C):
int snd_pcm_hw_params_set_format(snd_pcm_t*, snd_pcm_hw_params_t*, snd_pcm_format); int snd_pcm_hw_params_set_format(snd_pcm_t*, snd_pcm_hw_params_t*, snd_pcm_format);
int snd_pcm_hw_params_set_rate_near(snd_pcm_t*, snd_pcm_hw_params_t*, uint*, int*); int snd_pcm_hw_params_set_rate_near(snd_pcm_t*, snd_pcm_hw_params_t*, uint*, int*);
int snd_pcm_sw_params_malloc(snd_pcm_sw_params_t**);
void snd_pcm_sw_params_free(snd_pcm_sw_params_t*);
int snd_pcm_sw_params_current(snd_pcm_t *pcm, snd_pcm_sw_params_t *params);
int snd_pcm_sw_params(snd_pcm_t *pcm, snd_pcm_sw_params_t *params);
int snd_pcm_sw_params_set_avail_min(snd_pcm_t*, snd_pcm_sw_params_t*, snd_pcm_uframes_t);
int snd_pcm_sw_params_set_start_threshold(snd_pcm_t*, snd_pcm_sw_params_t*, snd_pcm_uframes_t);
int snd_pcm_sw_params_set_stop_threshold(snd_pcm_t*, snd_pcm_sw_params_t*, snd_pcm_uframes_t);
alias snd_pcm_sframes_t = c_long; alias snd_pcm_sframes_t = c_long;
alias snd_pcm_uframes_t = c_ulong; alias snd_pcm_uframes_t = c_ulong;
snd_pcm_sframes_t snd_pcm_writei(snd_pcm_t*, const void*, snd_pcm_uframes_t size); snd_pcm_sframes_t snd_pcm_writei(snd_pcm_t*, const void*, snd_pcm_uframes_t size);
snd_pcm_sframes_t snd_pcm_readi(snd_pcm_t*, void*, snd_pcm_uframes_t size); snd_pcm_sframes_t snd_pcm_readi(snd_pcm_t*, void*, snd_pcm_uframes_t size);
int snd_pcm_wait(snd_pcm_t *pcm, int timeout);
snd_pcm_sframes_t snd_pcm_avail(snd_pcm_t *pcm);
snd_pcm_sframes_t snd_pcm_avail_update(snd_pcm_t *pcm);
// raw midi // raw midi
static if(is(ssize_t == uint)) static if(is(ssize_t == uint))