blob: 96c5122213bc032bb6863678dd8fb02421f0fe95 [file]
#if AUDIO_WORKLET && !WASM_WORKERS
#error "Building with -sAUDIO_WORKLET also requires enabling -sWASM_WORKERS"
#endif
#if AUDIO_WORKLET && TEXTDECODER == 2
#warning "-sAUDIO_WORKLET does not support -sTEXTDECODER=2 since TextDecoder is not available in AudioWorkletGlobalScope. Text decoding will be unavailable in Audio Worklets. If you need string marshalling in Audio Worklets, build with -sTEXTDECODER=1."
#endif
#if AUDIO_WORKLET && SINGLE_FILE
#error "-sAUDIO_WORKLET does not support -sSINGLE_FILE"
#endif
var LibraryWebAudio = {
$emAudio: {},
$emAudioCounter: 0,
// Call this function from JavaScript to register a Wasm-side handle to an AudioContext that
// you have already created manually without calling emscripten_create_audio_context().
// Note: To let that AudioContext be garbage collected later, call the function
// emscriptenDestroyAudioContext() to unbind it from Wasm.
$emscriptenRegisterAudioObject__deps: ['$emAudio', '$emAudioCounter'],
$emscriptenRegisterAudioObject: (object) => {
#if ASSERTIONS
assert(object, 'null pointer passed to emscriptenRegisterAudioObject');
#endif
emAudio[++emAudioCounter] = object;
#if WEBAUDIO_DEBUG
dbg(`Registered new WebAudio object ${object} with ID ${emAudioCounter}`);
#endif
return emAudioCounter;
},
#if ASSERTIONS || WEBAUDIO_DEBUG
$_emAudioExpectHandle__internal: true,
$_emAudioExpectHandle: (handle, methodName) => {
#if WEBAUDIO_DEBUG
dbg(`called ${methodName}() with ID ${handle}`);
#endif
#if ASSERTIONS
var obj = emAudio[handle];
assert(obj, `${methodName}() called on a nonexisting handle ${handle}`);
return obj;
#endif
},
$emAudioExpectContext__internal: true,
$emAudioExpectContext: (handle, methodName) => {
var obj = _emAudioExpectHandle(handle, methodName);
#if ASSERTIONS
assert(obj instanceof (window.AudioContext || window.webkitAudioContext), `${methodName}() called with ${handle} that is not an AudioContext, but of type ${typeof obj}`);
#endif
},
$emAudioExpectNode__internal: true,
$emAudioExpectNode: (handle, methodName) => {
var obj = _emAudioExpectHandle(handle, methodName);
#if ASSERTIONS
assert(obj instanceof window.AudioNode, `${methodName}() called with a handle ${handle} that is not an AudioNode, but of type ${typeof obj}`);
#endif
},
$emAudioExpectNodeOrContext_internal: true,
$emAudioExpectNodeOrContext: (handle, methodName) => {
var obj = _emAudioExpectHandle(handle, methodName);
#if ASSERTIONS
assert(obj instanceof window.AudioNode || obj instanceof (window.AudioContext || window.webkitAudioContext), `${methodName}() called with a handle ${handle} that is not an AudioContext or AudioNode, but of type ${typeof obj}`);
#endif
},
#endif
// Call this function from JavaScript to destroy a Wasm-side handle to an AudioContext.
// After calling this function, it is no longer possible to reference this AudioContext
// from Wasm code - and the GC can reclaim it after all references to it are cleared.
$emscriptenDestroyAudioContext: 'emscripten_destroy_audio_context',
// Call this function from JavaScript to get the Web Audio object corresponding to the given
// Wasm handle ID.
$emscriptenGetAudioObject: (objectHandle) => emAudio[objectHandle],
// Performs the work of getting the AudioContext's render quantum size.
$emscriptenGetContextQuantumSize: (contextHandle) => {
return emAudio[contextHandle]['renderQuantumSize'] || 128;
},
// emscripten_create_audio_context() does not itself use the
// emscriptenGetAudioObject() function, but mark it as a dependency, because
// the user will not be able to utilize the node unless they call
// emscriptenGetAudioObject() on it on JS side to connect it to the graph, so
// this avoids the user needing to manually add the dependency on the command line.
emscripten_create_audio_context__deps: ['$emscriptenRegisterAudioObject', '$emscriptenGetAudioObject'],
emscripten_create_audio_context: (options) => {
// Safari added unprefixed AudioContext support in Safari 14.5 on iOS: https://caniuse.com/audio-api
#if MIN_SAFARI_VERSION < 140500 || ENVIRONMENT_MAY_BE_NODE || ENVIRONMENT_MAY_BE_SHELL
var ctx = window.AudioContext || window.webkitAudioContext;
#if ASSERTIONS
if (!ctx) console.error('emscripten_create_audio_context failed! Web Audio is not supported.');
#endif
#endif
// Converts AUDIO_CONTEXT_RENDER_SIZE_* into AudioContextRenderSizeCategory
// enums, otherwise returns a positive int value.
function readRenderSizeHint(val) {
return (val < 0) ? 'hardware' : (val || 'default');
}
var opts = options ? {
latencyHint: UTF8ToString({{{ makeGetValue('options', C_STRUCTS.EmscriptenWebAudioCreateAttributes.latencyHint, '*') }}}) || undefined,
sampleRate: {{{ makeGetValue('options', C_STRUCTS.EmscriptenWebAudioCreateAttributes.sampleRate, 'u32') }}} || undefined,
renderSizeHint: readRenderSizeHint({{{ makeGetValue('options', C_STRUCTS.EmscriptenWebAudioCreateAttributes.renderSizeHint, 'i32') }}})
} : undefined;
#if WEBAUDIO_DEBUG
dbg(`Creating new WebAudio context with parameters:`);
console.dir(opts);
#endif
#if MIN_SAFARI_VERSION < 140500 || ENVIRONMENT_MAY_BE_NODE || ENVIRONMENT_MAY_BE_SHELL
return ctx && emscriptenRegisterAudioObject(new ctx(opts));
#else
// We are targeting an environment where we assume that AudioContext() API unconditionally exists.
return emscriptenRegisterAudioObject(new AudioContext(opts));
#endif
},
emscripten_resume_audio_context_async: (contextHandle, callback, userData) => {
function cb(state) {
#if WEBAUDIO_DEBUG
console.log(`emscripten_resume_audio_context_async() callback: New audio state="${emAudio[contextHandle].state}", ID=${state}`);
#endif
{{{ makeDynCall('viip', 'callback') }}}(contextHandle, state, userData);
}
#if WEBAUDIO_DEBUG
dbg('emscripten_resume_audio_context_async() resuming...');
#endif
emAudio[contextHandle].resume().then(() => { cb(1/*running*/) }).catch(() => { cb(0/*suspended*/) });
},
emscripten_resume_audio_context_sync: (contextHandle) => {
#if ASSERTIONS || WEBAUDIO_DEBUG
emAudioExpectContext(contextHandle, 'emscripten_resume_audio_context_sync');
#endif
emAudio[contextHandle].resume();
},
emscripten_audio_context_state: (contextHandle) => {
#if ASSERTIONS || WEBAUDIO_DEBUG
emAudioExpectContext(contextHandle, 'emscripten_audio_context_state');
#endif
return ['suspended', 'running', 'closed', 'interrupted'].indexOf(emAudio[contextHandle].state);
},
emscripten_destroy_audio_context: (contextHandle) => {
#if ASSERTIONS || WEBAUDIO_DEBUG
emAudioExpectContext(contextHandle, 'emscripten_destroy_audio_context');
#endif
emAudio[contextHandle].suspend();
delete emAudio[contextHandle];
},
emscripten_destroy_web_audio_node: (objectHandle) => {
#if ASSERTIONS || WEBAUDIO_DEBUG
emAudioExpectNode(objectHandle, 'emscripten_destroy_web_audio_node');
#endif
// Explicitly disconnect the node from Web Audio graph before letting it GC,
// to work around browser bugs such as https://webkit.org/b/222098#c23
emAudio[objectHandle].disconnect();
delete emAudio[objectHandle];
},
#if AUDIO_WORKLET
// _emscripten_create_audio_worklet() doesn't use stackAlloc,
// etc., but the created worklet does.
_emscripten_create_audio_worklet__deps: [
'$_emAudioDispatchProcessorCallback',
'$stackAlloc', '$stackRestore', '$stackSave'],
_emscripten_create_audio_worklet: (wwID, contextHandle, stackLowestAddress, stackSize, callback, userData) => {
#if ASSERTIONS || WEBAUDIO_DEBUG
emAudioExpectContext(contextHandle, '_emscripten_create_audio_worklet');
#endif
var audioContext = emAudio[contextHandle];
var audioWorklet = audioContext.audioWorklet;
#if ASSERTIONS
assert(stackLowestAddress != 0, 'AudioWorklets require a dedicated stack space for audio data marshalling between Wasm and JS');
assert(stackLowestAddress % 16 == 0, `AudioWorklet stack should be aligned to 16 bytes (was ${stackLowestAddress} == ${stackLowestAddress%16} mod 16) Use e.g. memalign(16, stackSize) to align the stack`);
assert(stackSize != 0, 'AudioWorklets require a dedicated stack space for audio data marshalling between Wasm and JS');
assert(stackSize % 16 == 0, `AudioWorklet stack size should be a multiple of 16 bytes! (was ${stackSize} == ${stackSize%16} mod 16)`);
assert(!audioContext.audioWorkletInitialized, `emscripten_create_wasm_audio_worklet() was already called for AudioContext ${contextHandle}! Only call this function once per AudioContext`);
audioContext.audioWorkletInitialized = 1;
#endif
#if WEBAUDIO_DEBUG
dbg(`_emscripten_create_audio_worklet() adding audioworklet.js...`);
#endif
var audioWorkletCreationFailed = () => {
#if ASSERTIONS || WEBAUDIO_DEBUG
dbg(`_emscripten_create_audio_worklet() addModule() failed!`);
#endif
{{{ makeDynCall('viip', 'callback') }}}(contextHandle, 0/*EM_FALSE*/, userData);
};
// Does browser not support AudioWorklets?
if (!audioWorklet) {
#if ASSERTIONS || WEBAUDIO_DEBUG
if (location.protocol == 'http:') {
console.error(`AudioWorklets are not supported. This is possibly due to running the page over unsecure http:// protocol. Try running over https://, or debug via a localhost-based server, which should also allow AudioWorklets to function.`);
} else {
console.error(`AudioWorklets are not supported by current browser.`);
}
#endif
return audioWorkletCreationFailed();
}
audioWorklet.addModule({{{ wasmWorkerJs }}}).then(() => {
#if WEBAUDIO_DEBUG
dbg(`_emscripten_create_audio_worklet() addModule() completed`);
#endif
#if MIN_FIREFOX_VERSION < 138 || MIN_CHROME_VERSION != TARGET_NOT_SUPPORTED || MIN_SAFARI_VERSION != TARGET_NOT_SUPPORTED
// If this browser does not support the up-to-date AudioWorklet standard
// that has a MessagePort over to the AudioWorklet, then polyfill that by
// instantiating a dummy AudioWorkletNode to get a MessagePort over.
// Firefox added support in https://hg-edge.mozilla.org/integration/autoland/rev/ab38a1796126f2b3fc06475ffc5a625059af59c1
// Chrome ticket: https://crbug.com/446920095
// Safari ticket: https://webkit.org/b/299386
if (!audioWorklet.port) {
audioWorklet.port = {
postMessage: (msg) => {
if (msg['_boot']) {
audioWorklet.bootstrapMessage = new AudioWorkletNode(audioContext, 'em-bootstrap', {
processorOptions: msg
});
audioWorklet.bootstrapMessage.port.onmessage = (msg) => {
audioWorklet.port.onmessage(msg);
}
} else {
audioWorklet.bootstrapMessage.port.postMessage(msg);
}
}
}
}
#endif
audioWorklet.port.postMessage({
// This is the bootstrap message to the Audio Worklet.
'_boot': 1,
// Assign the loaded AudioWorkletGlobalScope a Wasm Worker ID so that
// it can utilized its own TLS slots, and it is recognized to not be
// the main browser thread.
wwID,
#if MINIMAL_RUNTIME
wasm: Module['wasm'],
#else
wasm: wasmModule,
#endif
wasmMemory,
stackLowestAddress, // sb = stack base
stackSize, // sz = stack size
});
audioWorklet.port.onmessage = _emAudioDispatchProcessorCallback;
{{{ makeDynCall('viip', 'callback') }}}(contextHandle, 1/*EM_TRUE*/, userData);
}).catch(audioWorkletCreationFailed);
},
$_emAudioDispatchProcessorCallback__deps: ['$getWasmTableEntry'],
$_emAudioDispatchProcessorCallback: (e) => {
var data = e.data;
// '_wsc' is short for 'wasm call', trying to use an identifier name that
// will never conflict with user code. This is used to call both the 3-param
// call (handle, true, userData) and the variable argument post functions.
var wasmCall = data['_wsc'];
wasmCall && getWasmTableEntry(wasmCall)(...data.args);
},
emscripten_create_wasm_audio_worklet_processor_async: (contextHandle, options, callback, userData) => {
#if ASSERTIONS || WEBAUDIO_DEBUG
emAudioExpectContext(contextHandle, 'emscripten_create_wasm_audio_worklet_processor_async');
#endif
var processorName = UTF8ToString({{{ makeGetValue('options', C_STRUCTS.WebAudioWorkletProcessorCreateOptions.name, '*') }}});
#if AUDIO_WORKLET_SUPPORT_AUDIO_PARAMS
var numAudioParams = {{{ makeGetValue('options', C_STRUCTS.WebAudioWorkletProcessorCreateOptions.numAudioParams, 'i32') }}};
var audioParamDescriptors = {{{ makeGetValue('options', C_STRUCTS.WebAudioWorkletProcessorCreateOptions.audioParamDescriptors, '*') }}};
var audioParams = [];
var paramIndex = 0;
while (numAudioParams--) {
audioParams.push({
name: paramIndex++,
defaultValue: {{{ makeGetValue('audioParamDescriptors', C_STRUCTS.WebAudioParamDescriptor.defaultValue, 'float') }}},
minValue: {{{ makeGetValue('audioParamDescriptors', C_STRUCTS.WebAudioParamDescriptor.minValue, 'float') }}},
maxValue: {{{ makeGetValue('audioParamDescriptors', C_STRUCTS.WebAudioParamDescriptor.maxValue, 'float') }}},
automationRate: ({{{ makeGetValue('audioParamDescriptors', C_STRUCTS.WebAudioParamDescriptor.automationRate, 'i32') }}} ? 'k' : 'a') + '-rate',
});
audioParamDescriptors += {{{ C_STRUCTS.WebAudioParamDescriptor.__size__ }}};
}
#elif ASSERTIONS
var numAudioParams = {{{ makeGetValue('options', C_STRUCTS.WebAudioWorkletProcessorCreateOptions.numAudioParams, 'i32') }}};
assert(numAudioParams == 0 && "Rebuild with -sAUDIO_WORKLET_SUPPORT_AUDIO_PARAMS to utilize AudioParams");
#endif
#if WEBAUDIO_DEBUG
console.log(`emscripten_create_wasm_audio_worklet_processor_async() creating a new AudioWorklet processor with name ${processorName}`);
#endif
emAudio[contextHandle].audioWorklet.port.postMessage({
// Deliberately mangled and short names used here ('_wpn', the 'Worklet
// Processor Name' used as a 'key' to verify the message type so as to
// not get accidentally mixed with user submitted messages, the remainder
// for space saving reasons, abbreviated from their variable names).
'_wpn': processorName,
#if AUDIO_WORKLET_SUPPORT_AUDIO_PARAMS
audioParams,
#endif
contextHandle,
callback,
userData,
});
},
emscripten_create_wasm_audio_worklet_node__deps: ['$emscriptenGetContextQuantumSize'],
emscripten_create_wasm_audio_worklet_node: (contextHandle, name, options, callback, userData) => {
#if ASSERTIONS || WEBAUDIO_DEBUG
emAudioExpectContext(contextHandle, 'emscripten_create_wasm_audio_worklet_node');
#endif
function readChannelCountArray(heapIndex, numOutputs) {
if (!heapIndex) return undefined;
heapIndex = {{{ getHeapOffset('heapIndex', 'i32') }}};
var channelCounts = [];
while (numOutputs--) channelCounts.push(HEAPU32[heapIndex++]);
return channelCounts;
}
var optionsOutputs = options ? {{{ makeGetValue('options', C_STRUCTS.EmscriptenAudioWorkletNodeCreateOptions.numberOfOutputs, 'i32') }}} : 0;
var opts = options ? {
numberOfInputs: {{{ makeGetValue('options', C_STRUCTS.EmscriptenAudioWorkletNodeCreateOptions.numberOfInputs, 'i32') }}},
numberOfOutputs: optionsOutputs,
outputChannelCount: readChannelCountArray({{{ makeGetValue('options', C_STRUCTS.EmscriptenAudioWorkletNodeCreateOptions.outputChannelCounts, 'i32*') }}}, optionsOutputs),
channelCount: {{{ makeGetValue('options', C_STRUCTS.EmscriptenAudioWorkletNodeCreateOptions.channelCount, 'u32') }}} || undefined,
channelCountMode: [/*'max'*/,'clamped-max','explicit'][{{{ makeGetValue('options', C_STRUCTS.EmscriptenAudioWorkletNodeCreateOptions.channelCountMode, 'i32') }}}],
channelInterpretation: [/*'speakers'*/,'discrete'][{{{ makeGetValue('options', C_STRUCTS.EmscriptenAudioWorkletNodeCreateOptions.channelInterpretation, 'i32') }}}],
processorOptions: {
callback,
userData,
samplesPerChannel: emscriptenGetContextQuantumSize(contextHandle),
}
} : undefined;
#if WEBAUDIO_DEBUG
dbg(`Creating AudioWorkletNode "${UTF8ToString(name)}" on context=${contextHandle} with options:`);
console.dir(opts);
#endif
return emscriptenRegisterAudioObject(new AudioWorkletNode(emAudio[contextHandle], UTF8ToString(name), opts));
},
#endif // ~AUDIO_WORKLET
emscripten_audio_context_quantum_size__deps: ['$emscriptenGetContextQuantumSize'],
emscripten_audio_context_quantum_size: (contextHandle) => {
#if ASSERTIONS || WEBAUDIO_DEBUG
emAudioExpectContext(contextHandle, 'emscripten_audio_context_quantum_size')
#endif
return emscriptenGetContextQuantumSize(contextHandle);
},
emscripten_audio_context_sample_rate: (contextHandle) => {
#if ASSERTIONS || WEBAUDIO_DEBUG
emAudioExpectContext(contextHandle, 'emscripten_audio_context_sample_rate');
#endif
return emAudio[contextHandle]['sampleRate'];
},
emscripten_audio_node_connect: (source, destination, outputIndex, inputIndex) => {
#if ASSERTIONS || WEBAUDIO_DEBUG
emAudioExpectNode(source, 'emscripten_audio_node_connect');
emAudioExpectNodeOrContext(destination, 'emscripten_audio_node_connect');
#endif
var srcNode = emAudio[source];
var dstNode = emAudio[destination];
#if WEBAUDIO_DEBUG
dbg(`Connecting audio node ID ${source} to audio node ID ${destination} (${srcNode} to ${dstNode})`);
#endif
srcNode.connect(dstNode.destination || dstNode, outputIndex, inputIndex);
},
$emAudioWorkletPostFunction__internal: true,
$emAudioWorkletPostFunction(audioContext, funcPtr, args) {
#if ASSERTIONS
assert(funcPtr);
#endif
#if ASSERTIONS || WEBAUDIO_DEBUG
if (audioContext) emAudioExpectContext(audioContext, 'emAudioWorkletPostFunction');
#endif
// _wsc = "WaSm Call"
(audioContext ? emAudio[audioContext].audioWorklet.port : port).postMessage({'_wsc': funcPtr, args});
},
emscripten_current_thread_is_audio_worklet: () => ENVIRONMENT_IS_AUDIO_WORKLET,
emscripten_audio_worklet_post_function_v__deps: ['$emAudioWorkletPostFunction'],
emscripten_audio_worklet_post_function_v: (audioContext, funcPtr) => {
emAudioWorkletPostFunction(audioContext, funcPtr, []);
},
$emscripten_audio_worklet_post_function_1__deps: ['$emAudioWorkletPostFunction'],
$emscripten_audio_worklet_post_function_1: (audioContext, funcPtr, arg0) => {
emAudioWorkletPostFunction(audioContext, funcPtr, [arg0]);
},
emscripten_audio_worklet_post_function_vi__deps: ['$emscripten_audio_worklet_post_function_1'],
emscripten_audio_worklet_post_function_vi(audioContext, funcPtr, arg0) {
emscripten_audio_worklet_post_function_1(audioContext, funcPtr, arg0)
},
emscripten_audio_worklet_post_function_vd__deps: ['$emscripten_audio_worklet_post_function_1'],
emscripten_audio_worklet_post_function_vd(audioContext, funcPtr, arg0) {
emscripten_audio_worklet_post_function_1(audioContext, funcPtr, arg0)
},
$emscripten_audio_worklet_post_function_2__deps: ['$emAudioWorkletPostFunction'],
$emscripten_audio_worklet_post_function_2: (audioContext, funcPtr, arg0, arg1) => {
emAudioWorkletPostFunction(audioContext, funcPtr, [arg0, arg1]);
},
emscripten_audio_worklet_post_function_vii__deps: ['$emscripten_audio_worklet_post_function_2'],
emscripten_audio_worklet_post_function_vii: (audioContext, funcPtr, arg0, arg1) => {
emscripten_audio_worklet_post_function_2(audioContext, funcPtr, arg0, arg1);
},
emscripten_audio_worklet_post_function_vdd__deps: ['$emscripten_audio_worklet_post_function_2'],
emscripten_audio_worklet_post_function_vdd: (audioContext, funcPtr, arg0, arg1) => {
emscripten_audio_worklet_post_function_2(audioContext, funcPtr, arg0, arg1);
},
$emscripten_audio_worklet_post_function_3__deps: ['$emAudioWorkletPostFunction'],
$emscripten_audio_worklet_post_function_3: (audioContext, funcPtr, arg0, arg1, arg2) => {
emAudioWorkletPostFunction(audioContext, funcPtr, [arg0, arg1, arg2]);
},
emscripten_audio_worklet_post_function_viii__deps: ['$emscripten_audio_worklet_post_function_3'],
emscripten_audio_worklet_post_function_viii: (audioContext, funcPtr, arg0, arg1, arg2) => {
emscripten_audio_worklet_post_function_3(audioContext, funcPtr, arg0, arg1, arg2);
},
emscripten_audio_worklet_post_function_vddd__deps: ['$emscripten_audio_worklet_post_function_3'],
emscripten_audio_worklet_post_function_vddd: (audioContext, funcPtr, arg0, arg1, arg2) => {
emscripten_audio_worklet_post_function_3(audioContext, funcPtr, arg0, arg1, arg2);
},
emscripten_audio_worklet_post_function_sig__deps: ['$readEmAsmArgs', '$emAudioWorkletPostFunction'],
emscripten_audio_worklet_post_function_sig: (audioContext, funcPtr, sigPtr, varargs) => {
#if ASSERTIONS
assert(sigPtr);
assert(UTF8ToString(sigPtr)[0] != 'v', 'emscripten_audio_worklet_post_function_sig() supports only void return type');
assert(varargs);
#endif
emAudioWorkletPostFunction(audioContext, funcPtr, readEmAsmArgs(sigPtr, varargs));
}
};
#if ASSERTIONS || WEBAUDIO_DEBUG
autoAddDeps(LibraryWebAudio, '$_emAudioExpectHandle');
autoAddDeps(LibraryWebAudio, '$emAudioExpectNode');
autoAddDeps(LibraryWebAudio, '$emAudioExpectContext');
autoAddDeps(LibraryWebAudio, '$emAudioExpectNodeOrContext');
#endif
addToLibrary(LibraryWebAudio);