ADC, November 2020
Benedict R. Gaster /
@cuberoo_
my pronouns are he/him/they or xe/xim/xir
[] == ![];
NaN === NaN; // false
x = 100;
while(true) {
if (x == 999) {
x = "OMG";
break;
}
x = x + 1;
}
log.console(x);
What if you could compile code for the Browser?
" a source-to-source compiler that runs as a back end to the LLVM compiler and produces a subset of JavaScript known as asm.js. This allows applications and libraries originally designed to run as standard executables to be integrated into client side web applications"
"WebAssembly (abbreviated Wasm) is a binary instruction format for a stack-based virtual machine. Wasm is designed as a portable compilation target for programming languages, enabling deployment on the web for client and server applications."
W3C Working Group
(module
(import "math" "callback" (func $callback))
(export "add" (func $add))
(export "subtract" (func $substract))
(func $add (param $a i32) (param $b i32) (result i32)
local.get $a
local.get $b
i32.add
)
(func $subtract (param $a i32) (param $b i32) (result i32)
local.get $a
local.get $b
i32.sub
)
)
"WASI is a modular system interface for WebAssembly. As described in the initial announcement, it’s focused on security and portability"
W3C Working Group
Alon Zakai, 2020
Tiffany Aching
{
"wasm": [String],
"gui": String,
"info": {
...
"inputs": int,
"outputs": int,
...
}
}
Single hosting app for each supported platform
// initialize module
#[no_mangle]
pub fn init(sample_rate: f64) -> u64 // returns handle to module
// module meta data
#[no_mangle]
pub fn get_sample_rate(module: u64) -> f64
#[no_mangle]
pub fn get_inputs(module: u64) -> u32
#[no_mangle]
pub fn get_outputs(module: u64) -> u32
#[no_mangle]
pub fn get_voices(module: u64) -> i32
// parameters
#[no_mangle]
pub fn get_param_index(module: u64, length: i32) -> i32
#[no_mangle]
pub fn get_num_params_float(module: u64) -> u32
#[no_mangle]
pub fn set_param_float(module: u64, index: u32, v: f32)
#[no_mangle]
pub fn get_param_float(module: u64, index: u32) -> f32
// MIDI
#[no_mangle]
pub fn handle_note_on(module: u64, mn: i32, vel: f32)
#[no_mangle]
pub fn handle_note_off(module: u64, mn: i32, vel: f32)
// compute audio
#[no_mangle]
pub fn compute(module: u64, frames: u32)
// input and output buffer management
#[no_mangle]
pub fn get_input(module: u64, index: u32) -> u32
#[no_mangle]
pub fn get_output(module: u64, index: u32) -> u32
#[no_mangle]
pub fn set_input(module: u64, index: u32, offset: u32)
#[no_mangle]
pub fn set_output(module: u64, index: u32, offset: u32)
// extern functions provided by AA Host
extern "C" pub fn set(module: u64, index: u32, v: f32)
FAUST (Functional AUdio STream) is a domain-specific purely functional programming language for implementing signal processing algorithms in the form of libraries, audio plug-ins, or standalone applications.
declare aavoices "2";
import("stdfaust.lib");
process = vgroup("voices", par(n, 2, vgroup("aavoice%n", voice))) ;
voice = hgroup("midi", osc(freq))
with {
freq = hslider("freq",200,50,1000,0.01);
gain = hslider("gain",0.5,0,1,0.01); gate = button("gate");
envelope = en.adsr(0.01,0.01,0.8,0.1,gate)*gain;
osc(freq) = os.sawtooth(freq)*envelope;
};
faust2audioanywhere synth.dsp
Fork of Faust compiler for AA on my Github
struct mydsp {
...
}
impl mydsp {
pub fn get_voices(&self) -> i32 { ... }
pub fn get_input(&self, index: u32) -> u32 { ... }
pub fn get_output(&self, index: u32) -> u32 { ... }
pub fn get_num_outputs(&self) -> i32 { ... }
pub fn handle_note_on(&mut self, mn: Note, vel: f32) { ... }
unsafe fn compute(
&mut self, count: i32, inputs: &[T], outputs: &mut [&mut [T];2]) { ... }
pub fn compute_external(&mut self, count: i32) { ... }
}
pub fn compute_external(&mut self, count: i32) {
unsafe {
let (output0, output1) =
(::std::slice::from_raw_parts_mut(OUTPUTS[0], count as usize),
::std::slice::from_raw_parts_mut(OUTPUTS[1], count as usize));
self.compute(count, &[], &mut [output0, output1]);
}
}
#[target_feature(enable = "simd128")]
unsafe fn compute(
&mut self, count: i32, inputs: &[T], outputs: &mut [&mut [T];2]) {
let [outputs0, outputs1] = outputs;
let (outputs0, outputs1) = {
let outputs0 = outputs0[..count as usize].iter_mut();
let outputs1 = outputs1[..count as usize].iter_mut();
(outputs0, outputs1)
};
...
let zipped_iterators = outputs0.zip(outputs1);
for (output0, output1) in zipped_iterators {
...
*output0 = ... ...
*output1 = ... ...
}
}
C++ | Rust | Rust Optimized |
---|---|---|
84860.747 MB/sec | 3379.915 MB/sec | 88353.927 MB/sec |