wam-community
Version:
A collection of prebuilt Web Audio Modules ready for use
919 lines (807 loc) • 42.6 kB
JavaScript
/*
Code generated with Faust version 2.40.7
Compilation options: -lang wasm-ib -cn OscTube -es 1 -mcd 16 -single -ftz 2
*/
function getJSONOscTube() {
return '{"name": "OscTube","filename": "OscTube.dsp","version": "2.40.7","compile_options": "-lang wasm-ib -cn OscTube -es 1 -mcd 16 -single -ftz 2","include_pathnames": ["/usr/local/share/faust","/usr/local/share/faust","/usr/share/faust",".","/tmp/sessions/601B8B9174BDD04CFAF3092BCC29C3F0194B4737/web/wap"],"size": 16560,"inputs": 1,"outputs": 1,"meta": [ { "basics_lib_name": "Faust Basic Element Library" },{ "basics_lib_version": "0.6" },{ "compilation_options": "-single -scal -I libraries/ -I project/ -lang wasm" },{ "compile_options": "-lang wasm-ib -cn OscTube -es 1 -mcd 16 -single -ftz 2" },{ "effect_lib_author": "Julius O. Smith (jos at ccrma.stanford.edu)" },{ "effect_lib_copyright": "Julius O. Smith III" },{ "effect_lib_deprecated": "This library is deprecated and is not maintained anymore. It will be removed in August 2017." },{ "effect_lib_exciter_author": "Priyanka Shekar (pshekar@ccrma.stanford.edu)" },{ "effect_lib_exciter_copyright": "Copyright (c) 2013 Priyanka Shekar" },{ "effect_lib_exciter_license": "MIT License (MIT)" },{ "effect_lib_exciter_name": "Harmonic Exciter" },{ "effect_lib_exciter_version": "1.0" },{ "effect_lib_license": "STK-4.3" },{ "effect_lib_name": "Faust Audio Effect Library" },{ "effect_lib_version": "1.33" },{ "filename": "OscTube.dsp" },{ "filter_lib_author": "Julius O. Smith (jos at ccrma.stanford.edu)" },{ "filter_lib_copyright": "Julius O. Smith III" },{ "filter_lib_deprecated": "This library is deprecated and is not maintained anymore. It will be removed in August 2017." },{ "filter_lib_license": "STK-4.3" },{ "filter_lib_name": "Faust Filter Library" },{ "filter_lib_reference": "https://ccrma.stanford.edu/~jos/filters/" },{ "filter_lib_version": "1.29" },{ "id": "tube2" },{ "library_path0": "/libraries/maxmsp.lib" },{ "library_path1": "/libraries/stdfaust.lib" },{ "library_path2": "/libraries/effect.lib" },{ "library_path3": "/libraries/filter.lib" },{ "library_path4": "/libraries/music.lib" },{ "library_path5": "/libraries/math.lib" },{ "library_path6": "/libraries/basics.lib" },{ "library_path7": "/libraries/maths.lib" },{ "library_path8": "/libraries/platform.lib" },{ "math_lib_author": "GRAME" },{ "math_lib_copyright": "GRAME" },{ "math_lib_deprecated": "This library is deprecated and is not maintained anymore. It will be removed in August 2017." },{ "math_lib_license": "LGPL with exception" },{ "math_lib_name": "Math Library" },{ "math_lib_version": "1.0" },{ "maths_lib_author": "GRAME" },{ "maths_lib_copyright": "GRAME" },{ "maths_lib_license": "LGPL with exception" },{ "maths_lib_name": "Faust Math Library" },{ "maths_lib_version": "2.5" },{ "maxmsp_lib_author": "GRAME" },{ "maxmsp_lib_copyright": "GRAME" },{ "maxmsp_lib_license": "LGPL with exception" },{ "maxmsp_lib_name": "MaxMSP compatibility Library" },{ "maxmsp_lib_version": "1.1" },{ "music_lib_author": "GRAME" },{ "music_lib_copyright": "GRAME" },{ "music_lib_deprecated": "This library is deprecated and is not maintained anymore. It will be removed in August 2017." },{ "music_lib_license": "LGPL with exception" },{ "music_lib_name": "Music Library" },{ "music_lib_version": "1.0" },{ "name": "OscTube" },{ "platform_lib_name": "Generic Platform Library" },{ "platform_lib_version": "0.2" }],"ui": [ {"type": "vgroup","label": "OscTube","items": [ {"type": "vslider","label": "Fuzz","address": "/OscTube/Fuzz","index": 32,"meta": [{ "OWL": "PARAMETER_A" },{ "name": "tube" },{ "style": "knob" }],"init": 1,"min": -3,"max": 10,"step": 1},{"type": "vslider","label": "Resonance","address": "/OscTube/Resonance","index": 44,"meta": [{ "OWL": "PARAMETER_B" },{ "style": "knob" }],"init": 0.5,"min": 0,"max": 0.9,"step": 0.01},{"type": "vslider","label": "Vibrato","address": "/OscTube/Vibrato","index": 16472,"meta": [{ "OWL": "PARAMETER_C" },{ "style": "knob" }],"init": 1,"min": 0,"max": 1,"step": 0.01},{"type": "checkbox","label": "bypass","address": "/OscTube/bypass","index": 16}]}]}';
}
function getBase64CodeOscTube() {
return "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cy8gLUkgcHJvamVjdC8gLWxhbmcgd2FzbSIgfSx7ICJjb21waWxlX29wdGlvbnMiOiAiLWxhbmcgd2FzbS1pYiAtY24gT3NjVHViZSAtZXMgMSAtbWNkIDE2IC1zaW5nbGUgLWZ0eiAyIiB9LHsgImVmZmVjdF9saWJfYXV0aG9yIjogIkp1bGl1cyBPLiBTbWl0aCAoam9zIGF0IGNjcm1hLnN0YW5mb3JkLmVkdSkiIH0seyAiZWZmZWN0X2xpYl9jb3B5cmlnaHQiOiAiSnVsaXVzIE8uIFNtaXRoIElJSSIgfSx7ICJlZmZlY3RfbGliX2RlcHJlY2F0ZWQiOiAiVGhpcyBsaWJyYXJ5IGlzIGRlcHJlY2F0ZWQgYW5kIGlzIG5vdCBtYWludGFpbmVkIGFueW1vcmUuIEl0IHdpbGwgYmUgcmVtb3ZlZCBpbiBBdWd1c3QgMjAxNy4iIH0seyAiZWZmZWN0X2xpYl9leGNpdGVyX2F1dGhvciI6ICJQcml5YW5rYSBTaGVrYXIgKHBzaGVrYXJAY2NybWEuc3RhbmZvcmQuZWR1KSIgfSx7ICJlZmZlY3RfbGliX2V4Y2l0ZXJfY29weXJpZ2h0IjogIkNvcHlyaWdodCAoYykgMjAxMyBQcml5YW5rYSBTaGVrYXIiIH0seyAiZWZmZWN0X2xpYl9leGNpdGVyX2xpY2Vuc2UiOiAiTUlUIExpY2Vuc2UgKE1JVCkiIH0seyAiZWZmZWN0X2xpYl9leGNpdGVyX25hbWUiOiAiSGFybW9uaWMgRXhjaXRlciIgfSx7ICJlZmZlY3RfbGliX2V4Y2l0ZXJfdmVyc2lvbiI6ICIxLjAiIH0seyAiZWZmZWN0X2xpYl9saWNlbnNlIjogIlNUSy00LjMiIH0seyAiZWZmZWN0X2xpYl9uYW1lIjogIkZhdXN0IEF1ZGlvIEVmZmVjdCBMaWJyYXJ5IiB9LHsgImVmZmVjdF9saWJfdmVyc2lvbiI6ICIxLjMzIiB9LHsgImZpbGVuYW1lIjogIk9zY1R1YmUuZHNwIiB9LHsgImZpbHRlcl9saWJfYXV0aG9yIjogIkp1bGl1cyBPLiBTbWl0aCAoam9zIGF0IGNjcm1hLnN0YW5mb3JkLmVkdSkiIH0seyAiZmlsdGVyX2xpYl9jb3B5cmlnaHQiOiAiSnVsaXVzIE8uIFNtaXRoIElJSSIgfSx7ICJmaWx0ZXJfbGliX2RlcHJlY2F0ZWQiOiAiVGhpcyBsaWJyYXJ5IGlzIGRlcHJlY2F0ZWQgYW5kIGlzIG5vdCBtYWludGFpbmVkIGFueW1vcmUuIEl0IHdpbGwgYmUgcmVtb3ZlZCBpbiBBdWd1c3QgMjAxNy4iIH0seyAiZmlsdGVyX2xpYl9saWNlbnNlIjogIlNUSy00LjMiIH0seyAiZmlsdGVyX2xpYl9uYW1lIjogIkZhdXN0IEZpbHRlciBMaWJyYXJ5IiB9LHsgImZpbHRlcl9saWJfcmVmZXJlbmNlIjogImh0dHBzOi8vY2NybWEuc3RhbmZvcmQuZWR1L35qb3MvZmlsdGVycy8iIH0seyAiZmlsdGVyX2xpYl92ZXJzaW9uIjogIjEuMjkiIH0seyAiaWQiOiAidHViZTIiIH0seyAibGlicmFyeV9wYXRoMCI6ICIvbGlicmFyaWVzL21heG1zcC5saWIiIH0seyAibGlicmFyeV9wYXRoMSI6ICIvbGlicmFyaWVzL3N0ZGZhdXN0LmxpYiIgfSx7ICJsaWJyYXJ5X3BhdGgyIjogIi9saWJyYXJpZXMvZWZmZWN0LmxpYiIgfSx7ICJsaWJyYXJ5X3BhdGgzIjogIi9saWJyYXJpZXMvZmlsdGVyLmxpYiIgfSx7ICJsaWJyYXJ5X3BhdGg0IjogIi9saWJyYXJpZXMvbXVzaWMubGliIiB9LHsgImxpYnJhcnlfcGF0aDUiOiAiL2xpYnJhcmllcy9tYXRoLmxpYiIgfSx7ICJsaWJyYXJ5X3BhdGg2IjogIi9saWJyYXJpZXMvYmFzaWNzLmxpYiIgfSx7ICJsaWJyYXJ5X3BhdGg3IjogIi9saWJyYXJpZXMvbWF0aHMubGliIiB9LHsgImxpYnJhcnlfcGF0aDgiOiAiL2xpYnJhcmllcy9wbGF0Zm9ybS5saWIiIH0seyAibWF0aF9saWJfYXV0aG9yIjogIkdSQU1FIiB9LHsgIm1hdGhfbGliX2NvcHlyaWdodCI6ICJHUkFNRSIgfSx7ICJtYXRoX2xpYl9kZXByZWNhdGVkIjogIlRoaXMgbGlicmFyeSBpcyBkZXByZWNhdGVkIGFuZCBpcyBub3QgbWFpbnRhaW5lZCBhbnltb3JlLiBJdCB3aWxsIGJlIHJlbW92ZWQgaW4gQXVndXN0IDIwMTcuIiB9LHsgIm1hdGhfbGliX2xpY2Vuc2UiOiAiTEdQTCB3aXRoIGV4Y2VwdGlvbiIgfSx7ICJtYXRoX2xpYl9uYW1lIjogIk1hdGggTGlicmFyeSIgfSx7ICJtYXRoX2xpYl92ZXJzaW9uIjogIjEuMCIgfSx7ICJtYXRoc19saWJfYXV0aG9yIjogIkdSQU1FIiB9LHsgIm1hdGhzX2xpYl9jb3B5cmlnaHQiOiAiR1JBTUUiIH0seyAibWF0aHNfbGliX2xpY2Vuc2UiOiAiTEdQTCB3aXRoIGV4Y2VwdGlvbiIgfSx7ICJtYXRoc19saWJfbmFtZSI6ICJGYXVzdCBNYXRoIExpYnJhcnkiIH0seyAibWF0aHNfbGliX3ZlcnNpb24iOiAiMi41IiB9LHsgIm1heG1zcF9saWJfYXV0aG9yIjogIkdSQU1FIiB9LHsgIm1heG1zcF9saWJfY29weXJpZ2h0IjogIkdSQU1FIiB9LHsgIm1heG1zcF9saWJfbGljZW5zZSI6ICJMR1BMIHdpdGggZXhjZXB0aW9uIiB9LHsgIm1heG1zcF9saWJfbmFtZSI6ICJNYXhNU1AgY29tcGF0aWJpbGl0eSBMaWJyYXJ5IiB9LHsgIm1heG1zcF9saWJfdmVyc2lvbiI6ICIxLjEiIH0seyAibXVzaWNfbGliX2F1dGhvciI6ICJHUkFNRSIgfSx7ICJtdXNpY19saWJfY29weXJpZ2h0IjogIkdSQU1FIiB9LHsgIm11c2ljX2xpYl9kZXByZWNhdGVkIjogIlRoaXMgbGlicmFyeSBpcyBkZXByZWNhdGVkIGFuZCBpcyBub3QgbWFpbnRhaW5lZCBhbnltb3JlLiBJdCB3aWxsIGJlIHJlbW92ZWQgaW4gQXVndXN0IDIwMTcuIiB9LHsgIm11c2ljX2xpYl9saWNlbnNlIjogIkxHUEwgd2l0aCBleGNlcHRpb24iIH0seyAibXVzaWNfbGliX25hbWUiOiAiTXVzaWMgTGlicmFyeSIgfSx7ICJtdXNpY19saWJfdmVyc2lvbiI6ICIxLjAiIH0seyAibmFtZSI6ICJPc2NUdWJlIiB9LHsgInBsYXRmb3JtX2xpYl9uYW1lIjogIkdlbmVyaWMgUGxhdGZvcm0gTGlicmFyeSIgfSx7ICJwbGF0Zm9ybV9saWJfdmVyc2lvbiI6ICIwLjIiIH1dLCJ1aSI6IFsgeyJ0eXBlIjogInZncm91cCIsImxhYmVsIjogIk9zY1R1YmUiLCJpdGVtcyI6IFsgeyJ0eXBlIjogInZzbGlkZXIiLCJsYWJlbCI6ICJGdXp6IiwiYWRkcmVzcyI6ICIvT3NjVHViZS9GdXp6IiwiaW5kZXgiOiAzMiwibWV0YSI6IFt7ICJPV0wiOiAiUEFSQU1FVEVSX0EiIH0seyAibmFtZSI6ICJ0dWJlIiB9LHsgInN0eWxlIjogImtub2IiIH1dLCJpbml0IjogMSwibWluIjogLTMsIm1heCI6IDEwLCJzdGVwIjogMX0seyJ0eXBlIjogInZzbGlkZXIiLCJsYWJlbCI6ICJSZXNvbmFuY2UiLCJhZGRyZXNzIjogIi9Pc2NUdWJlL1Jlc29uYW5jZSIsImluZGV4IjogNDQsIm1ldGEiOiBbeyAiT1dMIjogIlBBUkFNRVRFUl9CIiB9LHsgInN0eWxlIjogImtub2IiIH1dLCJpbml0IjogMC41LCJtaW4iOiAwLCJtYXgiOiAwLjksInN0ZXAiOiAwLjAxfSx7InR5cGUiOiAidnNsaWRlciIsImxhYmVsIjogIlZpYnJhdG8iLCJhZGRyZXNzIjogIi9Pc2NUdWJlL1ZpYnJhdG8iLCJpbmRleCI6IDE2NDcyLCJtZXRhIjogW3sgIk9XTCI6ICJQQVJBTUVURVJfQyIgfSx7ICJzdHlsZSI6ICJrbm9iIiB9XSwiaW5pdCI6IDEsIm1pbiI6IDAsIm1heCI6IDEsInN0ZXAiOiAwLjAxfSx7InR5cGUiOiAiY2hlY2tib3giLCJsYWJlbCI6ICJieXBhc3MiLCJhZGRyZXNzIjogIi9Pc2NUdWJlL2J5cGFzcyIsImluZGV4IjogMTZ9XX1dfQ==";
}
/*
faust2wasm: GRAME 2017-2019
*/
("use strict");
if (typeof AudioWorkletNode === "undefined") {
alert("AudioWorklet is not supported in this browser !");
}
class OscTubeNode extends AudioWorkletNode {
constructor(context, baseURL, options) {
super(context, "OscTube", options);
this.baseURL = baseURL;
this.json = options.processorOptions.json;
this.json_object = JSON.parse(this.json);
// JSON parsing functions
this.parse_ui = function (ui, obj) {
for (var i = 0; i < ui.length; i++) {
this.parse_group(ui[i], obj);
}
};
this.parse_group = function (group, obj) {
if (group.items) {
this.parse_items(group.items, obj);
}
};
this.parse_items = function (items, obj) {
for (var i = 0; i < items.length; i++) {
this.parse_item(items[i], obj);
}
};
this.parse_item = function (item, obj) {
if (item.type === "vgroup" || item.type === "hgroup" || item.type === "tgroup") {
this.parse_items(item.items, obj);
} else if (item.type === "hbargraph" || item.type === "vbargraph") {
// Keep bargraph adresses
obj.outputs_items.push(item.address);
} else if (
item.type === "vslider" ||
item.type === "hslider" ||
item.type === "button" ||
item.type === "checkbox" ||
item.type === "nentry"
) {
// Keep inputs adresses
obj.inputs_items.push(item.address);
obj.descriptor.push(item);
// Decode MIDI
if (item.meta !== undefined) {
for (var i = 0; i < item.meta.length; i++) {
if (item.meta[i].midi !== undefined) {
if (item.meta[i].midi.trim() === "pitchwheel") {
obj.fPitchwheelLabel.push({
path: item.address,
min: parseFloat(item.min),
max: parseFloat(item.max),
});
} else if (item.meta[i].midi.trim().split(" ")[0] === "ctrl") {
obj.fCtrlLabel[parseInt(item.meta[i].midi.trim().split(" ")[1])].push({
path: item.address,
min: parseFloat(item.min),
max: parseFloat(item.max),
});
}
}
}
}
// Define setXXX/getXXX, replacing '/c' with 'C' everywhere in the string
var set_name = "set" + item.address;
var get_name = "get" + item.address;
set_name = set_name.replace(/\/./g, (x) => {
return x.substr(1, 1).toUpperCase();
});
get_name = get_name.replace(/\/./g, (x) => {
return x.substr(1, 1).toUpperCase();
});
obj[set_name] = (val) => {
obj.setParamValue(item.address, val);
};
obj[get_name] = () => {
return obj.getParamValue(item.address);
};
//console.log(set_name);
//console.log(get_name);
}
};
this.output_handler = null;
// input/output items
this.inputs_items = [];
this.outputs_items = [];
this.descriptor = [];
// MIDI
this.fPitchwheelLabel = [];
this.fCtrlLabel = new Array(128);
for (var i = 0; i < this.fCtrlLabel.length; i++) {
this.fCtrlLabel[i] = [];
}
// Parse UI
this.parse_ui(this.json_object.ui, this);
// Set message handler
this.port.onmessage = this.handleMessage.bind(this);
try {
if (this.parameters) this.parameters.forEach((p) => (p.automationRate = "k-rate"));
} catch (e) {}
}
// To be called by the message port with messages coming from the processor
handleMessage(event) {
var msg = event.data;
if (this.output_handler) {
this.output_handler(msg.path, msg.value);
}
}
// Public API
/**
* Destroy the node, deallocate resources.
*/
destroy() {
this.port.postMessage({ type: "destroy" });
this.port.close();
}
/**
* Returns a full JSON description of the DSP.
*/
getJSON() {
return this.json;
}
// For WAP
async getMetadata() {
return new Promise((resolve) => {
let real_url = this.baseURL === "" ? "main.json" : this.baseURL + "/main.json";
fetch(real_url)
.then((responseJSON) => {
return responseJSON.json();
})
.then((json) => {
resolve(json);
});
});
}
/**
* Set the control value at a given path.
*
* @param path - a path to the control
* @param val - the value to be set
*/
setParamValue(path, val) {
// Needed for sample accurate control
this.parameters.get(path).setValueAtTime(val, 0);
}
// For WAP
setParam(path, val) {
// Needed for sample accurate control
this.parameters.get(path).setValueAtTime(val, 0);
}
/**
* Get the control value at a given path.
*
* @return the current control value
*/
getParamValue(path) {
return this.parameters.get(path).value;
}
// For WAP
getParam(path) {
return this.parameters.get(path).value;
}
/**
* Setup a control output handler with a function of type (path, value)
* to be used on each generated output value. This handler will be called
* each audio cycle at the end of the 'compute' method.
*
* @param handler - a function of type function(path, value)
*/
setOutputParamHandler(handler) {
this.output_handler = handler;
}
/**
* Get the current output handler.
*/
getOutputParamHandler() {
return this.output_handler;
}
getNumInputs() {
return parseInt(this.json_object.inputs);
}
getNumOutputs() {
return parseInt(this.json_object.outputs);
}
// For WAP
inputChannelCount() {
return parseInt(this.json_object.inputs);
}
outputChannelCount() {
return parseInt(this.json_object.outputs);
}
/**
* Returns an array of all input paths (to be used with setParamValue/getParamValue)
*/
getParams() {
return this.inputs_items;
}
// For WAP
getDescriptor() {
var desc = {};
for (const item in this.descriptor) {
if (this.descriptor.hasOwnProperty(item)) {
if (this.descriptor[item].label != "bypass") {
desc = Object.assign(
{
[this.descriptor[item].label]: {
minValue: this.descriptor[item].min,
maxValue: this.descriptor[item].max,
defaultValue: this.descriptor[item].init,
},
},
desc
);
}
}
}
return desc;
}
/**
* Control change
*
* @param channel - the MIDI channel (0..15, not used for now)
* @param ctrl - the MIDI controller number (0..127)
* @param value - the MIDI controller value (0..127)
*/
ctrlChange(channel, ctrl, value) {
if (this.fCtrlLabel[ctrl] !== []) {
for (var i = 0; i < this.fCtrlLabel[ctrl].length; i++) {
var path = this.fCtrlLabel[ctrl][i].path;
this.setParamValue(
path,
OscTubeNode.remap(value, 0, 127, this.fCtrlLabel[ctrl][i].min, this.fCtrlLabel[ctrl][i].max)
);
if (this.output_handler) {
this.output_handler(path, this.getParamValue(path));
}
}
}
}
/**
* PitchWeel
*
* @param channel - the MIDI channel (0..15, not used for now)
* @param value - the MIDI controller value (0..16383)
*/
pitchWheel(channel, wheel) {
for (var i = 0; i < this.fPitchwheelLabel.length; i++) {
var pw = this.fPitchwheelLabel[i];
this.setParamValue(pw.path, OscTubeNode.remap(wheel, 0, 16383, pw.min, pw.max));
if (this.output_handler) {
this.output_handler(pw.path, this.getParamValue(pw.path));
}
}
}
/**
* Generic MIDI message handler.
*/
midiMessage(data) {
var cmd = data[0] >> 4;
var channel = data[0] & 0xf;
var data1 = data[1];
var data2 = data[2];
if (channel === 9) {
return;
} else if (cmd === 11) {
this.ctrlChange(channel, data1, data2);
} else if (cmd === 14) {
this.pitchWheel(channel, data2 * 128.0 + data1);
}
}
// For WAP
onMidi(data) {
midiMessage(data);
}
/**
* @returns {Object} describes the path for each available param and its current value
*/
async getState() {
var params = new Object();
for (let i = 0; i < this.getParams().length; i++) {
Object.assign(params, { [this.getParams()[i]]: `${this.getParam(this.getParams()[i])}` });
}
return new Promise((resolve) => {
resolve(params);
});
}
/**
* Sets each params with the value indicated in the state object
* @param {Object} state
*/
async setState(state) {
return new Promise((resolve) => {
for (const param in state) {
if (state.hasOwnProperty(param)) this.setParam(param, state[param]);
}
try {
this.gui.setAttribute("state", JSON.stringify(state));
} catch (error) {
console.warn("Plugin without gui or GUI not defined", error);
}
resolve(state);
});
}
/**
* A different call closer to the preset management
* @param {Object} patch to assign as a preset to the node
*/
setPatch(patch) {
this.setState(this.presets[patch]);
}
static remap(v, mn0, mx0, mn1, mx1) {
return ((1.0 * (v - mn0)) / (mx0 - mn0)) * (mx1 - mn1) + mn1;
}
}
// Factory class
export default class OscTube {
static fWorkletProcessors;
/**
* Factory constructor.
*
* @param context - the audio context
* @param baseURL - the baseURL of the plugin folder
*/
constructor(context, baseURL = "") {
console.log("baseLatency " + context.baseLatency);
console.log("outputLatency " + context.outputLatency);
console.log("sampleRate " + context.sampleRate);
this.context = context;
this.baseURL = baseURL;
this.pathTable = [];
this.fWorkletProcessors = this.fWorkletProcessors || [];
}
heap2Str(buf) {
let str = "";
let i = 0;
while (buf[i] !== 0) {
str += String.fromCharCode(buf[i++]);
}
return str;
}
/**
* Load additionnal resources to prepare the custom AudioWorkletNode. Returns a promise to be used with the created node.
*/
async load() {
try {
const importObject = {
env: {
memoryBase: 0,
tableBase: 0,
_abs: Math.abs,
// Float version
_acosf: Math.acos,
_asinf: Math.asin,
_atanf: Math.atan,
_atan2f: Math.atan2,
_ceilf: Math.ceil,
_cosf: Math.cos,
_expf: Math.exp,
_floorf: Math.floor,
_fmodf: (x, y) => x % y,
_logf: Math.log,
_log10f: Math.log10,
_max_f: Math.max,
_min_f: Math.min,
_remainderf: (x, y) => x - Math.round(x / y) * y,
_powf: Math.pow,
_roundf: Math.fround,
_sinf: Math.sin,
_sqrtf: Math.sqrt,
_tanf: Math.tan,
_acoshf: Math.acosh,
_asinhf: Math.asinh,
_atanhf: Math.atanh,
_coshf: Math.cosh,
_sinhf: Math.sinh,
_tanhf: Math.tanh,
_isnanf: Number.isNaN,
_isinff: function (x) {
return !isFinite(x);
},
_copysignf: function (x, y) {
return Math.sign(x) === Math.sign(y) ? x : -x;
},
// Double version
_acos: Math.acos,
_asin: Math.asin,
_atan: Math.atan,
_atan2: Math.atan2,
_ceil: Math.ceil,
_cos: Math.cos,
_exp: Math.exp,
_floor: Math.floor,
_fmod: (x, y) => x % y,
_log: Math.log,
_log10: Math.log10,
_max_: Math.max,
_min_: Math.min,
_remainder: (x, y) => x - Math.round(x / y) * y,
_pow: Math.pow,
_round: Math.fround,
_sin: Math.sin,
_sqrt: Math.sqrt,
_tan: Math.tan,
_acosh: Math.acosh,
_asinh: Math.asinh,
_atanh: Math.atanh,
_cosh: Math.cosh,
_sinh: Math.sinh,
_tanh: Math.tanh,
_isnan: Number.isNaN,
_isinf: function (x) {
return !isFinite(x);
},
_copysign: function (x, y) {
return Math.sign(x) === Math.sign(y) ? x : -x;
},
table: new WebAssembly.Table({ initial: 0, element: "anyfunc" }),
},
};
let real_url = this.baseURL === "" ? "OscTube.wasm" : this.baseURL + "/OscTube.wasm";
const dspFile = await fetch(real_url);
const dspBuffer = await dspFile.arrayBuffer();
const dspModule = await WebAssembly.compile(dspBuffer);
const dspInstance = await WebAssembly.instantiate(dspModule, importObject);
let HEAPU8 = new Uint8Array(dspInstance.exports.memory.buffer);
let json = this.heap2Str(HEAPU8);
let json_object = JSON.parse(json);
let options = { wasm_module: dspModule, json: json };
if (this.fWorkletProcessors.indexOf(name) === -1) {
try {
let re = /JSON_STR/g;
let OscTubeProcessorString1 = OscTubeProcessorString.replace(re, json);
let real_url = window.URL.createObjectURL(new Blob([OscTubeProcessorString1], { type: "text/javascript" }));
await this.context.audioWorklet.addModule(real_url);
// Keep the DSP name
console.log("Keep the DSP name");
this.fWorkletProcessors.push(name);
} catch (e) {
console.error(e);
console.error("Faust " + this.name + " cannot be loaded or compiled");
return null;
}
}
this.node = new OscTubeNode(this.context, this.baseURL, {
numberOfInputs: parseInt(json_object.inputs) > 0 ? 1 : 0,
numberOfOutputs: parseInt(json_object.outputs) > 0 ? 1 : 0,
channelCount: Math.max(1, parseInt(json_object.inputs)),
outputChannelCount: [parseInt(json_object.outputs)],
channelCountMode: "explicit",
channelInterpretation: "speakers",
processorOptions: options,
});
this.node.onprocessorerror = () => {
console.log("An error from OscTube-processor was detected.");
};
return this.node;
} catch (e) {
console.error(e);
console.error("Faust " + this.name + " cannot be loaded or compiled");
return null;
}
}
async loadGui() {
return new Promise((resolve, reject) => {
try {
// DO THIS ONLY ONCE. If another instance has already been added, do not add the html file again
let real_url = this.baseURL === "" ? "main.html" : this.baseURL + "/main.html";
if (!this.linkExists(real_url)) {
// LINK DOES NOT EXIST, let's add it to the document
var link = document.createElement("link");
link.rel = "import";
link.href = real_url;
document.head.appendChild(link);
link.onload = (e) => {
// the file has been loaded, instanciate GUI
// and get back the HTML elem
// HERE WE COULD REMOVE THE HARD CODED NAME
var element = createOscTubeGUI(this.node);
resolve(element);
};
} else {
// LINK EXIST, WE AT LEAST CREATED ONE INSTANCE PREVIOUSLY
// so we can create another instance
var element = createOscTubeGUI(this.node);
resolve(element);
}
} catch (e) {
console.log(e);
reject(e);
}
});
}
linkExists(url) {
return document.querySelectorAll(`link[href="${url}"]`).length > 0;
}
}
// Template string for AudioWorkletProcessor
let OscTubeProcessorString = `
'use strict';
// Monophonic Faust DSP
class OscTubeProcessor extends AudioWorkletProcessor {
// JSON parsing functions
static parse_ui(ui, obj, callback)
{
for (var i = 0; i < ui.length; i++) {
OscTubeProcessor.parse_group(ui[i], obj, callback);
}
}
static parse_group(group, obj, callback)
{
if (group.items) {
OscTubeProcessor.parse_items(group.items, obj, callback);
}
}
static parse_items(items, obj, callback)
{
for (var i = 0; i < items.length; i++) {
callback(items[i], obj, callback);
}
}
static parse_item1(item, obj, callback)
{
if (item.type === "vgroup"
|| item.type === "hgroup"
|| item.type === "tgroup") {
OscTubeProcessor.parse_items(item.items, obj, callback);
} else if (item.type === "hbargraph"
|| item.type === "vbargraph") {
// Nothing
} else if (item.type === "vslider"
|| item.type === "hslider"
|| item.type === "button"
|| item.type === "checkbox"
|| item.type === "nentry") {
obj.push({ name: item.address,
defaultValue: item.init,
minValue: item.min,
maxValue: item.max });
}
}
static parse_item2(item, obj, callback)
{
if (item.type === "vgroup"
|| item.type === "hgroup"
|| item.type === "tgroup") {
OscTubeProcessor.parse_items(item.items, obj, callback);
} else if (item.type === "hbargraph"
|| item.type === "vbargraph") {
// Keep bargraph adresses
obj.outputs_items.push(item.address);
obj.pathTable[item.address] = parseInt(item.index);
} else if (item.type === "vslider"
|| item.type === "hslider"
|| item.type === "button"
|| item.type === "checkbox"
|| item.type === "nentry") {
// Keep inputs adresses
obj.inputs_items.push(item.address);
obj.pathTable[item.address] = parseInt(item.index);
}
}
static get parameterDescriptors()
{
// Analyse JSON to generate AudioParam parameters
var params = [];
OscTubeProcessor.parse_ui(JSON.parse(\`JSON_STR\`).ui, params, OscTubeProcessor.parse_item1);
return params;
}
constructor(options)
{
super(options);
this.running = true;
const importObject = {
env: {
memoryBase: 0,
tableBase: 0,
// Integer version
_abs: Math.abs,
// Float version
_acosf: Math.acos,
_asinf: Math.asin,
_atanf: Math.atan,
_atan2f: Math.atan2,
_ceilf: Math.ceil,
_cosf: Math.cos,
_expf: Math.exp,
_floorf: Math.floor,
_fmodf: function(x, y) { return x % y; },
_logf: Math.log,
_log10f: Math.log10,
_max_f: Math.max,
_min_f: Math.min,
_remainderf: function(x, y) { return x - Math.round(x/y) * y; },
_powf: Math.pow,
_roundf: Math.fround,
_sinf: Math.sin,
_sqrtf: Math.sqrt,
_tanf: Math.tan,
_acoshf: Math.acosh,
_asinhf: Math.asinh,
_atanhf: Math.atanh,
_coshf: Math.cosh,
_sinhf: Math.sinh,
_tanhf: Math.tanh,
// Double version
_acos: Math.acos,
_asin: Math.asin,
_atan: Math.atan,
_atan2: Math.atan2,
_ceil: Math.ceil,
_cos: Math.cos,
_exp: Math.exp,
_floor: Math.floor,
_fmod: function(x, y) { return x % y; },
_log: Math.log,
_log10: Math.log10,
_max_: Math.max,
_min_: Math.min,
_remainder:function(x, y) { return x - Math.round(x/y) * y; },
_pow: Math.pow,
_round: Math.fround,
_sin: Math.sin,
_sqrt: Math.sqrt,
_tan: Math.tan,
_acosh: Math.acosh,
_asinh: Math.asinh,
_atanh: Math.atanh,
_cosh: Math.cosh,
_sinh: Math.sinh,
_tanh: Math.tanh,
table: new WebAssembly.Table({ initial: 0, element: 'anyfunc' })
}
};
this.OscTube_instance = new WebAssembly.Instance(options.processorOptions.wasm_module, importObject);
this.json_object = JSON.parse(options.processorOptions.json);
this.output_handler = function(path, value) { this.port.postMessage({ path: path, value: value }); };
this.ins = null;
this.outs = null;
this.dspInChannnels = [];
this.dspOutChannnels = [];
this.numIn = parseInt(this.json_object.inputs);
this.numOut = parseInt(this.json_object.outputs);
// Memory allocator
this.ptr_size = 4;
this.sample_size = 4;
this.integer_size = 4;
this.factory = this.OscTube_instance.exports;
this.HEAP = this.OscTube_instance.exports.memory.buffer;
this.HEAP32 = new Int32Array(this.HEAP);
this.HEAPF32 = new Float32Array(this.HEAP);
// Warning: keeps a ref on HEAP in Chrome and prevent proper GC
//console.log(this.HEAP);
//console.log(this.HEAP32);
//console.log(this.HEAPF32);
// bargraph
this.outputs_timer = 5;
this.outputs_items = [];
// input items
this.inputs_items = [];
// Start of HEAP index
// DSP is placed first with index 0. Audio buffer start at the end of DSP.
this.audio_heap_ptr = parseInt(this.json_object.size);
// Setup pointers offset
this.audio_heap_ptr_inputs = this.audio_heap_ptr;
this.audio_heap_ptr_outputs = this.audio_heap_ptr_inputs + (this.numIn * this.ptr_size);
// Setup buffer offset
this.audio_heap_inputs = this.audio_heap_ptr_outputs + (this.numOut * this.ptr_size);
this.audio_heap_outputs = this.audio_heap_inputs + (this.numIn * NUM_FRAMES * this.sample_size);
// Start of DSP memory : DSP is placed first with index 0
this.dsp = 0;
this.pathTable = [];
// Send output values to the AudioNode
this.update_outputs = function ()
{
if (this.outputs_items.length > 0 && this.output_handler && this.outputs_timer-- === 0) {
this.outputs_timer = 5;
for (var i = 0; i < this.outputs_items.length; i++) {
this.output_handler(this.outputs_items[i], this.HEAPF32[this.pathTable[this.outputs_items[i]] >> 2]);
}
}
}
this.initAux = function ()
{
var i;
if (this.numIn > 0) {
this.ins = this.audio_heap_ptr_inputs;
for (i = 0; i < this.numIn; i++) {
this.HEAP32[(this.ins >> 2) + i] = this.audio_heap_inputs + ((NUM_FRAMES * this.sample_size) * i);
}
// Prepare Ins buffer tables
var dspInChans = this.HEAP32.subarray(this.ins >> 2, (this.ins + this.numIn * this.ptr_size) >> 2);
for (i = 0; i < this.numIn; i++) {
this.dspInChannnels[i] = this.HEAPF32.subarray(dspInChans[i] >> 2, (dspInChans[i] + NUM_FRAMES * this.sample_size) >> 2);
}
}
if (this.numOut > 0) {
this.outs = this.audio_heap_ptr_outputs;
for (i = 0; i < this.numOut; i++) {
this.HEAP32[(this.outs >> 2) + i] = this.audio_heap_outputs + ((NUM_FRAMES * this.sample_size) * i);
}
// Prepare Out buffer tables
var dspOutChans = this.HEAP32.subarray(this.outs >> 2, (this.outs + this.numOut * this.ptr_size) >> 2);
for (i = 0; i < this.numOut; i++) {
this.dspOutChannnels[i] = this.HEAPF32.subarray(dspOutChans[i] >> 2, (dspOutChans[i] + NUM_FRAMES * this.sample_size) >> 2);
}
}
// Parse UI
OscTubeProcessor.parse_ui(this.json_object.ui, this, OscTubeProcessor.parse_item2);
// Init DSP
this.factory.init(this.dsp, sampleRate); // 'sampleRate' is defined in AudioWorkletGlobalScope
}
this.setParamValue = function (path, val)
{
this.HEAPF32[this.pathTable[path] >> 2] = val;
}
this.getParamValue = function (path)
{
return this.HEAPF32[this.pathTable[path] >> 2];
}
// Init resulting DSP
this.initAux();
}
process(inputs, outputs, parameters)
{
var input = inputs[0];
var output = outputs[0];
// Check inputs
if (this.numIn > 0 && (!input || !input[0] || input[0].length === 0)) {
//console.log("Process input error");
return true;
}
// Check outputs
if (this.numOut > 0 && (!output || !output[0] || output[0].length === 0)) {
//console.log("Process output error");
return true;
}
// Copy inputs
if (input !== undefined) {
for (var chan = 0; chan < Math.min(this.numIn, input.length); ++chan) {
var dspInput = this.dspInChannnels[chan];
dspInput.set(input[chan]);
}
}
/*
TODO: sample accurate control change is not yet handled
When no automation occurs, params[i][1] has a length of 1,
otherwise params[i][1] has a length of NUM_FRAMES with possible control change each sample
*/
// Update controls
for (const path in parameters) {
const paramArray = parameters[path];
this.setParamValue(path, paramArray[0]);
}
// Compute
try {
this.factory.compute(this.dsp, NUM_FRAMES, this.ins, this.outs);
} catch(e) {
console.log("ERROR in compute (" + e + ")");
}
// Update bargraph
this.update_outputs();
// Copy outputs
if (output !== undefined) {
for (var chan = 0; chan < Math.min(this.numOut, output.length); ++chan) {
var dspOutput = this.dspOutChannnels[chan];
output[chan].set(dspOutput);
}
}
return this.running;
}
handleMessage(event)
{
var msg = event.data;
switch (msg.type) {
case "destroy": this.running = false; break;
}
}
}
// Globals
const NUM_FRAMES = 128;
try {
registerProcessor('OscTube', OscTubeProcessor);
} catch (error) {
console.warn(error);
}
`;
const dspName = "OscTube";
// WAP factory or npm package module
if (typeof module === "undefined") {
window.OscTube = OscTube;
window.FaustOscTube = OscTube;
window[dspName] = OscTube;
} else {
module.exports = { OscTube };
}