mirror of https://github.com/adamdruppe/arsd.git
opl3 emulation for old school midi
This commit is contained in:
parent
c950ce3ae8
commit
e0c4e2bba3
File diff suppressed because it is too large
Load Diff
108
simpleaudio.d
108
simpleaudio.d
|
@ -204,21 +204,30 @@ interface SampleController {
|
||||||
object.
|
object.
|
||||||
+/
|
+/
|
||||||
void stop();
|
void stop();
|
||||||
|
/++
|
||||||
|
Reports the current stream position, in seconds, if available (NaN if not).
|
||||||
|
+/
|
||||||
|
float position();
|
||||||
}
|
}
|
||||||
|
|
||||||
class DummySample : SampleController {
|
private class DummySample : SampleController {
|
||||||
void pause() {}
|
void pause() {}
|
||||||
void resume() {}
|
void resume() {}
|
||||||
void stop() {}
|
void stop() {}
|
||||||
|
float position() { return float.init; }
|
||||||
}
|
}
|
||||||
|
|
||||||
class SampleControlFlags : SampleController {
|
private class SampleControlFlags : SampleController {
|
||||||
void pause() { paused = true; }
|
void pause() { paused = true; }
|
||||||
void resume() { paused = false; }
|
void resume() { paused = false; }
|
||||||
void stop() { stopped = true; }
|
void stop() { stopped = true; }
|
||||||
|
|
||||||
bool paused;
|
bool paused;
|
||||||
bool stopped;
|
bool stopped;
|
||||||
|
|
||||||
|
float position() { return currentPosition; }
|
||||||
|
|
||||||
|
float currentPosition = 0.0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/++
|
/++
|
||||||
|
@ -246,6 +255,8 @@ struct AudioOutputThread {
|
||||||
|
|
||||||
History:
|
History:
|
||||||
Parameter `default` added on Nov 8, 2020.
|
Parameter `default` added on Nov 8, 2020.
|
||||||
|
|
||||||
|
The sample rate parameter was not correctly applied to the device on Linux until December 24, 2020.
|
||||||
+/
|
+/
|
||||||
this(bool enable, int SampleRate = 44100, int channels = 2, string device = "default") {
|
this(bool enable, int SampleRate = 44100, int channels = 2, string device = "default") {
|
||||||
if(enable) {
|
if(enable) {
|
||||||
|
@ -305,6 +316,12 @@ struct AudioOutputThread {
|
||||||
else static assert(0);
|
else static assert(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SampleController playEmulatedOpl3Midi()(string filename) {
|
||||||
|
if(impl)
|
||||||
|
return impl.playEmulatedOpl3Midi(filename);
|
||||||
|
return new DummySample;
|
||||||
|
}
|
||||||
|
|
||||||
SampleController playOgg()(string filename, bool loop = false) {
|
SampleController playOgg()(string filename, bool loop = false) {
|
||||||
if(impl)
|
if(impl)
|
||||||
return impl.playOgg(filename, loop);
|
return impl.playOgg(filename, loop);
|
||||||
|
@ -485,10 +502,60 @@ final class AudioPcmOutThreadImplementation : Thread {
|
||||||
}
|
}
|
||||||
|
|
||||||
/++
|
/++
|
||||||
Requires vorbis.d to be compiled in (module arsd.vorbis)
|
Plays the given midi files with the nuked opl3 emulator.
|
||||||
|
|
||||||
|
Requires nukedopl3.d (module [arsd.nukedopl3]) to be compiled in, which is GPL.
|
||||||
|
|
||||||
|
History:
|
||||||
|
Added December 24, 2020.
|
||||||
|
License:
|
||||||
|
If you use this function, you are opting into the GPL version 2 or later.
|
||||||
|
Authors:
|
||||||
|
Based on ketmar's code.
|
||||||
|
+/
|
||||||
|
SampleController playEmulatedOpl3Midi()(string filename, bool loop = false) {
|
||||||
|
import arsd.nukedopl3;
|
||||||
|
auto scf = new SampleControlFlags;
|
||||||
|
|
||||||
|
import std.file;
|
||||||
|
auto bytes = cast(ubyte[]) std.file.read(filename);
|
||||||
|
auto player = new OPLPlayer(this.SampleRate, true, channels == 2);
|
||||||
|
player.looped = loop;
|
||||||
|
player.load(bytes);
|
||||||
|
player.play();
|
||||||
|
|
||||||
|
addChannel(
|
||||||
|
delegate bool(short[] buffer) {
|
||||||
|
if(scf.paused) {
|
||||||
|
buffer[] = 0;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!player.playing)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
auto pos = player.generate(buffer[]);
|
||||||
|
scf.currentPosition += cast(float) buffer.length / SampleRate/ channels;
|
||||||
|
if(pos == 0)
|
||||||
|
return false;
|
||||||
|
return !scf.stopped;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
return scf;
|
||||||
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
Requires vorbis.d to be compiled in (module arsd.vorbis)
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
An implementation of [SampleController] which lets you pause, etc., the file.
|
||||||
|
|
||||||
|
Please note that the static type may change in the future. It will always be a subtype of [SampleController], but it may be more specialized as I add more features and this will not necessarily match its sister functions, [playMp3] and [playWav], though all three will share an ancestor in [SampleController]. Therefore, if you use `auto`, there's no guarantee the static type won't change in future versions and I will NOT consider that a breaking change since the base interface will remain compatible.
|
||||||
History:
|
History:
|
||||||
Automatic resampling support added Nov 7, 2020.
|
Automatic resampling support added Nov 7, 2020.
|
||||||
|
|
||||||
|
Return value changed from `void` to a sample control object on December 23, 2020.
|
||||||
+/
|
+/
|
||||||
SampleController playOgg()(string filename, bool loop = false) {
|
SampleController playOgg()(string filename, bool loop = false) {
|
||||||
import arsd.vorbis;
|
import arsd.vorbis;
|
||||||
|
@ -522,10 +589,13 @@ final class AudioPcmOutThreadImplementation : Thread {
|
||||||
if(got == 0) {
|
if(got == 0) {
|
||||||
if(loop) {
|
if(loop) {
|
||||||
v.seekStart();
|
v.seekStart();
|
||||||
|
scf.currentPosition = 0;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
} else {
|
||||||
|
scf.currentPosition += cast(float) got / v.sampleRate;
|
||||||
}
|
}
|
||||||
return !scf.stopped;
|
return !scf.stopped;
|
||||||
}
|
}
|
||||||
|
@ -558,12 +628,20 @@ final class AudioPcmOutThreadImplementation : Thread {
|
||||||
}
|
}
|
||||||
|
|
||||||
/++
|
/++
|
||||||
Requires mp3.d to be compiled in (module arsd.mp3) which is LGPL licensed.
|
Requires mp3.d to be compiled in (module [arsd.mp3]) which is LGPL licensed.
|
||||||
|
That LGPL license will extend to your code.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
An implementation of [SampleController] which lets you pause, etc., the file.
|
||||||
|
|
||||||
|
Please note that the static type may change in the future. It will always be a subtype of [SampleController], but it may be more specialized as I add more features and this will not necessarily match its sister functions, [playOgg] and [playWav], though all three will share an ancestor in [SampleController]. Therefore, if you use `auto`, there's no guarantee the static type won't change in future versions and I will NOT consider that a breaking change since the base interface will remain compatible.
|
||||||
|
|
||||||
History:
|
History:
|
||||||
Automatic resampling support added Nov 7, 2020.
|
Automatic resampling support added Nov 7, 2020.
|
||||||
|
|
||||||
|
Return value changed from `void` to a sample control object on December 23, 2020.
|
||||||
+/
|
+/
|
||||||
SampleControlFlags playMp3()(string filename) {
|
SampleController playMp3()(string filename) {
|
||||||
import arsd.mp3;
|
import arsd.mp3;
|
||||||
|
|
||||||
import std.stdio;
|
import std.stdio;
|
||||||
|
@ -597,10 +675,14 @@ final class AudioPcmOutThreadImplementation : Thread {
|
||||||
if(next.length >= buffer.length) {
|
if(next.length >= buffer.length) {
|
||||||
buffer[] = next[0 .. buffer.length];
|
buffer[] = next[0 .. buffer.length];
|
||||||
next = next[buffer.length .. $];
|
next = next[buffer.length .. $];
|
||||||
|
|
||||||
|
scf.currentPosition += cast(float) buffer.length / mp3.sampleRate / mp3.channels;
|
||||||
} else {
|
} else {
|
||||||
buffer[0 .. next.length] = next[];
|
buffer[0 .. next.length] = next[];
|
||||||
buffer = buffer[next.length .. $];
|
buffer = buffer[next.length .. $];
|
||||||
|
|
||||||
|
scf.currentPosition += cast(float) next.length / mp3.sampleRate / mp3.channels;
|
||||||
|
|
||||||
next = next[$..$];
|
next = next[$..$];
|
||||||
|
|
||||||
if(buffer.length) {
|
if(buffer.length) {
|
||||||
|
@ -680,8 +762,14 @@ final class AudioPcmOutThreadImplementation : Thread {
|
||||||
|
|
||||||
Also requires the resampler to be compiled in at this time, but that may change in the future, I was just lazy.
|
Also requires the resampler to be compiled in at this time, but that may change in the future, I was just lazy.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
An implementation of [SampleController] which lets you pause, etc., the file.
|
||||||
|
|
||||||
|
Please note that the static type may change in the future. It will always be a subtype of [SampleController], but it may be more specialized as I add more features and this will not necessarily match its sister functions, [playMp3] and [playOgg], though all three will share an ancestor in [SampleController]. Therefore, if you use `auto`, there's no guarantee the static type won't change in future versions and I will NOT consider that a breaking change since the base interface will remain compatible.
|
||||||
History:
|
History:
|
||||||
Added Nov 8, 2020.
|
Added Nov 8, 2020.
|
||||||
|
|
||||||
|
Return value changed from `void` to a sample control object on December 23, 2020.
|
||||||
+/
|
+/
|
||||||
SampleController playWav()(string filename) {
|
SampleController playWav()(string filename) {
|
||||||
auto scf = new SampleControlFlags;
|
auto scf = new SampleControlFlags;
|
||||||
|
@ -984,7 +1072,7 @@ final class AudioPcmOutThreadImplementation : Thread {
|
||||||
assert(!err);
|
assert(!err);
|
||||||
}
|
}
|
||||||
|
|
||||||
AudioOutput ao = AudioOutput(0);
|
AudioOutput ao = AudioOutput(device, SampleRate, channels);
|
||||||
this.ao = &ao;
|
this.ao = &ao;
|
||||||
scope(exit) this.ao = null;
|
scope(exit) this.ao = null;
|
||||||
auto omg = this;
|
auto omg = this;
|
||||||
|
@ -1610,7 +1698,7 @@ version(Posix) {
|
||||||
private sigaction_t oldSigIntr;
|
private sigaction_t oldSigIntr;
|
||||||
void setSigIntHandler() {
|
void setSigIntHandler() {
|
||||||
sigaction_t n;
|
sigaction_t n;
|
||||||
n.sa_handler = &interruptSignalHandler;
|
n.sa_handler = &interruptSignalHandlerSAudio;
|
||||||
n.sa_mask = cast(sigset_t) 0;
|
n.sa_mask = cast(sigset_t) 0;
|
||||||
n.sa_flags = 0;
|
n.sa_flags = 0;
|
||||||
sigaction(SIGINT, &n, &oldSigIntr);
|
sigaction(SIGINT, &n, &oldSigIntr);
|
||||||
|
@ -1623,7 +1711,7 @@ version(Posix) {
|
||||||
|
|
||||||
private
|
private
|
||||||
extern(C)
|
extern(C)
|
||||||
void interruptSignalHandler(int sigNumber) nothrow {
|
void interruptSignalHandlerSAudio(int sigNumber) nothrow {
|
||||||
interrupted = true;
|
interrupted = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3728,6 +3816,8 @@ abstract class ResamplingContext {
|
||||||
resamplerDataRight.dataOut = resamplerDataRight.dataOut[1 .. $];
|
resamplerDataRight.dataOut = resamplerDataRight.dataOut[1 .. $];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
scflags.currentPosition += cast(float) buffer.length / outputSampleRate / outputChannels;
|
||||||
} else if(outputChannels == 2) {
|
} else if(outputChannels == 2) {
|
||||||
foreach(idx, ref s; buffer) {
|
foreach(idx, ref s; buffer) {
|
||||||
if(resamplerDataLeft.dataOut.length == 0) {
|
if(resamplerDataLeft.dataOut.length == 0) {
|
||||||
|
@ -3749,6 +3839,8 @@ abstract class ResamplingContext {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
scflags.currentPosition += cast(float) buffer.length / outputSampleRate / outputChannels;
|
||||||
} else assert(0);
|
} else assert(0);
|
||||||
|
|
||||||
return !scflags.stopped;
|
return !scflags.stopped;
|
||||||
|
|
Loading…
Reference in New Issue