forked from utz82/HoustonTracker2
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathplayer.asm
1295 lines (1088 loc) · 25.6 KB
/
player.asm
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
;*******************************************************************************
;MUSIC PLAYER
;*******************************************************************************
;setup function for rowPlay
;*******************************************************************************
rowPlay ;rowPlay subroutine
call waitForKeyRelease
ld a,(PlayerFlag) ;check if player is running
or a
jp nz,_ignore ;and ignore rowPlay if that's the case
cpl
ld (PlayerFlag),a ;set PlayerFlag to prevent double call
;call waitForKeyRelease
xor a
ld h,a
ld d,a
ld a,(OldCsrPos)
call findCurrLineNoSeq ;find current line in sequence
ld de,3 ;reading in backwards, therefore add 3 to offset
add hl,de
ex de,hl
ld bc,fxptn00
ld a,(CsrPos)
sub #50
cp #f
jr c,_skipadj
sub 8
_skipadj
add a,a
ld l,a
ld h,0
add hl,bc
ld b,h
ld c,l
ld a,(de)
add a,a ;a*2
call calcPtnOffset+2
push hl ;fx - ;stack+0 = ch1, stack+2 = ch2, stack+4 = ch3, stack+6 = fx
ld bc,ptn00
ld a,(CsrPos)
sub #50
cp #f
jr c,_skipdiv
sub 8
_skipdiv
ld l,a
ld h,0
add hl,bc
ld b,h
ld c,l
call calcPtnOffset
push hl ;ch3
call calcPtnOffset
push hl ;ch2
call calcPtnOffset
push hl ;ch1
ld (oldSP),sp
ld hl,_next
ld (rowplaySwap),hl
xor a
out (kbd),a
jp rdnotesRP ;call player
_next
pop hl ;clean stack and reset switch
pop hl
pop hl
pop hl
ld hl,rdnotes
ld (rowplaySwap),hl
xor a ;clear PlayerFlag
ld (PlayerFlag),a
_ignore
;jp waitForKeyRelease
ret
;*******************************************************************************
;main entry point for music player
;*******************************************************************************
mplay
play
call waitForKeyRelease ;keyhandler is very fast, so make sure there are currently no keys pressed
init ;sandboxing the actual player so we can handle keypresses vs looping quickly
ld de,ptns
initrp ;init point for row play
ld hl,-12 ;calculate stack restore point
add hl,sp
ld (oldSP),hl
push de
call resetFX0
ld hl,(musicData+1) ;load user (drum) sample pointer \ this will become redundant, (usrdrum) can be set on loading song
ld (usrdrum),hl ;save it to drum table / and if song is already in mem, it is then also already loaded
ld a,(looprow) ;set loop point
;add a,a ;A*4
;add a,a
ld l,a
ld h,0
add hl,hl ;loop point x4 because each seqrow is 4 bytes
add hl,hl
ld de,ptns
ld a,(de) ;while we're at it, check if player is set to start from an empty row (first pattern # = #ff)
add hl,de ;and calculate loop point
ld (looppoint),hl
pop de
inc a ;continue "empty row" check
ret z ;and exit if an "empty" row is found
call rdnotes1
init0
call rdnotes0
jp z,init0 ;loop if no key pressed
;*************************************************************************************
rdnotes0 ;initialize read song data process
looppoint equ $+1
ld de,0 ;point song data pointer at loop point
ld a,(de) ;check if loop point is still valid - exit with error if not
inc a
jp z,lpInvalidErr
rdnotes1
push de ;preserve song data pointer
xor a
out (kbd),a
jr ptnselect
;*************************************************************************************
ptnselect_playPtnLoop
pop hl ;clean stack - PROBABLY BETTER TO USE LD SP,nnnn
pop hl
pop hl
pop hl
pop de
dec de
dec de
dec de
dec de
jr ptnselect_mod
ptnselect0
pop hl ;clean stack - PROBABLY BETTER TO USE LD SP,nnnn
pop hl
pop hl
pop hl
;*************************************************************************************
ptnselect ;FIRST, set up ptn sequence pointers and push them (preserve song seq pointer in mem?)
;SECOND, check if row counter has reached 0
;THIRD, pop back ptn sequence pointers and read in data
pop de ;10 ;restore song data pointer
ptnselect_mod
ld a,#11 ;7 ;initialize ptn position counter - all patterns are 16 rows long
ld (reptpos),a ;13
ld a,(de) ;7 ;load ptn# ch1
cp #ff ;7 ;check for end marker
ret z ;11/5 ;exit if found
ld h,HIGH(ptntab) ;point to pattern position LUT
add a,a ;4 ;A=A*2
ld l,a ;4
ld c,(hl) ;7 ;seq pointer ch1 to bc
inc hl ;6 ;TODO: in theory inc l should be sufficient?
ld b,(hl) ;7
push bc ;11
inc de ;6 ;point to ch3
ld a,(de) ;7 ;load ptn# ch3
add a,a ;4
ld l,a ;4
ld c,(hl) ;7 ;seq pointer ch3 to bc
inc hl ;6
ld b,(hl) ;7
push bc ;11
inc de ;6 ;point to ch2
ld a,(de) ;7 ;load ptn# ch2
add a,a ;4
ld l,a ;4
ld c,(hl) ;7 ;seq pointer ch2 to bc
inc hl ;6
ld b,(hl) ;7
push bc ;6
inc de ;6 ;point to fx
ld a,(de) ;7 ;load ptn# fx
ld hl,fxptntab ;10
add a,a ;4
add a,l ;4
ld l,a ;4
IF MODEL != TI8X || MODEL != TI8XS ;fx ptn table crosses page boundary on 82/83 TODO: conditional should depend on the table crossing the page bound
jr nc,psskip4 ;12/7
inc h ;4
psskip4
ENDIF
ld c,(hl) ;7 ;seq pointer fx to bc
inc hl ;6
ld b,(hl) ;7 54t
inc de ;6 ;point to next row
exx ;4
pop bc ;10 ;seq pointer ch2 to bc'
pop de ;10 ;seq pointer ch3 to de'
pop hl ;10 ;seq pointer ch1 to hl'
exx ;4
push de ;11 ;stack loc. 1 - sng seq pointer
push bc ;11 ;stack loc. 2 - fx ptn pointer
exx ;4
push bc ;11 ;stack loc. 3 - ch2 ptn pointer
push de ;11 ;stack loc. 4 - ch3 ptn pointer
push hl ;11 ;stack loc. 5 - ch1 ptn pointer
;*************************************************************************************
rdnotes ;read in next step from song data
;ch1 - sp, ch2 - de, ch3 - bc, speed - de'
ld hl,reptpos ;10 ;update pattern row counter
dec (hl) ;11
ptnLoopSwitch equ $+1
jr z,ptnselect0 ;12/7
rdnotesRP ;entry point for RowPlay
ld h,HIGH(NoteTab) ;7 ;point to frequency LUT
pop bc ;10 ;ch1 ptn pointer to bc
ld a,(bc) ;7
inc bc ;6 ;increment ptn pos pointer
add a,a ;4
ld l,a ;4 ;set offset in freq.LUT
ld a,(hl) ;7 ;lookup lo byte of frequency value
ld (ch1),a ;13 ;and store it to base counter
inc l ;4
ld a,(hl) ;7
ld (ch1+1),a ;13 ;and store it to base counter
pop de ;10 ;ch2 ptn pointer to de
ld a,(de) ;7
inc de ;6 ;increment ptn pos pointer
add a,a ;4
ld l,a ;4
ld a,(hl) ;7 ;get freq.counter for ch2(!)
inc l ;4
ld h,(hl) ;7
ld l,a ;4
ld (ch3),hl ;16
exx ;4
ld h,HIGH(NoteTab) ;7 ;reset freq.LUT pointer
pop bc ;10 ;ch3 ptn pointer to bc'
ld a,(bc) ;7
inc bc ;6
add a,a ;4
ld l,a ;4
ld a,(hl) ;7 ;get freq.counter for ch3(!)
inc l ;4
ld h,(hl) ;7
ld l,a ;4
ld (ch2),hl ;16
or h ;4 ;deactivate pitch slide and table execution on rest notes, else activate
ld i,a ;9 ;(de)activate table execution
or a
jr z,_disable9xx ;disable 9xx glitch effect on rests
ld a,#57
_disable9xx
ld (ch3GlitchEnable),a
pop de ;10 ;fx ptn pointer to de
ld a,(de) ;7 ;read drum/fx cmd
and %11110000 ;7 ;check for drum trigger (lower nibble)
jp nz,drums ;10
ld a,#ee ;7 ;deactivate drum (#ee = xor n)
drx
ld (noDrum),a ;13
ld a,(de) ;7 ;read fx cmd again
and %00001111 ;7 ;
inc de ;6 ;point to fx parameter
jp nz,fxselect ;10
fxreturn
inc de ;6 ;point to next row
push de ;11 ;stack 2
push bc ;11 ;stack 3
ch3 equ $+1 ;misnomer, this is ch2
ld bc,0 ;10 ;BC' is base divider value for ch2
ld iy,0 ;14 ;reset ch2 accu
xtab equ $+2
ld ix,ptn00 ;14 ;pattern pointer for "execute note table" effect
ch2 equ $+1 ;misnomer, this is ch3
ld de,0 ;10 ;DE' holds the base counter for ch3
ch3Phase equ $+2
ld hl,0 ;10 ;reset ch3 accu
exx ;4
push de ;11 ;stack 4
push bc ;11 ;stack 5
drumtrig equ $+1
ld bc,0 ;10
cspeed equ $+2
ld de,#1000 ;10 ;speed
ch1 equ $+1
ld sp,#0000 ;10 ;load base frequency counter val
ld hl,0 ;10 ;HL holds "add" value for ch1, zero it
maskD equ $+1 ;panning switch for drum ch
ld a,lp_off ;initialize output mask for drum channel
ex af,af'
;*************************************************************************************
noXFX
playnote ;synthesize and output current step
ld a,(bc) ;7 ;load sample byte
drumswap2
nop ;4 ;daa/cpl/etc
noDrum
dru equ $+1
add a,0 ;7 ;add current drum pitch counter - swapped out with xor n (#ee) when not playing a drum (add a,n = #c6)
ld (dru),a ;13 ;and save it as current pitch counter
muteD ;mute switch for drums
jp nc,waitD ;10 ;jp nc = #d2, jp = #c3
ex af,af' ;4 ;load output mask for drum channel
out (link),a ;11 ;output drum channel state
;---- CH2: 96t
drumswap ;switch for drum mode. inc bc = #03, dec bc = #0b, inc c = #0c, dec c = #0d, nop
inc bc ;6 ;increment sample data pointer
panD equ $+1
xor lp_sw ;7 ;toggle output mask
ret c ;5 ;43t, ret never taken
ex af,af' ;4
outdr
;ex af,af' ;4
add hl,sp ;11 ;add current counter to base freq.counter val. ch1 and save result in HL
phaseshift1 equ $+1
ld a,#80 ;7 ;set duty
cp h ;4
mute1 ;mute switch for ch1
sbc a,a ;4
or lp_off ;7
pan1 equ $+1
and lp_on ;7
exx ;4 ;back to the normal register set
add hl,de ;11 ;update counter ch3
out1
out (link),a ;11 ;output state ch1
;----- DRUMS: 88/88t
jr nc,noSlideShift ;12/7
ch3grind equ $+1
rlc a ;8 ;#cb 07, swap with rlc d (#cb 02) for "grind" effect
ld a,e ;4
slideDirectionA
pitchslide equ $+1
add a,0 ;7 ;add a,n = #ce, sub n = #d6
ld e,a ;4
slideDirectionB
adc a,d ;4 ;sbc a,a ;adc a,d; sub e = #938a | sbc a,a; add a,d = #829f
sub e ;4 ;add a,d
ld d,a ;4
;34 ;+34 -25 = +9t but 18t saved because HL' is now single-use!!!
noShiftRet
phaseshift3 equ $+1 ;switch for phase shift/set duty cycle
ld a,#80 ;7
cp h ;8
mute3
sbc a,a ;4 ;result of this will be either #00 or #ff. for mute, swap #9f (sbc a,a) with #97 (sub a,a)
or lp_off ;7
pan3 equ $+1
and lp_on ;7 ;and thus we derive the output state
out3
out (link),a ;11
;---- CH1: 82t
add iy,bc ;15 ;add counters ch2
syncSwitch ;ch2 Duty Modulation FX
sbc a,a ;4 ;supply sync with main osc for duty modulation fx (sub a,a = #97; sbc a,a = #9f)
;and 0, xor/add = regular mode
;and n, xor = synced pwm mod --> 7xx: xx=0 off, xx < #7f fast pwm, xx > #7f synced pwm
;and n, add = SID/PWM --> 5xx w/ xx > #80 -> n = xxf (so 581 would be regular SID)
;add n, add = fastPWM/auto chord
;<=580: dutyModSwitch1 = [AND N, dutyModSwitch2 = XOR/ADD N,] dutyMod = 0, phaseShift = xx
;>580: dutyModSwitch1 = [AND N, dutyModSwitch2 = ADD N,] dutyMod = xxf
;=700: dutyModSwitch1 = AND N, dutyModSwitch2 = ADD N, dutyMod = 0, sync on [, phaseShift = #80]
;<780: dutyModSwitch1 = ADD N, dutyModSwitch2 = ADD N, dutyMod = xx, sync off
;>=780: dutyModSwitch1 = AND N, dutyModSwitch2 = XOR N, dutyMod = xx - #7f, sync on
;add n, xor = also valid, but for what?
dutyModSwitch1
dutyMod equ $+1
and #0 ;7 ;and n = #e6; add a,n = #c6
dutyModSwitch2
phaseshift2 equ $+1
add a,#80 ;7 ;add a,n = #c6; xor n = #ee
ld (phaseshift2),a ;13
;keycheck here wouldn't work, would need to reload A
cp iyh ;8
mute2
sbc a,a ;4
or lp_off ;7
pan2 equ $+1
and lp_on ;7
out2
exx ;4
out (link),a ;11
;---- CH3: 87t
readkeys ;check if a key has been pressed
in a,(kbd) ;11
cpl ;4 ;COULD IN THEORY OPTIMIZE THIS AWAY
or a ;4
jr nz,keyPressed ;12/7 ;and exit if necessary
reentry
timerHold
dec e ;4 ;update timer lo byte
;swap with inc a for synth mode (hold row) - a is always 0 at this point so inc a reliably returns nz
;dec e = #1d, inc a = #3c
jp nz,playnote ;10
;353
dec d ;update timer hi-byte
xFX equ $+1
jp nz,noXFX ;execute extended fx if present, and jump @playnote
oldSP equ $+1
ld sp,0 ;retrieve SP
rowplaySwap equ $+1 ;switch for jumping to exitRowplay instead
jp rdnotes
;*************************************************************************************
noSlideShift
ret c ;5 ;timing, ret never taken
ch3GlitchAdd equ $+1
ld a,#0 ;7 ;disable = 0
add a,d ;4
ch3GlitchEnable
ld d,a ;4 ;#57, temporarily disable with nop
jp noShiftRet ;10+12=34
;*************************************************************************************
noteCut ;note cut effect for ch1
nLength equ $+1
ld a,0 ;update length counter
dec a
ld (nLength),a
jr nz,noXFX ;resume playback if length counter != 0
ld hl,0 ;else, zero freq. counters
ld sp,hl
nLengthBackup equ $+1
ld a,0 ;reset length counter
ld (nLength),a
jp playnote ;resume playback
;*************************************************************************************
execTable ;execute note table effect ch3
;TODO: preserve HL'?
ld a,i ;skip table execution on rests
jr z,playnote
exx
ld (_HLrestore),hl
ld h,HIGH(NoteTab) ;point to frequency table
ld a,(ix+0) ;read note byte
inc ix ;inc pattern pointer
add a,a ;fetch new freq.counter value
ld l,a
ld e,(hl)
inc l
ld d,(hl)
_HLrestore equ $+1
ld hl,0
exx
jp playnote ;resume playback
;*************************************************************************************
keyPressed
ld (counterSP),sp ;20 ;save reload value (to be used after keyhandling)
ld sp,(oldSP)
push de ;preserve all counters
push bc ;TODO: push HL'/pop HL'
push hl
exx
push de
push hl
push bc
ex af,af'
push af
call keyhand
pop af
ex af,af'
pop bc ;retrieve all counters
pop hl
pop de
exx
pop hl
pop bc
pop de
counterSP equ $+1
ld sp,0
xor a
out (kbd),a
jp reentry ;and continue playing
;*************************************************************************************
waitD
ex af,af' ;4 ;load output mask for drum channel
out (link),a ;11 ;output drum channel state
;---- CH2: 96t
ex af,af'
; fxswap1
; fxswap2 equ $+1
; dw 0 ;8 ;swap with rlc h (cb 04) for noise/glitch effect, with rlc l (cb 05) for phase effect
; ;TODO: could be optimized, rlc a to deactivate
noiseEnable equ $+1
rlc a ;8 ;switch for enabling ch1 noise effect. rlc a (cb 07) = off, rlc h (cb 04) = on, rlc l (cb 05) = pitch inacc. (usused)
jp outdr ;10
;15+18+10=43
;*************************************************************************************
drums ;select drum
ld hl,samples-2 ;point to beginning of sample pointer table - 2 (because minimum offset will be +2)
rra ;divide drum # by 8 to get offset in table (carry is cleared before calling drum select)
rra
rra
add a,l ;add offset to (h)l
ld l,a
ld a,(hl) ;fetch drum data pointer
inc l
ld h,(hl)
ld l,a
ld (drumtrig),hl
ld a,#c6 ;#c6 = ld a,n
jp drx
;*************************************************************************************
fxselect ;select fx
dec a ;calculate jump
add a,a
add a,a
ld (_jump),a
_jump equ $+1
jr $
fxJumpTab
jp fx1
nop
jp fx2
nop
jp fx3
nop
jp fx4
nop
jp fx5
nop
jp fx6
nop
jp fx7 ;fx7
nop
jp fx8 ;fx8
nop
jp fx9 ;fx9
nop
jp fxA
nop
jp fxB
nop
jp fxC
nop
jp fxD
nop
jp fxE
nop
fxF ;#0f = set speed
ld a,(de)
ld l,a
and #3f
ld (cspeed),a
ld a,l
and #c0
ld (cspeed-1),a
jp fxcont
ptnBreak_pm_loop
ld a,(reptpos) ;break ptn (pattern loop playmode)
cp #10
jp z,fxreturn ;ignore cmd if triggered on the first pattern row
pop de
dec de
dec de
dec de
dec de
jp ptnselect_mod
ptnBreak
ld a,(reptpos) ;break ptn
cp #10 ;ignore cmd if triggered on the first pattern row
jp nz,ptnselect ;select next ptn if found ATTN: leaves register set swapped!
jp fxreturn
fxB ;#0b = break/ptn loop cmd
ld a,(de) ;if param=0
or a
fxb_pm_loop_switch equ $+1
jr z,ptnBreak ;break ptn
_ptnloop
ptnreptpos equ $+1
ld a,0
sub 1
jp z,fxcont0 ;if rept.counter is about to reach 0, all loops have been executed -> reset counter and proceed as normal
jp nc,execPtnRept ;if rept.counter is now -1, initialize ptn loop
setupPtnRept
ld a,(de) ;x = # of repeats, y = # of lines to jump back
and #f
inc a
ld h,a
ld a,(reptpos)
add a,h
cp #12
jp nc,fxcont ;invalidate effect if backjump crosses pattern boundary
ld (xreptPos),a ;TODO: Need to prevent nested Bxy loops. Maybe.
ld a,(de)
and #f0
rra
rra
rra
rra
execPtnRept
ld (ptnreptpos),a
xreptPos equ $+1
ld a,0
ld (reptpos),a
ld a,(de)
and #f
inc a ;y=0 means jump back 1 row, because pointers are already incremented
neg
exx
ld l,a ;load to HL and sign-extend
ld h,#ff
ex de,hl
add hl,de ;set ch1 pointer
ex de,hl
add hl,bc ;set ch2 pointer
ld b,h
ld c,l
exx
ld l,a
ld h,#ff
ex de,hl
add hl,de ;*2 because fx ptns take 2 bytes per row
add hl,de
ex de,hl
add hl,bc ;set ch3 pointer
ld b,h
ld c,l
jp fxcont
fx1 ;#01 = set panning
ld a,(de)
pch1
rrca
jp c,setright1
rrca
jp c,setleft1
ex af,af'
ld a,lp_on
ld (pan1),a
ex af,af'
pch2
rrca
jp c,setright2
rrca
jp c,setleft2
ex af,af'
ld a,lp_on
ld (pan2),a
ex af,af'
pch3
rrca
jp c,setright3
rrca
jp c,setleft3
ex af,af'
ld a,lp_on
ld (pan3),a
ex af,af'
pchD
rrca
jp c,setrightD
rrca
jp c,setleftD
;ex af,af'
ld a,lp_sw
ld (panD),a
ld a,lp_off
ld (maskD),a
;ex af,af'
jp fxcont
fx2 ;pitch slide up
ld a,(de)
ld (pitchslide),a
ld a,#ce
ld (slideDirectionA),a
ld hl,#938a
ld (slideDirectionB),hl
jp fxcont
fx3 ;pitch slide down
ld a,(de)
ld (pitchslide),a
ld a,#d6
ld (slideDirectionA),a
ld hl,#829f
ld (slideDirectionB),hl
jp fxcont
fx4 ;duty cycle ch1
ld a,(de)
ld (phaseshift1),a
cp #81 ;if duty < #80, deactivate noise effect
ld a,7
;jr c,Askip
jr c,_disableNoise
ld a,4 ;else, activate noise mode (formerly A01)
_disableNoise
ld (noiseEnable),a
jp fxcont
fxA
ld a,(de) ;set ch3 phase. Can be modified as follows: if bit 7 is set, interpret bits 0..6 as bitmask toggeling various fx
ld (ch3Phase),a
jp fxcont
fx5 ;duty cycle ch2
ld a,(de)
cp #81 ;if fx param > #80, switch to pwm sweep mode
jr nc,_activatePWM
ld (phaseshift2),a ;set duty
xor a
ld (dutyMod),a ;reset duty modulator
;ld a,#c6 ;deactivate pwm sweep
;ld (pwmswitch),a
jp fxcont
_activatePWM
;ld a,#ce
;ld (pwmswitch),a
and #7f
ld (dutyMod),a
jp fxcont
fx6 ;duty cycle ch3
ld a,(de)
cp #81
jr nc,_activateGrind
ld (phaseshift3),a
ld a,7
ld (ch3grind),a
jp fxcont
_activateGrind
add a,a
ld (phaseshift3),a
ld a,2
ld (ch3grind),a
jp fxcont
fx7
ld a,(de)
or a
jr nz,_activate
ld (dutyMod),a
ld a,#e6
ld (dutyModSwitch1),a
ld a,#c6
ld (dutyModSwitch2),a
ld a,#9f
ld (syncSwitch),a
jp fxcont
_activate
cp #80
jr c,_autochord
sub #7f
ld (dutyMod),a
ld a,#e6
ld (dutyModSwitch1),a
ld a,#ee
ld (dutyModSwitch2),a
ld a,#9f
ld (syncSwitch),a
jp fxcont
_autochord
ld (dutyMod),a
ld a,#c6
ld (dutyModSwitch1),a
ld (dutyModSwitch2),a
ld a,#97
ld (syncSwitch),a
jp fxcont
fx8 ;execute note table ch2
ld a,(de)
;exx
ld h,HIGH(ptntab) ;point to pattern position LUT
add a,a ;A=A*2
jr c,disableExt
ld l,a
ld a,(hl) ;lo-byte pattern pointer
inc hl
ld h,(hl) ;hi-byte pattern pointer
ld l,a
ld (xtab),hl
ld hl,execTable
ld (xFX),hl
;exx
jp fxcont
fx9 ;ch3 "glitch" effect
ld a,(de)
ld (ch3GlitchAdd),a
jp fxcont
fxC ;note cut ch1
ld a,(de)
or a
jr z,disableExt
ld (nLength),a
ld (nLengthBackup),a
;exx
ld hl,noteCut
ld (xFX),hl
;exx
jp fxcont
disableExt ;disable extended effects (8xx, Cxx)
;exx
ld hl,noXFX
ld (xFX),hl
;exx
jp fxcont
fxD ;set drum mode. TODO: instead of awkward jump, point HL to table, get offset, store
ld hl,drumModeTable1
ld a,(de)
and #0f ;calculate offset
add a,l
ld l,a
ld a,(hl)
ld (drumswap2),a
ld hl,drumModeTable2
ld a,(de)
cp #50
jr nc,_ignore
and #70
rra
rra
rra
rra
add a,l
ld l,a
ld a,(hl)
ld (drumswap),a
_ignore
jp fxcont
IF ((HIGH($))<(HIGH($+15)))
org 256*(1+(HIGH($))) ;align to next page if necessary
.WARNING drumModeTable1 crosses page
ENDIF
drumModeTable1
db #00, #27, #8f, #1f, #2f, #79, #89, #88, #90, #91, #a0, #a1, #b0, #b1, #a8, #a9
; nop daa add a,a rra cpl ld c,a add a,c add a,b sub b sub c and b or b or b or c xor b xor c
IF ((HIGH($))<(HIGH($+4)))
org 256*(1+(HIGH($))) ;align to next page if necessary
.WARNING drumModeTable2 crosses page
ENDIF
drumModeTable2
db #03, #0b, #0c, #0d, #00
; inc bc dec bc inc c dec c nop
fxE ;reset all fx
ld a,(de)
cp #80 ;if fx param >= 0x80, it's a reset fx
jr c,_extfx
and #7f
call z,resetFX ;resetFX always returns with A=0 and Z-flag set
dec a
call z,resetFX1
dec a
call z,resetFX2
call nz,resetFX3
jp fxreturn
_extfx
and #3f ;make sure ptn # is valid
push de ;preserve original fx ptn pointer