add max volume to psg envelope

This commit is contained in:
Irmen de Jong 2022-07-03 10:58:57 +02:00
parent 3ec05709d5
commit ea2812f50f
4 changed files with 51 additions and 40 deletions

View File

@ -19,6 +19,7 @@ psg {
cx16.vpoke(1, $f9c2 + voice_num * 4, channel | volume)
cx16.vpoke(1, $f9c3 + voice_num * 4, waveform | pulsewidth)
envelope_volumes[voice_num] = mkword(volume, 0)
envelope_maxvolumes[voice_num] = volume
}
; sub freq_hz(ubyte voice_num, float hertz) {
@ -37,6 +38,7 @@ psg {
uword reg = $f9c2 + voice_num * 4
cx16.vpoke(1, reg, cx16.vpeek(1, reg) & %11000000 | vol)
envelope_volumes[voice_num] = mkword(vol, 0)
envelope_maxvolumes[voice_num] = vol
}
sub pulse_width(ubyte voice_num, ubyte pw) {
@ -44,61 +46,68 @@ psg {
cx16.vpoke(1, reg, cx16.vpeek(1, reg) & %11000000 | pw)
}
sub envelope(ubyte voice_num, ubyte attack, ubyte sustain, ubyte release) {
sub envelope(ubyte voice_num, ubyte maxvolume, ubyte attack, ubyte sustain, ubyte release) {
envelope_states[voice_num] = 255
envelope_attacks[voice_num] = attack * $0040
envelope_attacks[voice_num] = attack
envelope_sustains[voice_num] = sustain
envelope_releases[voice_num] = release * $0040
envelope_releases[voice_num] = release
if attack
attack = 0
else
attack = 63 ; max volume when no attack is set
attack = maxvolume ; max volume when no attack is set
envelope_volumes[voice_num] = mkword(attack, 0)
envelope_maxvolumes[voice_num] = maxvolume
envelope_states[voice_num] = 0
}
sub silent() {
for cx16.r15L in 0 to 15 {
envelope_states[cx16.r15L] = 255
envelope_volumes[cx16.r15L] = 0
volume(cx16.r15L, 0)
for cx16.r1L in 0 to 15 {
envelope_states[cx16.r1L] = 255
envelope_volumes[cx16.r1L] = 0
volume(cx16.r1L, 0)
}
}
sub envelopes_irq() {
; If you want to use real-time volume envelopes (Attack-Sustain-Release),
; you have to call this routine every 1/60th second, for example from your vsync irq handler,
; or just install this routine as the only irq handler if you don't have to do other things there.
; Example: cx16.set_irq(&psg.envelopes_irq, true)
; cx16.r0 = the volume word (volume scaled by 256)
; cx16.r15L = the voice number
; the other virtual registers are used to backup vera registers.
; cx16.r1L = the voice number
; cx16.r2L = attack value
; calculate new volumes
for cx16.r15L in 0 to 15 {
when envelope_states[cx16.r15L] {
for cx16.r1L in 0 to 15 {
when envelope_states[cx16.r1L] {
0 -> {
; attack
cx16.r0 = envelope_volumes[cx16.r15L] + envelope_attacks[cx16.r15L]
if msb(cx16.r0) & %11000000 or envelope_attacks[cx16.r15L]==0 {
cx16.r0 = mkword(63, 0)
envelope_attacks[cx16.r15L] = 0
envelope_states[cx16.r15L] = 1 ; start sustain
cx16.r2L = envelope_maxvolumes[cx16.r1L]
cx16.r0 = envelope_volumes[cx16.r1L] + envelope_attacks[cx16.r1L] * $0040
if msb(cx16.r0) > cx16.r2L or envelope_attacks[cx16.r1L]==0 {
cx16.r0 = mkword(cx16.r2L, 0)
envelope_attacks[cx16.r1L] = 0
envelope_states[cx16.r1L] = 1 ; start sustain
}
envelope_volumes[cx16.r15L] = cx16.r0
envelope_volumes[cx16.r1L] = cx16.r0
}
1 -> {
; sustain
if envelope_sustains[cx16.r15L] {
envelope_sustains[cx16.r15L]--
if envelope_sustains[cx16.r1L] {
envelope_sustains[cx16.r1L]--
} else {
envelope_states[cx16.r15L] = 2 ; start release
envelope_states[cx16.r1L] = 2 ; start release
}
}
2 -> {
; release
cx16.r0 = envelope_volumes[cx16.r15L] - envelope_releases[cx16.r15L]
cx16.r0 = envelope_volumes[cx16.r1L] - envelope_releases[cx16.r1L] * $0040
if msb(cx16.r0) & %11000000 {
cx16.r0 = 0
envelope_releases[cx16.r15L] = 0
envelope_releases[cx16.r1L] = 0
}
envelope_volumes[cx16.r15L] = cx16.r0
envelope_volumes[cx16.r1L] = cx16.r0
}
}
}
@ -113,15 +122,16 @@ psg {
cx16.VERA_ADDR_L = $c2
cx16.VERA_ADDR_M = $f9
cx16.VERA_ADDR_H = 1 | %00110000
for cx16.r15L in 0 to 15 {
cx16.VERA_DATA0 = cx16.VERA_DATA1 & %11000000 | msb(envelope_volumes[cx16.r15L])
for cx16.r1L in 0 to 15 {
cx16.VERA_DATA0 = cx16.VERA_DATA1 & %11000000 | msb(envelope_volumes[cx16.r1L])
}
cx16.pop_vera_context()
}
ubyte[16] envelope_states
uword[16] envelope_volumes
uword[16] envelope_attacks
uword[16] envelope_volumes ; scaled by 256
ubyte[16] envelope_attacks
ubyte[16] envelope_sustains
uword[16] envelope_releases
ubyte[16] envelope_releases
ubyte[16] envelope_maxvolumes
}

View File

@ -5,6 +5,7 @@ For next release
^^^^^^^^^^^^^^^^
...
Need help with
^^^^^^^^^^^^^^
- c128 target: various machine specific things (free zp locations, how banking works, getting the floating point routines working, ...)

View File

@ -12,8 +12,8 @@ main {
psg.voice(1, psg.RIGHT, 0, psg.NOISE, 0)
psg.freq(0, 1000)
psg.freq(1, 2000)
psg.envelope(0, 50, 0, 5)
psg.envelope(1, 80, 0, 6)
psg.envelope(0, 63, 50, 0, 5)
psg.envelope(1, 63, 80, 0, 6)
sys.wait(100)
psg.silent()
cx16.restore_irq()
@ -68,8 +68,8 @@ main {
ubyte note1 = msb(note)
psg.freq(0, vera_freqs[note0])
psg.freq(1, vera_freqs[note1])
psg.envelope(0, 255, 0, 6)
psg.envelope(1, 255, 0, 6)
psg.envelope(0, 63, 255, 0, 6)
psg.envelope(1, 63, 255, 0, 6)
print_notes(note0, note1)
sys.wait(10)
}

View File

@ -602,35 +602,35 @@ sound {
; soft click/"tschk" sound
psg.freq(0, 15600)
psg.voice(0, psg.LEFT | psg.RIGHT, 32, psg.NOISE, 0)
psg.envelope(0, 200, 1, 100)
psg.envelope(0, 32, 200, 1, 100)
}
sub blockdrop() {
; swish
psg.freq(1, 4600)
psg.voice(1, psg.LEFT | psg.RIGHT, 32, psg.NOISE, 0)
psg.envelope(1, 200, 5, 20)
psg.envelope(1, 32, 200, 5, 20)
}
sub swapping() {
; beep
psg.freq(2, 1500)
psg.voice(2, psg.LEFT | psg.RIGHT, 32, psg.TRIANGLE, 0)
psg.envelope(2, 100, 6, 10)
psg.envelope(2, 40, 100, 6, 10)
}
sub lineclear() {
; explosion
psg.freq(3, 1400)
psg.voice(3, psg.LEFT | psg.RIGHT, 63, psg.NOISE, 0)
psg.envelope(3, 100, 8, 10)
psg.envelope(3, 63, 100, 8, 10)
}
sub lineclear_big() {
; big explosion
psg.freq(4, 2500)
psg.voice(4, psg.LEFT | psg.RIGHT, 63, psg.NOISE, 0)
psg.envelope(4, 100, 20, 10)
psg.envelope(4, 63, 100, 20, 10)
}
sub gameover() {
@ -639,7 +639,7 @@ sound {
psg.freq(6, 600)
psg.voice(5, psg.LEFT | psg.RIGHT, 0, psg.SAWTOOTH, 0)
psg.voice(6, psg.LEFT | psg.RIGHT, 0, psg.TRIANGLE, 0)
psg.envelope(5, 100, 30, 10)
psg.envelope(6, 100, 30, 10)
psg.envelope(5, 50, 100, 30, 10)
psg.envelope(6, 50, 100, 30, 10)
}
}