2015-07-26 22:08:14 +00:00
|
|
|
"use strict";
|
2013-08-28 04:03:25 +00:00
|
|
|
|
|
|
|
var LibraryRWebAudio = {
|
2013-09-11 02:48:24 +00:00
|
|
|
$RA__deps: ['$Browser', 'usleep'],
|
2013-09-03 01:29:40 +00:00
|
|
|
$RA: {
|
|
|
|
BUFFER_SIZE: 256,
|
|
|
|
|
|
|
|
context: null,
|
|
|
|
buffers: [],
|
|
|
|
numBuffers: 0,
|
|
|
|
bufIndex: 0,
|
|
|
|
bufOffset: 0,
|
|
|
|
startTime: 0,
|
2015-07-26 22:08:14 +00:00
|
|
|
performance: null,
|
2013-09-03 01:29:40 +00:00
|
|
|
nonblock: false,
|
2015-07-26 22:08:14 +00:00
|
|
|
performanceSupported: false,
|
2013-09-03 01:29:40 +00:00
|
|
|
|
|
|
|
setStartTime: function() {
|
2015-07-26 22:08:14 +00:00
|
|
|
RA.performance = window['performance'] || window['webkitPerformance'] || window['msPerformance'] || window['mozPerformance'];
|
|
|
|
if (RA.performance) {
|
|
|
|
RA.performanceSupported = true;
|
|
|
|
} else {
|
|
|
|
RA.performanceSupported = false;
|
|
|
|
}
|
|
|
|
|
2013-09-03 01:29:40 +00:00
|
|
|
if (RA.context.currentTime) {
|
2015-07-26 22:08:14 +00:00
|
|
|
var now = RA.performanceSupported ? RA.performance.now() : Date.now();
|
|
|
|
RA.startTime = now - RA.context.currentTime * 1000;
|
|
|
|
Module["resumeMainLoop"];
|
|
|
|
} else {
|
|
|
|
window.setTimeout(RA.setStartTime, 0);
|
|
|
|
}
|
2013-09-03 01:29:40 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
getCurrentPerfTime: function() {
|
2015-07-26 22:08:14 +00:00
|
|
|
if (true === RA.performanceSupported) {
|
|
|
|
return (RA.performance.now() - RA.startTime) / 1000;
|
|
|
|
} else {
|
|
|
|
return (Date.now() - RA.startTime) / 1000;
|
|
|
|
}
|
2013-09-03 01:29:40 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
process: function(queueBuffers) {
|
|
|
|
var currentTime = RA.getCurrentPerfTime();
|
|
|
|
for (var i = 0; i < RA.bufIndex; i++) {
|
|
|
|
if (RA.buffers[i].endTime < currentTime) {
|
|
|
|
var buf = RA.buffers.splice(i, 1);
|
|
|
|
RA.buffers[RA.numBuffers - 1] = buf[0];
|
|
|
|
i--;
|
|
|
|
RA.bufIndex--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
fillBuffer: function(buf, samples) {
|
|
|
|
var count = 0;
|
|
|
|
var leftBuffer = RA.buffers[RA.bufIndex].getChannelData(0);
|
|
|
|
var rightBuffer = RA.buffers[RA.bufIndex].getChannelData(1);
|
|
|
|
while (samples && RA.bufOffset !== RA.BUFFER_SIZE) {
|
|
|
|
leftBuffer[RA.bufOffset] = {{{ makeGetValue('buf', 'count * 8', 'float') }}};
|
|
|
|
rightBuffer[RA.bufOffset] = {{{ makeGetValue('buf', 'count * 8 + 4', 'float') }}};
|
|
|
|
RA.bufOffset++;
|
|
|
|
count++;
|
|
|
|
samples--;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
},
|
|
|
|
|
|
|
|
queueAudio: function() {
|
|
|
|
var index = RA.bufIndex;
|
|
|
|
|
|
|
|
var startTime;
|
|
|
|
if (RA.bufIndex) startTime = RA.buffers[RA.bufIndex - 1].endTime;
|
|
|
|
else startTime = RA.context.currentTime;
|
2013-11-01 20:23:59 +00:00
|
|
|
RA.buffers[index].endTime = startTime + RA.buffers[index].duration;
|
2013-09-03 01:29:40 +00:00
|
|
|
|
|
|
|
var bufferSource = RA.context.createBufferSource();
|
|
|
|
bufferSource.buffer = RA.buffers[index];
|
|
|
|
bufferSource.connect(RA.context.destination);
|
|
|
|
bufferSource.start(startTime);
|
|
|
|
|
|
|
|
RA.bufIndex++;
|
|
|
|
RA.bufOffset = 0;
|
|
|
|
},
|
|
|
|
|
|
|
|
block: function() {
|
|
|
|
do {
|
|
|
|
RA.process();
|
|
|
|
} while (RA.bufIndex === RA.numBuffers - 1);
|
2013-08-28 04:03:25 +00:00
|
|
|
}
|
2013-09-03 01:29:40 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
RWebAudioInit: function(latency) {
|
2015-07-26 22:08:14 +00:00
|
|
|
Module.pauseMainLoop();
|
|
|
|
if (!RA.context) {
|
|
|
|
var ac = window['AudioContext'] || window['webkitAudioContext'];
|
|
|
|
if (!ac) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
RA.context = new ac();
|
|
|
|
}
|
2013-09-03 01:29:40 +00:00
|
|
|
|
|
|
|
RA.numBuffers = ((latency * RA.context.sampleRate) / (1000 * RA.BUFFER_SIZE))|0;
|
|
|
|
if (RA.numBuffers < 2) RA.numBuffers = 2;
|
|
|
|
|
|
|
|
for (var i = 0; i < RA.numBuffers; i++) RA.buffers[i] = RA.context.createBuffer(2, RA.BUFFER_SIZE, RA.context.sampleRate);
|
|
|
|
|
|
|
|
RA.nonblock = false;
|
|
|
|
RA.startTime = 0;
|
|
|
|
// chrome hack to get currentTime running
|
|
|
|
RA.context.createGain();
|
2015-07-26 22:08:14 +00:00
|
|
|
RA.setStartTime();
|
|
|
|
|
2013-09-03 01:29:40 +00:00
|
|
|
return 1;
|
|
|
|
},
|
|
|
|
|
|
|
|
RWebAudioSampleRate: function() {
|
|
|
|
return RA.context.sampleRate;
|
|
|
|
},
|
|
|
|
|
|
|
|
RWebAudioWrite: function (buf, size) {
|
|
|
|
RA.process();
|
|
|
|
var samples = size / 8;
|
|
|
|
var count = 0;
|
|
|
|
|
|
|
|
while (samples) {
|
|
|
|
var fill = RA.fillBuffer(buf, samples);
|
|
|
|
samples -= fill;
|
|
|
|
count += fill;
|
|
|
|
buf += fill * 8;
|
|
|
|
|
|
|
|
if (RA.bufOffset === RA.BUFFER_SIZE) {
|
|
|
|
if (RA.bufIndex === RA.numBuffers - 1) {
|
|
|
|
if (RA.nonblock) break;
|
|
|
|
else RA.block();
|
|
|
|
}
|
|
|
|
RA.queueAudio();
|
|
|
|
}
|
2013-08-28 04:03:25 +00:00
|
|
|
}
|
|
|
|
|
2013-09-03 01:29:40 +00:00
|
|
|
return count * 8;
|
|
|
|
},
|
|
|
|
|
|
|
|
RWebAudioStop: function() {
|
|
|
|
RA.bufIndex = 0;
|
|
|
|
RA.bufOffset = 0;
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
|
|
|
RWebAudioStart: function() {
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
|
|
|
RWebAudioSetNonblockState: function(state) {
|
|
|
|
RA.nonblock = state;
|
|
|
|
},
|
|
|
|
|
|
|
|
RWebAudioFree: function() {
|
|
|
|
RA.bufIndex = 0;
|
|
|
|
RA.bufOffset = 0;
|
|
|
|
return;
|
|
|
|
},
|
|
|
|
|
|
|
|
RWebAudioBufferSize: function() {
|
|
|
|
return RA.numBuffers * RA.BUFFER_SIZE + RA.BUFFER_SIZE;
|
|
|
|
},
|
|
|
|
|
|
|
|
RWebAudioWriteAvail: function() {
|
|
|
|
RA.process();
|
|
|
|
return ((RA.numBuffers - RA.bufIndex) * RA.BUFFER_SIZE - RA.bufOffset) * 8;
|
|
|
|
}
|
2013-08-28 04:03:25 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
autoAddDeps(LibraryRWebAudio, '$RA');
|
|
|
|
mergeInto(LibraryManager.library, LibraryRWebAudio);
|