mirror of https://github.com/adamdruppe/arsd.git
callback based playing. will prolly do it on record too. will want a thread.
This commit is contained in:
parent
780ee597eb
commit
688b42a785
197
simpleaudio.d
197
simpleaudio.d
|
@ -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))
|
||||||
|
|
Loading…
Reference in New Issue