Skip to content

Commit

Permalink
variable number of samples to fill (#103)
Browse files Browse the repository at this point in the history
Add a new function fillSampleBufferWithNumSamples to exported wasm, so that it is possible to support building plugins with variable block sizes.

v0.0.42
  • Loading branch information
petersalomonsen authored Mar 30, 2024
1 parent 2eef23e commit 361521d
Show file tree
Hide file tree
Showing 7 changed files with 83 additions and 22 deletions.
2 changes: 1 addition & 1 deletion wasmaudioworklet/package.json
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
{
"name": "wasm-music",
"description": "Javascript/WebAssembly live coding environment for music and synthesis",
"version": "0.0.41",
"version": "0.0.42",
"repository": {
"url": "https://github.com/petersalomonsen/javascriptmusic"
},
Expand Down
60 changes: 57 additions & 3 deletions wasmaudioworklet/synth1/assembly/__tests__/midi/midisynth.spec.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { samplebuffer, sampleBufferFrames, playActiveVoices, cleanupInactiveVoices, shortmessage, activeVoices, MidiVoice, midichannels, MidiChannel, numActiveVoices, fillSampleBuffer, allNotesOff, getActiveVoicesStatusSnapshot } from '../../midi/midisynth';
import { freeverb, samplebuffer, sampleBufferFrames, playActiveVoices, cleanupInactiveVoices, shortmessage, activeVoices, MidiVoice, midichannels, MidiChannel, numActiveVoices, fillSampleBuffer, allNotesOff, getActiveVoicesStatusSnapshot, fillSampleBufferWithNumSamples } from '../../midi/midisynth';
import { SineOscillator } from '../../synth/sineoscillator.class';
import { Envelope, EnvelopeState } from '../../synth/envelope.class';
import { notefreq } from '../../synth/note';
Expand All @@ -7,8 +7,6 @@ import { Pan } from '../../synth/pan.class';

let signal: f32 = 0;

midichannels[1] = midichannels[0];

class TestMidiInstrument extends MidiVoice {
osc: SineOscillator = new SineOscillator();
env: Envelope = new Envelope(0.1, 0.0, 1.0, 0.1);
Expand Down Expand Up @@ -117,6 +115,18 @@ class UpperKeys extends MidiVoice {
}
}

class LinearVoice extends MidiVoice {
pos: f32 = 0.0;
nextframe(): void {
const val = this.pos / 128;
this.channel.signal.add(val, val);
this.pos++;
if (this.pos === 128.0) {
this.pos = 0;
}
}
}

describe("midisynth", () => {
it("should activate and deactivate one midivoice", () => {
const channel = midichannels[0] = new MidiChannel(1, (channel: MidiChannel) => new TestMidiInstrument(channel));
Expand Down Expand Up @@ -654,4 +664,48 @@ describe("midisynth", () => {
expect<u8>(load<u8>(activeVoicesShapshotLocation + 7)).toBe(0);
expect<u8>(load<u8>(activeVoicesShapshotLocation + 8)).toBe(0);
});
it("should be able to fill variable sized blocks in the sample buffer", () => {
expect<i32>(numActiveVoices).toBe(0, 'should be no active voices');
freeverb.set_wet(0.0);
const channel = new MidiChannel(1, (channel: MidiChannel) => new LinearVoice(channel));
midichannels[0] = channel;
channel.volume = 1.0;
channel.pan.leftLevel = 1.0;
channel.pan.rightLevel = 1.0;

fillSampleBuffer();
for (let n = 0; n < samplebuffer.length; n++) {
expect<f32>(samplebuffer[n]).toBe(0);
}

shortmessage(0x90, 69, 100);

expect<i32>(numActiveVoices).toBe(1, 'should be one active voice');

fillSampleBufferWithNumSamples(64);
for (let n = 0; n < 64; n++) {
expect<f32>(samplebuffer[n]).toBe((n / 128.0) as f32);
}

for (let n = 64; n < 128; n++) {
expect<f32>(samplebuffer[n]).toBe(0);
}

fillSampleBufferWithNumSamples(64);

for (let n = 0; n < 64; n++) {
expect<f32>(samplebuffer[n]).toBe(((n + 64) / 128.0) as f32);
}

for (let n = 64; n < 128; n++) {
expect<f32>(samplebuffer[n]).toBe(0);
}

shortmessage(0x90, 69, 0);

fillSampleBufferWithNumSamples(64);
for (let n = 0; n < samplebuffer.length; n++) {
expect<f32>(samplebuffer[n]).toBe(0, 'signal should be quiet');
}
});
});
35 changes: 20 additions & 15 deletions wasmaudioworklet/synth1/assembly/midi/midisynth.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,6 @@ export const sampleBufferBytesPerChannel = sampleBufferFrames * 4;
export const sampleBufferChannels = 2;
export const samplebuffer = new StaticArray<f32>(sampleBufferFrames * sampleBufferChannels);
const bufferposstart = changetype<usize>(samplebuffer);
const bufferposend = changetype<usize>(samplebuffer) + sampleBufferBytesPerChannel;

const CONTROL_SUSTAIN: u8 = 64;
const CONTROL_VOLUME: u8 = 7;
Expand All @@ -30,7 +29,7 @@ const CONTROL_REVERB: u8 = 91;

const mainline = new StereoSignal();
const reverbline = new StereoSignal();
const freeverb = new Freeverb();
export const freeverb = new Freeverb();
export const outputline = new StereoSignal();
export class MidiChannel {
controllerValues: StaticArray<u8> = new StaticArray<u8>(128);
Expand Down Expand Up @@ -84,7 +83,7 @@ export class MidiChannel {
if (voice.note === note) {
if (this.controllerValues[CONTROL_SUSTAIN] >= 64) {
this.sustainedVoices[note] = voice;
} else {
} else {
voice.noteoff();
}
break;
Expand Down Expand Up @@ -143,15 +142,15 @@ export class MidiChannel {
}
if (oldestVoice !== null) {
const voice = (oldestVoice as MidiVoice);
for (let n = 0;n<sampleBufferFrames;n++) {
for (let n = 0; n < sampleBufferFrames; n++) {
voice.nextframe();
const fact: f32 = ((sampleBufferFrames as f32) - (n as f32)) / (sampleBufferFrames as f32);
this.voiceTransitionBuffer[n<<1] += this.signal.left * fact;
this.voiceTransitionBuffer[(n<<1) + 1] += this.signal.right * fact;
this.voiceTransitionBuffer[n << 1] += this.signal.left * fact;
this.voiceTransitionBuffer[(n << 1) + 1] += this.signal.right * fact;
this.signal.clear();
}
voice.activationCount = voiceActivationCount++;
this.removeFromSustainedVoices(voice);
this.removeFromSustainedVoices(voice);
}
return oldestVoice;
}
Expand Down Expand Up @@ -238,10 +237,10 @@ export function shortmessage(val1: u8, val2: u8, val3: u8): void {
}

export function getActiveVoicesStatusSnapshot(): usize {
for (let n=0;n<activeVoices.length;n++) {
for (let n = 0; n < activeVoices.length; n++) {
const activeVoicesStatusSnapshotIndex = n * 3;
if (activeVoices[n] != null) {
const voice = (activeVoices[n] as MidiVoice);
if (activeVoices[n] != null) {
const voice = (activeVoices[n] as MidiVoice);
activeVoicesStatusSnapshot[activeVoicesStatusSnapshotIndex] = voice.channelNo;
activeVoicesStatusSnapshot[activeVoicesStatusSnapshotIndex + 1] = voice.note;
activeVoicesStatusSnapshot[activeVoicesStatusSnapshotIndex + 2] = voice.velocity;
Expand Down Expand Up @@ -285,6 +284,12 @@ export function playActiveVoices(): void {
}

export function fillSampleBuffer(): void {
fillSampleBufferWithNumSamples(sampleBufferFrames);
}

export function fillSampleBufferWithNumSamples(numSamples: i32): void {
const bufferposend = changetype<usize>(samplebuffer) + 4 * numSamples;

cleanupInactiveVoices();

let voiceTransitionBufferNdx = 0;
Expand All @@ -298,11 +303,11 @@ export function fillSampleBuffer(): void {

channelsignal.left += midichannel.voiceTransitionBuffer[voiceTransitionBufferNdx];
midichannel.voiceTransitionBuffer[voiceTransitionBufferNdx] = 0;
channelsignal.right +=midichannel.voiceTransitionBuffer[voiceTransitionBufferNdx+1];
midichannel.voiceTransitionBuffer[voiceTransitionBufferNdx+1] = 0;
midichannel.preprocess();
channelsignal.right += midichannel.voiceTransitionBuffer[voiceTransitionBufferNdx + 1];
midichannel.voiceTransitionBuffer[voiceTransitionBufferNdx + 1] = 0;

midichannel.preprocess();

channelsignal.left *= midichannel.pan.leftLevel * midichannel.volume;
channelsignal.right *= midichannel.pan.rightLevel * midichannel.volume;

Expand Down
2 changes: 2 additions & 0 deletions wasmaudioworklet/synth1/assembly/mixes/globalimports.ts
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,7 @@ export { sampleBufferFrames } from '../midi/midisynth';
export { sampleBufferBytesPerChannel } from '../midi/midisynth';
export { sampleBufferChannels } from '../midi/midisynth';
export { samplebuffer } from '../midi/midisynth';
export { freeverb } from '../midi/midisynth';
export { outputline } from '../midi/midisynth';
export { MidiChannel } from '../midi/midisynth';
export { MidiVoice } from '../midi/midisynth';
Expand All @@ -72,6 +73,7 @@ export { allNotesOff } from '../midi/midisynth';
export { cleanupInactiveVoices } from '../midi/midisynth';
export { playActiveVoices } from '../midi/midisynth';
export { fillSampleBuffer } from '../midi/midisynth';
export { fillSampleBufferWithNumSamples } from '../midi/midisynth';
export { Q_BUTTERWORTH } from '../synth/biquad';
export { FilterType } from '../synth/biquad';
export { Coefficients } from '../synth/biquad';
Expand Down
2 changes: 1 addition & 1 deletion wasmaudioworklet/synth1/browsercompilerwebworker.js
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ function createWebAssemblySongData(song, mode = EXPORT_MODE_MIDISYNTH_MULTIPART_
export const midipartschedule: MidiSequencerPartSchedule[] = [new MidiSequencerPartSchedule(0, 0)];
`;
assemblyscriptsynthsources[wasi_main_src] = `
export { fillSampleBuffer, samplebuffer, allNotesOff, shortmessage, getActiveVoicesStatusSnapshot } from './midi/midisynth';
export { fillSampleBuffer, fillSampleBufferWithNumSamples, samplebuffer, allNotesOff, shortmessage, getActiveVoicesStatusSnapshot } from './midi/midisynth';
export { seek, playEventsAndFillSampleBuffer, currentTimeMillis } from './midi/sequencer/midisequencer';
import { midipartschedule } from './midi/sequencer/midiparts';
Expand Down

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion wasmaudioworklet/wasmgit/wasmgitclient.spec.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import {
} from './wasmgitclient.js';

describe('wasm-git client', async function () {
this.timeout(20000);
this.timeout(40000);

it('should resolve synth and song filename from git repository', async () => {
const config = await initWASMGitClient('test');
Expand Down

0 comments on commit 361521d

Please sign in to comment.