Toolbox-2/source/boulder/bellatrix/SIDemu.spin

831 lines
36 KiB
Plaintext

' Propeller SID (MOS 6581) Emulator.
' Code released to public domain
' H. Peraza.
'
' Changes:
'
' 2009-12-07 - Started with the simple PWM example code from Propeller App Notes.
' - Simple frequency control added. 16-bit phase register used.
' - Added code to generate 3 of the SID waveforms: sawtooth, triangle
' and rectangle (square) wave. Sample uses upper 8 bits of the phase register.
' 2009-12-08 - Added a simple shift-register noise generator.
' - Changed sampling frequency to SID clk/32 = 31.25 KHz and phase register
' size to 19 bits so we can directly use SID frequency values.
' - Two more voices added, using identical code. The voices are simply added
' together to produce the output.
' - Scale waveforms according to an specified amplitude value.
' - "AND" waveforms together when more than one is selected in the control register.
' - Fixed noise generation. Now noise pitch also varies with frequency changes.
' 2009-12-09 - Added variables for all SID register.
' - Added envelope generation in a separate cog. Sounding already familiar!
' - Added a function to set a given SID register, to be used by external programs.
' - Got the BoulderDash main tune working! The transitions of one voice sounds
' somewhat "melted", though.
' - Implemented pulse-width control of the rectangle wave.
' - Added ring modulation option.
' 2009-12-10 - Changed the ring modulation method, hopefully is now closer to what SID does.
' Now harmonics and heterodynes can be heard when sweeping the frequency of
' the modulating generator.
' - Envelope generation is now done in the same main cog. The loop was split in two
' in order to have enough computing time. Waveform is now updated (sampled) at a
' half of the PWM frequency. The phase register is now 18 bits, but sample resolution
' is still the same upper 8 bits. As a side effect, digital artifacts seem less
' noticeable! Envelope is updated every 1.024 msec.
' 2009-12-11 - Track sustain changes while in sustain cycle, but only in the "down" direction.
' - Added main volume control.
' - Scale envelopes down a bit in order to avoid distortion when the three voices
' play simultaneously and the full volume is selected.
' - Had to rewrite the whole envelope generator again, in order to make it track
' the signal levels at any time and at any gate triggering moment. What a pain.
' Still not working correctly in all cases.
var
' The 29 SID registers
word voice1_Freq ' frequency
word voice1_PW ' 12-bit pulse width for the RECTANGLE waveform
byte voice1_ControlReg ' waveform type and control
byte voice1_AttackDecay ' bits 0-3 decay, 4-7 attack
byte voice1_SustainRelease ' bits 0-3 release, 4-7 sustain
word voice2_Freq ' frequency
word voice2_PW ' 12-bit pulse width for the RECTANGLE waveform
byte voice2_ControlReg ' waveform type and control
byte voice2_AttackDecay ' bits 0-3 decay, 4-7 attack
byte voice2_SustainRelease ' bits 0-3 release, 4-7 sustain
word voice3_Freq ' frequency
word voice3_PW ' 12-bit pulse width for the RECTANGLE waveform
byte voice3_ControlReg ' waveform type and control
byte voice3_AttackDecay ' bits 0-3 decay, 4-7 attack
byte voice3_SustainRelease ' bits 0-3 release, 4-7 sustain
word FC ' filter cutoff/center frequency, not implemented
byte Res_Filt ' bits 0-3 filter on/off selection, bits 4-7 filter resonance, not implemented
byte Mode_Vol ' bits 0-3 volume level, bits 4-6 filter mode, bit 7 voice 3 on/off, filter bits not implemented
byte PotX ' potentiometer X ADC value, not implemented
byte PotY ' potentiometer Y ADC value, not implemented
byte Osc3_Random ' upper 8-bits of voice 3 waveform, not implemented
byte Env3 ' upper 8-bits of voice 3 envelope generator, not implemented
con
' SID voice control register bits
CREG_GATE = %0000_0001
CREG_SYNC = %0000_0010
CREG_RINGMOD = %0000_0100
CREG_TEST = %0000_1000
CREG_TRIANGLE = %0001_0000
CREG_SAWTOOTH = %0010_0000
CREG_RECTANGLE = %0100_0000
CREG_NOISE = %1000_0000
' SID RES/FILT (reg 23)
FREG_FILT1 = %0000_0001
FREG_FILT2 = %0000_0010
FREG_FILT3 = %0000_0100
' SID MODE/VOL (reg 24)
MREG_VOICE3OFF = %1000_0000
dat
' attack, decay and release envelope timings (msecs)
AttackTimes
word 2, 8, 16, 24
word 38, 56, 68, 80
word 100, 250, 500, 800
word 1000, 3000, 5000, 8000
DecayReleaseTimes
word 6, 24, 48, 72
word 114, 168, 204, 240
word 300, 750, 1500, 2400
word 3000, 9000, 15000, 24000
var
long param[26]
word attack_time_1
word decay_time_1
word sustain_level_1
word release_time_1
word attack_time_2
word decay_time_2
word sustain_level_2
word release_time_2
word attack_time_3
word decay_time_3
word sustain_level_3
word release_time_3
pub start(pin)
param[0] := clkfreq / F_PWM
param[1] := |< pin
param[2] := (%00100 << 26) + pin 'NCO/PWM + pin
param[3] := @voice1_ControlReg
param[4] := @voice1_Freq
param[5] := @voice1_PW
param[6] := @attack_time_1
param[7] := @decay_time_1
param[8] := @sustain_level_1
param[9] := @release_time_1
param[10] := @voice2_ControlReg
param[11] := @voice2_Freq
param[12] := @voice2_PW
param[13] := @attack_time_2
param[14] := @decay_time_2
param[15] := @sustain_level_2
param[16] := @release_time_2
param[17] := @voice3_ControlReg
param[18] := @voice3_Freq
param[19] := @voice3_PW
param[20] := @attack_time_3
param[21] := @decay_time_3
param[22] := @sustain_level_3
param[23] := @release_time_3
param[24] := @RES_Filt
param[25] := @Mode_Vol
'initialize SID registers
voice1_SustainRelease := $F0
voice2_SustainRelease := $F0
voice3_SustainRelease := $F0
Mode_Vol := $00 'volume off
init_envelope1
init_envelope2
init_envelope3
cognew(@entry, @param)
pub set_register(regnum, val)
val &= $FF
case regnum
'voice1
0: voice1_Freq &= $FF00
voice1_Freq |= val
1: voice1_Freq &= $00FF
voice1_Freq |= val << 8 'shall we latch value and update freq only when writing hi-byte?
2: voice1_PW &= $FF00
voice1_PW |= val
3: voice1_PW &= $00FF
voice1_PW |= val << 8
4: voice1_ControlReg := val
5: voice1_AttackDecay := val
init_envelope1
6: voice1_SustainRelease := val
init_envelope1
'voice2
7: voice2_Freq &= $FF00
voice2_Freq |= val
8: voice2_Freq &= $00FF
voice2_Freq |= val << 8
9: voice2_PW &= $FF00
voice2_PW |= val
10: voice2_PW &= $00FF
voice2_PW |= val << 8
11: voice2_ControlReg := val
12: voice2_AttackDecay := val
init_envelope2
13: voice2_SustainRelease := val
init_envelope2
'voice3
14: voice3_Freq &= $FF00
voice3_Freq |= val
15: voice3_Freq &= $00FF
voice3_Freq |= val << 8
16: voice3_PW &= $FF00
voice3_PW |= val
17: voice3_PW &= $00FF
voice3_PW |= val << 8
18: voice3_ControlReg := val
19: voice3_AttackDecay := val
init_envelope3
20: voice3_SustainRelease := val
init_envelope3
'misc
21: FC := val
22: RES_Filt := val
23: Mode_Vol := val
con
F_PWM = 31250 ' PWM osc period is 32µS period, waveform samples are output at half of that rate
MAXLEVEL = 1000 * 16 ' = 16000 max level alloved per voice, must be divisible by 16
pri init_envelope1
sustain_level_1 := (voice1_SustainRelease >> 4) * constant(MAXLEVEL / 16)
attack_time_1 := word[@AttackTimes + (voice1_AttackDecay >> 4) * 2]
decay_time_1 := word[@DecayReleaseTimes + (voice1_AttackDecay & $0F) * 2]
release_time_1 := word[@DecayReleaseTimes + (voice1_SustainRelease & $0F) * 2]
pri init_envelope2
sustain_level_2 := (voice2_SustainRelease >> 4) * constant(MAXLEVEL / 16)
attack_time_2 := word[@AttackTimes + (voice2_AttackDecay >> 4) * 2]
decay_time_2 := word[@DecayReleaseTimes + (voice2_AttackDecay & $0F) * 2]
release_time_2 := word[@DecayReleaseTimes + (voice2_SustainRelease & $0F) * 2]
pri init_envelope3
sustain_level_3 := (voice3_SustainRelease >> 4) * constant(MAXLEVEL / 16)
attack_time_3 := word[@AttackTimes + (voice3_AttackDecay >> 4) * 2]
decay_time_3 := word[@DecayReleaseTimes + (voice3_AttackDecay & $0F) * 2]
release_time_3 := word[@DecayReleaseTimes + (voice3_SustainRelease & $0F) * 2]
dat
org
entry mov R0, par
movd :load, #period
mov R1, #26
:load rdlong 0, R0 'copy parameter block
add R0, #4
add :load, d0
djnz R1, #:load
mov dira, diraval 'set APIN to output
mov ctra, ctraval 'establish counter A mode and APIN
mov frqa, #1 'set counter to increment 1 each cycle
mov switch, #0
mov phase1, #0 'init phases
mov phase2, #0
mov phase3, #0
mov level1, #0 'reset envelope levels
mov level2, #0
mov level3, #0
mov level1init, #0
mov level2init, #0
mov level3init, #0
mov level1acc, #0
mov level2acc, #0
mov level3acc, #0
mov level1acc, max_level
mov level2acc, max_level
mov level3acc, max_level
mov t1cnt, #0
mov t2cnt, #0
mov t3cnt, #0
mov env_cycle, #%111 'all three voices start with attack on next cycle
mov cnt_msec, #1
mov time, cnt 'record current time
add time, period 'establish next period
:loop xor switch, #1 wz
if_z jmp #:L1
' Voice1
rdword next_phase, voice1freq
add next_phase, phase1 'advance phase
rdbyte R2, voice1creg 'get control register bits into R2
mov sig1, #$FF 'set initial sample result to FF, so we could "and" different waveforms together
test R2, #CREG_NOISE wz 'NOISE waveform?
if_z jmp #:L12 'jump if not
mov R0, next_phase
xor R0, phase1
test R0, Hex10000 wz
mov R0, noise1
if_z jmp #:L11
mov R0, noise
and R0, #$FF
mov noise1, R0
:L11 and sig1, R0
:L12 mov R0, phase1
shr R0, #10 'we use the upper 8 bits of the 18-bit phase value to generate the waveform
and R0, #$FF 'get the waveform sample value into R0
mov R3, R0 ' and into R3 (this is already a sawtooth)
test R2, #CREG_TRIANGLE wz 'TRIANGLE waveform?
if_z jmp #:L14
test R0, #$80 wz 'sample MSB set?
if_nz xor R0, #$FF 'invert sample if yes
shl R0, #1 'shift left and we have a triangle
test R2, #CREG_RINGMOD wz 'check for ring modulation (allowed only with triangle waveform)
if_z jmp #:L13 'skip if no modulation
mov R4, phase1 'this is as
xor R4, phase3 ' close as I can get
test R4, Hex8000 wz ' to 6581's idea
if_nz xor R0, #$FF ' of ring modulation
:L13 and sig1, R0 '"and" the waveforms
:L14 test R2, #CREG_SAWTOOTH wz 'SAWTOOTH?
if_nz and sig1, R3 'this is easy
test R2, #CREG_RECTANGLE wz 'RECTANGLE?
if_z jmp #:L15 'skip if not
rdword R0, voice1pw 'get the pulse width into R0
shr R0, #4 'use only the high 8 bits of the 12-bit value
cmp R0, R3 wc, wz 'compare with sawtooth
if_c_and_z and sig1, #0 'clear output if pw < sawtooth
:L15 mov R0, sig1 'get resulting sample into R0
mov R1, level1 'get current envelope amplitude into R1
call #Mult 'multiply with the current envelope amplitude
shr R1, #12 'shift result to get useful bits
mov sig1, R1 'store result for voice 1
mov phase1, next_phase 'store new waveform phase
' Voice2
rdword next_phase, voice2freq
add next_phase, phase2
rdbyte R2, voice2creg 'get control register bits into R2
mov sig2, #$FF 'set initial sample value
test R2, #CREG_NOISE wz 'NOISE waveform?
if_z jmp #:L22 'skip if not
mov R0, next_phase
xor R0, phase2
test R0, Hex10000 wz
mov R0, noise2
if_z jmp #:L21
mov R0, noise
and R0, #$FF
mov noise2, R0
:L21 and sig2, R0
:L22 mov R0, phase2
shr R0, #10 'get the upper 8 bits of the 18-bit phase value
and R0, #$FF ' into R0
mov R3, R0 ' and into R3
test R2, #CREG_TRIANGLE wz 'TRIANGLE waveform?
if_z jmp #:L24
test R0, #$80 wz 'MSB set?
if_nz xor R0, #$FF 'invert sample if yes
shl R0, #1 'shift left and we have a triangle
test R2, #CREG_RINGMOD wz 'check for ring modulation
if_z jmp #:L23 'skip if no modulation
mov R4, phase2
xor R4, phase1
test R4, Hex8000 wz 'emulate 6581's ring modulation
if_nz xor R0, #$FF
:L23 and sig2, R0 '"and" the waveforms together
:L24 test R2, #CREG_SAWTOOTH wz 'SAWTOOTH?
if_nz and sig2, R3 'already have it
test R2, #CREG_RECTANGLE wz 'RECTANGLE?
if_z jmp #:L25 'skip if not
rdword R0, voice2pw 'get pulse width into R0
shr R0, #4 'use upper 8 bits
cmp R0, R3 wc, wz 'compare with sawtooth
if_c_or_z and sig2, #0 'clear output if pw < sawtooth
:L25 mov R0, sig2 'get resulting sample into R0
mov R1, level2 'get voice 2 amplitude into R1
call #Mult 'multiply by envelope amplitude
shr R1, #12 'get useful bits
mov sig2, R1 'store result for voice 2
mov phase2, next_phase 'store new waveform phase
' Voice3
rdword next_phase, voice3freq
add next_phase, phase3
rdbyte R2, voice3creg 'get control bits into R2
mov sig3, #$FF 'set initial sample value
test R2, #CREG_NOISE wz 'NOISE waveform?
if_z jmp #:L32 'skip if not
mov R0, next_phase
xor R0, phase3
test R0, Hex10000 wz
mov R0, noise3
if_z jmp #:L31
mov R0, noise
and R0, #$FF
mov noise3, R0
:L31 and sig3, R0
:L32 mov R0, phase3
shr R0, #10 'get the upper 8 bits of the 16-bit phase value
and R0, #$FF ' into R0
mov R3, R0 ' and R3
test R2, #CREG_TRIANGLE wz 'TRIANGLE waveform?
if_z jmp #:L34
test R0, #$80 wz 'MSB set?
if_nz xor R0, #$FF 'invert sample if yes
shl R0, #1 'shift left and we have a triangle
test R2, #CREG_RINGMOD wz 'check for ring modulation
if_z jmp #:L33 'skip if no modulation wanted
mov R4, phase3
xor R4, phase2
test R4, Hex8000 wz 'emulate SID's ring modulation
if_nz xor R0, #$FF
:L33 and sig3, R0 'merge the waveforms together
:L34 test R2, #CREG_SAWTOOTH wz 'SAWTOOTH?
if_nz and sig3, R3 'already was in R3
test R2, #CREG_RECTANGLE wz 'RECTANGLE?
if_z jmp #:L35 'jump if not
rdword R0, voice3pw 'else get pulse width into R0
shr R0, #4 'leave only the upper 8 bits
cmp R0, R3 wc, wz 'compare with sawtooth
if_c_and_z and sig2, #0 'result is zero if pw < sawtooth
:L35 mov R0, sig3 'get resulting sample into R0
mov R1, level3 'get current level of voice 3 into R1
call #Mult 'multiply by envelope amplitude
shr R1, #12 'get useful bits
mov sig3, R1 'and store result for voice3
mov phase3, next_phase
' Noise generator
mov R0, noise
shl noise, #1
xor R0, noise
test R0, Hex4000 wz
if_nz or noise, #1
jmp #:L2
:L1 'Update envelopes
djnz cnt_msec, #:L41 'jump if not yet time to update envelopes
mov cnt_msec, #16 '1.024 msec
'Voice 1
rdbyte R0, voice1creg 'load values
rdword R1, attack1time
rdword R2, decay1time
rdword R3, sustain1level
rdword R4, release1time
test R0, #CREG_GATE wz 'if gate bit is set then process attack, decay and sustain
if_z jmp #:L42 ' else release
test gate_edge, #%001 wz 'gate just applied?
or gate_edge, #%001
if_nz jmp #:L40 'jump if yes
or env_cycle, #%001
mov t1cnt, R1 'else init attack phase, set counter to attack time
mov level1init, level1 'set initial level
mov level1inc, max_level ' and compute increment
sub level1inc, level1init
mov level1acc, #0 'reset accumulator
:L40 test env_cycle, #%001 wz 'attack?
if_z jmp #:L43 'jump to decay/sustain if not
add level1acc, level1inc 'add increment to accum
mov R0, level1acc 'divisor (attack time) already in R1
call #Divide 'obtain scaled level
and R0, HexFFFF 'get rid of remainder
add R0, level1init ' add to starting level
mov level1, R0 ' and we have the current envelope level
sub t1cnt, #1 'decrement attack time counter
cmp t1cnt, #0 wz 'end of attack phase?
if_nz jmp #:L51 'jump if yes
and env_cycle, #%110 'switch to decay/sustain if yes
mov t1cnt, R2 'set envelope time counter to decay counter
mov level1init, level1 'set initial decay level
mov level1inc, level1init ' and compute increment accordingly
sub level1inc, R3
mov level1acc, #0 'reset accumulator
jmp #:L51
:L43 cmp t1cnt, #0 wz 'end of decay phase?
if_nz jmp #:L45 'jump if not, else sustain
cmp R3, level1 wc 'current level > sustain?
if_nc jmp #:L51 'jump if not
add level1acc, level1 'else track level
jmp #:L47
:L45 sub t1cnt, #1 'else decrement decay counter
add level1acc, level1inc 'add increment to accumulator
:L47 mov R0, level1acc
mov R1, R2 'get divisor (decay time) into R1
call #Divide
and R0, HexFFFF 'get rid of remainder
mov level1, level1init
sub level1, R0 'subtract from initial level
mins level1, #0
jmp #:L51
:L42 test gate_edge, #%001 wz 'gate just released?
and gate_edge, #%110
if_z jmp #:L46
mov t1cnt, R4 'prepare for release cycle
mov level1init, level1
mov level1inc, level1
mov level1acc, #0
:L46 cmp t1cnt, #0 wz 'end of release cycle?
if_z mov level1, #0 'silence voice if yes
if_z jmp #:L51 ' and skip processing
add level1acc, level1inc
mov R0, level1acc
mov R1, R4
call #Divide
and R0, HexFFFF
mov level1, level1init
sub level1, R0
mins level1, #0 'don't go below zero
sub t1cnt, #1 'decrement release counter
:L51 'Voice 2
rdbyte R0, voice2creg 'load values
rdword R1, attack2time
rdword R2, decay2time
rdword R3, sustain2level
rdword R4, release2time
test R0, #CREG_GATE wz 'if gate bit is set then process attack, decay and sustain
if_z jmp #:L52 ' else release
test gate_edge, #%010 wz 'gate just applied?
or gate_edge, #%010
if_nz jmp #:L50 'jump if yes
or env_cycle, #%010
mov t2cnt, R1 'else init attack phase, set counter to attack time
mov level2init, level2 'set initial level
mov level2inc, max_level ' and compute increment accordingly
sub level2inc, level2init
mov level2acc, #0 'reset accumulator
:L50 test env_cycle, #%010 wz 'attack?
if_z jmp #:L53 'jump to decay/sustain if not
mov R6, R1 'save decay end time in R6
add level2acc, level2inc 'add increment to accum
mov R0, level2acc 'divisor (attack time) already in R1
call #Divide 'obtain scaled level
and R0, HexFFFF 'get rid of remainder
add R0, level2init ' add to starting level
mov level2, R0 ' and we have the current envelope level
sub t2cnt, #1 'decrement attack time counter
cmp t2cnt, #0 wz 'end of attack phase?
if_nz jmp #:L61
and env_cycle, #%101 'switch to decay/sustain if yes
mov t2cnt, R2 'set envelope time counter to decay
mov level2init, level2 'set initial decay level
mov level2inc, level2init ' and compute increment accordingly
sub level2inc, R3
mov level2acc, #0 'reset accumulator
jmp #:L61
:L53 cmp t2cnt, #0 wz 'end of decay phase?
if_nz jmp #:L55 'jump if not, else sustain
cmp R3, level2 wc 'current level > sustain?
if_nc jmp #:L61 'jump if not
add level2acc, level2 'else track level
jmp #:L57
:L55 sub t2cnt, #1 'else decrement decay counter
add level2acc, level2inc 'add increment to accumulator
:L57 mov R0, level2acc
mov R1, R2 'get divisor (decay time) into R1
call #Divide
and R0, HexFFFF 'get rid of remainder
mov level2, level2init
sub level2, R0 'subtract from initial level
mins level2, #0
jmp #:L61
:L52 test gate_edge, #%010 wz 'gate just released?
and gate_edge, #%101
if_z jmp #:L56
mov t2cnt, R4 'prepare for release cycle
mov level2init, level2
mov level2inc, level2
mov level2acc, #0
:L56 cmp t2cnt, #0 wz 'end of release cycle?
if_z mov level2, #0 'silence voice if yes
if_z jmp #:L61 ' and skip processing
add level2acc, level2inc
mov R0, level2acc
mov R1, R4
call #Divide
and R0, HexFFFF
mov level2, level2init
sub level2, R0
mins level2, #0 'don't go below zero
sub t2cnt, #1 'decrease release counter
:L61 'Voice 3
rdbyte R0, voice3creg 'load values
rdword R1, attack3time
rdword R2, decay3time
rdword R3, sustain3level
rdword R4, release3time
test R0, #CREG_GATE wz 'if gate bit is set then process attack, decay and sustain
if_z jmp #:L62 ' else release
test gate_edge, #%100 wz 'gate just applied?
or gate_edge, #%100
if_nz jmp #:L60 'jump if yes
or env_cycle, #%100
mov t3cnt, R1 'else init attack phase, set envelope time counter to attack
mov level3init, level3 'set initial level
mov level3inc, max_level ' and compute increment
sub level3inc, level3init ' accordingly
mov level3acc, #0 'reset accumulator
:L60 test env_cycle, #%100 wz 'attack?
if_z jmp #:L63 'jump to decay/sustain if not
mov R6, R1 'save decay end time in R6
add level3acc, level3inc 'add increment to accum
mov R0, level3acc 'divisor (attack time) already in R1
call #Divide 'obtain scaled level
and R0, HexFFFF 'get rid of remainder
add R0, level3init ' add to starting level
mov level3, R0 ' and we have the current envelope level
sub t3cnt, #1 'decrement attack time counter
cmp t3cnt, #0 wz 'end of attack phase?
if_nz jmp #:L41
and env_cycle, #%011 'switch to decay/sustain if yes
mov t3cnt, R2 'reset envelope time counter, now decay counter
mov level3init, level3 'set initial decay level
mov level3inc, level3init ' and compute increment accordingly
sub level3inc, R3
mov level3acc, #0 'reset accumulator
jmp #:L41
:L63 cmp t3cnt, #0 wz 'end of decay phase?
if_nz jmp #:L65 'jump if not, else sustain
cmp R3, level3 wc 'current level > sustain?
if_nc jmp #:L41 'jump if not
add level3acc, level3 'else track level
jmp #:L67
:L65 sub t3cnt, #1 'else decrement decay counter
add level3acc, level3inc 'add increment to accumulator
:L67 mov R0, level3acc
mov R1, R2 'get divisor (decay time) into R1
call #Divide
and R0, HexFFFF 'get rid of remainder
mov level3, level3init
sub level3, R0 'subtract from initial level
mins level3, #0
jmp #:L41
:L62 test gate_edge, #%100 wz 'gate just released?
and gate_edge, #%011
if_z jmp #:L66
mov t3cnt, R4 'prepare for release cycle
mov level3init, level3
mov level3inc, level3
mov level3acc, #0
:L66 cmp t3cnt, #0 wz 'end of release cycle?
if_z mov level3, #0 'silence voice if yes
if_z jmp #:L41 ' and skip processing
add level3acc, level3inc
mov R0, level3acc
mov R1, R4
call #Divide
and R0, HexFFFF
mov level3, level3init
sub level3, R0
mins level3, #0 'don't go below zero
sub t3cnt, #1 'decrement release counter
:L41 ' Mixer (simple adder)
rdbyte R1, modevol 'get mode bits into R3
mov R0, sig1
add R0, sig2
test R1, #MREG_VOICE3OFF wz 'check if voice 3 is disabled
if_z add R0, sig3
and R1, #$0F 'get main volume bits
call #Mult 'multiply by signal level
shr R1, #4 'scale result
mov value, R1
:L2 waitcnt time, period 'wait until next period
neg phsa, value
jmp #:loop 'loop for next cycle
'────────────────────────────────────────────────────────────────────────────────────────
' Multiply R0[15..0] by R1[15..0] (R1[31..16] must be 0)
' On exit, product is in R1[31..0]
Mult shl R0, #16 'get multiplicand into R0[31..16]
mov R5, #16 'ready for 16 multiplier bits
shr R1, #1 wc 'get initial multiplier bit into c
:m1 if_c add R1, R0 wc 'if c set, add multiplicand to product
rcr R1, #1 wc 'put next multiplier in c, shift prod.
djnz R5, #:m1 'loop until done
Mult_ret ret
' Divide R0[31..0] by R1[15..0] (R1[16] must be 0)
' On exit, quotient is in R0[15..0] and remainder is in R0[31..16]
Divide shl R1, #15 'get divisor into R1[30..15]
mov R5, #16 'ready for 16 quotient bits
:d1 cmpsub R0, R1 wc 'R1 =< R0? Subtract it, quotient bit in c
rcl R0, #1 'rotate c into quotient, shift dividend
djnz R5, #:d1 'loop until done
divide_ret ret 'quotient in R0[15..0], remainder in R0[31..16]
'────────────────────────────────────────────────────────────────────────────────────────
d0 long 1 << 9 << 0
Hex4000 long $4000
Hex8000 long $8000
HexFFFF long $FFFF
Hex10000 long $10000
max_level long MAXLEVEL
noise long $6DE7
cnt_msec res 1
period res 1
diraval res 1 '|< pin
ctraval res 1 '%00100 << 26 + pin (NCO/PWM + pin)
voice1creg res 1
voice1freq res 1
voice1pw res 1
attack1time res 1
decay1time res 1
sustain1level res 1
release1time res 1
voice2creg res 1
voice2freq res 1
voice2pw res 1
attack2time res 1
decay2time res 1
sustain2level res 1
release2time res 1
voice3creg res 1
voice3freq res 1
voice3pw res 1
attack3time res 1
decay3time res 1
sustain3level res 1
release3time res 1
resfilt res 1
modevol res 1
time res 1
value res 1
switch res 1
next_phase res 1
phase1 res 1
phase2 res 1
phase3 res 1
level1 res 1
level2 res 1
level3 res 1
level1acc res 1
level2acc res 1
level3acc res 1
level1inc res 1
level2inc res 1
level3inc res 1
level1init res 1
level2init res 1
level3init res 1
sig1 res 1
sig2 res 1
sig3 res 1
noise1 res 1
noise2 res 1
noise3 res 1
t1cnt res 1
t2cnt res 1
t3cnt res 1
env_cycle res 1 'envelope cycle: bits 0..2 set if voice 1..3 starts attack on next gate
gate_edge res 1 'gate edge detection: bits 0..2 if up transition detected for voice 1..3
R0 res 1
R1 res 1
R2 res 1
R3 res 1
R4 res 1
R5 res 1
R6 res 1
fit