mirror of
https://github.com/exoticorn/microw8.git
synced 2026-01-20 19:26:43 +01:00
123 lines
4.0 KiB
Plaintext
123 lines
4.0 KiB
Plaintext
include "../include/microw8-api.cwa"
|
|
|
|
export fn upd() {
|
|
let T = 32!32 / 116;
|
|
|
|
let inline riff_pos = abs(((T&31) - 16) as f32) as i32;
|
|
let lazy shift = ((1-((T>>5)&3))%2-1) * 2;
|
|
|
|
let inline note_count = 5 - (T >= 512);
|
|
let inline octave = (riff_pos/5) * 12;
|
|
let inline riff_note = 5514 >> (riff_pos % note_count * 4) & 15;
|
|
let inline melody_note = shift + octave - riff_note;
|
|
|
|
80?0 = ((T&1) << 1); // note trigger
|
|
80?4 = 0xa; // attack, decay
|
|
80?5 = 0xa; // sustain, release
|
|
80?3 = melody_note + 76;
|
|
}
|
|
|
|
export fn snd(t: i32) -> f32 {
|
|
gesSample(t)
|
|
}
|
|
|
|
const GesChannelState.Trigger = 0;
|
|
const GesChannelState.EnvState = 1;
|
|
const GesChannelState.EnvVol = 2;
|
|
const GesChannelState.Phase = 4;
|
|
const GesChannelState.Size = 6;
|
|
|
|
const GesState.Size = GesChannelState.Size * 4;
|
|
|
|
const GesStateOffset = 112;
|
|
const GesBufferOffset = 112 + GesState.Size;
|
|
|
|
fn gesSample(t: i32) -> f32 {
|
|
if !(t & 127) {
|
|
let i: i32;
|
|
loop clearLoop {
|
|
i!GesBufferOffset = 0;
|
|
branch_if (i := i + 4) < 128*8: clearLoop;
|
|
}
|
|
|
|
let ch: i32;
|
|
loop channelLoop {
|
|
let channelState = GesStateOffset + ch * GesChannelState.Size;
|
|
let channelReg = 80 + ch * 6;
|
|
let envState = channelState?GesChannelState.EnvState;
|
|
let envVol = i32.load16_u(channelState, GesChannelState.EnvVol);
|
|
|
|
let oldTrigger = channelState?GesChannelState.Trigger;
|
|
let ctrl = channelReg?0;
|
|
channelState?GesChannelState.Trigger = ctrl;
|
|
if (((oldTrigger ^ ctrl) >> 1) & 1) | (ctrl & !(oldTrigger & 1)) {
|
|
envState = 0;
|
|
envVol = 0;
|
|
}
|
|
|
|
if envState == 0 {
|
|
envVol = envVol + (16384 / ((channelReg?4 >> 4) + 1));
|
|
if envVol >= 65535 {
|
|
envVol = 65535;
|
|
envState = 1;
|
|
}
|
|
} else {
|
|
if envState == 1 & ctrl {
|
|
envVol = envVol - (16 - (channelReg?4 & 15)) * 48;
|
|
let sustain = (channelReg?5 >> 4) * 4096;
|
|
if envVol < sustain {
|
|
envVol = sustain;
|
|
}
|
|
} else {
|
|
envVol = envVol - (16 - (channelReg?5 & 15)) * 48;
|
|
if envVol < 0 {
|
|
envVol = 0;
|
|
}
|
|
}
|
|
}
|
|
channelState?GesChannelState.EnvState = envState;
|
|
|
|
i32.store16(envVol, channelState, GesChannelState.EnvVol);
|
|
|
|
let inline note = channelReg?3;
|
|
let inline freq = 440 as f32 * pow(2.0, (note - 69) as f32 / 12 as f32);
|
|
let phaseInc = (freq * (65536.0 / 88200.0)) as i32;
|
|
|
|
let phase = i32.load16_u(channelState, GesChannelState.Phase);
|
|
|
|
i = 0;
|
|
let wave = ctrl >> 6;
|
|
if wave < 2 {
|
|
if wave == 0 {
|
|
loop rectLoop {
|
|
i!(GesBufferOffset + 128*4) = select(phase & 32768, -32768, 32767);
|
|
phase = phase + phaseInc;
|
|
branch_if (i := i + 4) < 64*4: rectLoop;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
loop sawLoop {
|
|
i!(GesBufferOffset + 128*4) = (phase & 65535) - 32768;
|
|
phase = phase + phaseInc;
|
|
branch_if (i := i + 4) < 64*4: sawLoop;
|
|
}
|
|
}
|
|
}
|
|
|
|
i32.store16(phase, channelState, GesChannelState.Phase);
|
|
|
|
i = 0;
|
|
loop mixLoop {
|
|
let sample = (i!(GesBufferOffset + 128*4) * envVol) >> 18;
|
|
(i * 2)!GesBufferOffset = (i * 2)!GesBufferOffset + sample;
|
|
(i * 2)!(GesBufferOffset + 4) = (i * 2)!(GesBufferOffset + 4) + sample;
|
|
branch_if (i := i + 4) < 64*4: mixLoop;
|
|
}
|
|
|
|
branch_if (ch := ch + 1) < 4: channelLoop;
|
|
}
|
|
}
|
|
(((t & 127) * 4)!GesBufferOffset) as f32 / (32768 * 10) as f32
|
|
}
|