From 59971b085c57352ac659a77b2c841f30def84f27 Mon Sep 17 00:00:00 2001 From: Vince Weaver Date: Tue, 18 Jan 2022 23:53:24 -0500 Subject: [PATCH] tiny_tracker: crazy hacks for 256 --- demos/l/music_256/Makefile | 12 +- demos/l/music_256/apple2_80_zp.inc | 12 + demos/l/music_256/d2.s | 55 +- demos/l/music_256/play_frame.s | 22 +- demos/l/music_256/zp.inc | 18 +- music/tiny_tracker5/Makefile | 64 +++ music/tiny_tracker5/ay3_write_regs.s | 34 ++ music/tiny_tracker5/d2.s | 57 ++ music/tiny_tracker5/hardware.inc | 92 ++++ music/tiny_tracker5/hello.bas | 6 + music/tiny_tracker5/mA2E_2.txt | 212 ++++++++ .../tiny_tracker5}/mA2E_3.txt | 0 music/tiny_tracker5/mockingboard_init.s | 77 +++ music/tiny_tracker5/play_frame.s | 79 +++ music/tiny_tracker5/text_to_tiny.c | 511 ++++++++++++++++++ .../tiny_tracker5}/tracker_init.s | 0 music/tiny_tracker5/zp.inc | 32 ++ 17 files changed, 1245 insertions(+), 38 deletions(-) create mode 100644 demos/l/music_256/apple2_80_zp.inc create mode 100644 music/tiny_tracker5/Makefile create mode 100644 music/tiny_tracker5/ay3_write_regs.s create mode 100644 music/tiny_tracker5/d2.s create mode 100644 music/tiny_tracker5/hardware.inc create mode 100644 music/tiny_tracker5/hello.bas create mode 100644 music/tiny_tracker5/mA2E_2.txt rename {demos/l/music_256 => music/tiny_tracker5}/mA2E_3.txt (100%) create mode 100644 music/tiny_tracker5/mockingboard_init.s create mode 100644 music/tiny_tracker5/play_frame.s create mode 100644 music/tiny_tracker5/text_to_tiny.c rename {demos/l/music_256 => music/tiny_tracker5}/tracker_init.s (100%) create mode 100644 music/tiny_tracker5/zp.inc diff --git a/demos/l/music_256/Makefile b/demos/l/music_256/Makefile index 80698929..8b94c027 100644 --- a/demos/l/music_256/Makefile +++ b/demos/l/music_256/Makefile @@ -15,7 +15,7 @@ submit: hgr_d2_1k.zip hgr_d2.dsk: HELLO D2 cp $(EMPTY_DISK)/empty.dsk ./hgr_d2.dsk $(DOS33) -y hgr_d2.dsk SAVE A HELLO - $(DOS33) -y hgr_d2.dsk -t BIN -a 0x6000 BSAVE D2 + $(DOS33) -y hgr_d2.dsk -t BIN -a 0x80 BSAVE D2 #### @@ -32,22 +32,18 @@ peasant_music.s: peasant.txt text_to_tiny mA2E_2.s: mA2E_2.txt text_to_tiny ./text_to_tiny mA2E_2.txt > mA2E_2.s -#### - -mA2E_3.s: mA2E_3.txt text_to_tiny - ./text_to_tiny mA2E_3.txt > mA2E_3.s #### D2: d2.o - ld65 -o D2 d2.o -C $(LINKER_SCRIPTS)/apple2_6000.inc + ld65 -o D2 d2.o -C ./apple2_80_zp.inc d2.o: d2.s \ zp.inc hardware.inc \ - mA2E_2.s mA2E_3.s \ + mA2E_2.s \ mockingboard_init.s play_frame.s \ - tracker_init.s ay3_write_regs.s + ay3_write_regs.s ca65 -o d2.o d2.s -l d2.lst #### diff --git a/demos/l/music_256/apple2_80_zp.inc b/demos/l/music_256/apple2_80_zp.inc new file mode 100644 index 00000000..a911b240 --- /dev/null +++ b/demos/l/music_256/apple2_80_zp.inc @@ -0,0 +1,12 @@ +MEMORY { + ZP: start = $80, size = $120, type = rw; + RAM: start = $100, size = $8E00, file = %O; +} + +SEGMENTS { +#CODE: load = RAM, type = ro; +#RODATA: load = RAM, type = ro; +#DATA: load = RAM, type = rw; +#BSS: load = RAM, type = bss, define = yes; +ZEROPAGE: load = ZP, type = ro; +} diff --git a/demos/l/music_256/d2.s b/demos/l/music_256/d2.s index abb894ea..8d4b6b2e 100644 --- a/demos/l/music_256/d2.s +++ b/demos/l/music_256/d2.s @@ -20,7 +20,38 @@ ; 245 bytes -- try to optimize writing out volume ; 255 bytes -- add in some visualization ; 252 bytes -- re-arrange decode code +; 251 bytes -- load in zero page +; 256 bytes -- expand WAIT to not use jsr +; 258 bytes -- get rid of pha/pla +; 255 bytes -- use PLA to load the data after setting stack to 0 +; 249 bytes -- optimize init code to not write 0s to ZP and init $38/$e/$e/$e +; but instead count on A/B/C over-writing and have the +; ay buffer over-write the init code +; 247 bytes -- count on X always being $FF when hit delay +; 246 bytes -- make SONG_COUNTDOWN self-modify code + +.zeropage +;.globalzp rot_smc + d2: + ; this is also the start of AY_REGS + ; we count on A/B/C being played first note + ; so the code gets over-written + + ; depends on AY ignoring envelope values if unused + + jsr SETGR ; enable lo-res graphics + ; A=$D0, Z=1 + + ldx #$FF ; set stack offset + bmi skip_const + + .byte $38,$e,$e,$e ; mixer, A, B, C volume + +skip_const: + + txs ; write 0 to stack pointer + ;=================== ; music Player Setup @@ -33,9 +64,7 @@ tracker_song = peasant_song .include "mockingboard_init.s" -.include "tracker_init.s" - jsr SETGR ; enable lo-res graphics game_loop: ; typically A=0, X=FF, Y=0 here @@ -45,13 +74,29 @@ game_loop: .include "play_frame.s" .include "ay3_write_regs.s" - ; delay 20Hz, or 1/20s = 50ms + ; X is in theory $ff when we get here - lda #140 - jsr WAIT + ; delay 20Hz, or 1/20s = 50ms + ; 50,000 cycles + + ; 2 + 256*(2+39*5-1) = 49,922 + +; ldx #0 ; 2 +outer_wait: + ldy #39 ; 2 +inner_wait: + dey ; 2 + bne inner_wait ; 3/2 + + dex ; 2 + bne outer_wait ; 3/2 beq game_loop + +; pad so starts at $80 +.byte $00,$00,$00,$00 + ; music .include "mA2E_2.s" diff --git a/demos/l/music_256/play_frame.s b/demos/l/music_256/play_frame.s index 9be4ecf9..fd1fed5f 100644 --- a/demos/l/music_256/play_frame.s +++ b/demos/l/music_256/play_frame.s @@ -3,7 +3,8 @@ play_frame: ;============================ ; see if still counting down - lda SONG_COUNTDOWN +song_countdown_smc: + lda #$FF ; initially negative so we enter loop bpl done_update_song set_notes_loop: @@ -11,31 +12,31 @@ set_notes_loop: ;================== ; load next byte - ldy SONG_OFFSET - lda tracker_song,Y + pla ; located on stack ;================== ; see if hit end ; this song only 16 notes so valid notes always positive -; cmp #$80 bpl not_end ;==================================== ; if at end, loop back to beginning asl ; reset song offset to 0 - sta SONG_OFFSET + tax + txs beq set_notes_loop not_end: ; NNNNNECC -- c=channel, e=end, n=note - pha ; save note + sta SAVE ; save note and #3 tax + ldy #$0E sty AY_REGS+8,X ; $08 set volume A,B,C @@ -43,10 +44,10 @@ not_end: tax ; put channel offset in X - pla ; restore note + lda SAVE ; restore note tay and #$4 - sta SONG_COUNTDOWN ; always 4 long? + sta song_countdown_smc+1 ; always 4 long? tya lsr @@ -71,9 +72,8 @@ blah_urgh: ;============================ ; point to next - ; assume less than 256 bytes - inc SONG_OFFSET + ; don't have to, PLA did it for us done_update_song: - dec SONG_COUNTDOWN + dec song_countdown_smc+1 bmi set_notes_loop diff --git a/demos/l/music_256/zp.inc b/demos/l/music_256/zp.inc index 598dfecb..86c412d5 100644 --- a/demos/l/music_256/zp.inc +++ b/demos/l/music_256/zp.inc @@ -9,24 +9,14 @@ GBASH = $27 BASL = $28 BASH = $29 -AY_REGS = $70 +AY_REGS = $80 +SAVE = $60 -SONG_L = $80 -SONG_H = $81 -SONG_OFFSET = $82 -SONG_COUNTDOWN = $83 -OCTAVE = $84 -REGISTER = $85 - -OUR_ROT = $A5 - -HGR_X = $E0 -HGR_XH = $E1 HGR_Y = $E2 HGR_COLOR = $E4 HGR_PAGE = $E6 HGR_SCALE = $E7 -COUNT = $FB -FRAME = $FC +;COUNT = $FB +;FRAME = $FC diff --git a/music/tiny_tracker5/Makefile b/music/tiny_tracker5/Makefile new file mode 100644 index 00000000..80698929 --- /dev/null +++ b/music/tiny_tracker5/Makefile @@ -0,0 +1,64 @@ +include ../../../Makefile.inc + +DOS33 = ../../../utils/dos33fs-utils/dos33 +EMPTY_DISK = ../../../empty_disk +HGR2PNG = ../../../utils/hgr-utils/png2hgr +LINKER_SCRIPTS = ../../../linker_scripts +TOKENIZE = ../../../utils/asoft_basic-utils/tokenize_asoft + +all: hgr_d2.dsk text_to_tiny + +submit: hgr_d2_1k.zip + +#### + +hgr_d2.dsk: HELLO D2 + cp $(EMPTY_DISK)/empty.dsk ./hgr_d2.dsk + $(DOS33) -y hgr_d2.dsk SAVE A HELLO + $(DOS33) -y hgr_d2.dsk -t BIN -a 0x6000 BSAVE D2 + +#### + +HELLO: hello.bas + $(TOKENIZE) < hello.bas > HELLO + +#### + +peasant_music.s: peasant.txt text_to_tiny + ./text_to_tiny peasant.txt > peasant_music.s + +#### + +mA2E_2.s: mA2E_2.txt text_to_tiny + ./text_to_tiny mA2E_2.txt > mA2E_2.s + +#### + +mA2E_3.s: mA2E_3.txt text_to_tiny + ./text_to_tiny mA2E_3.txt > mA2E_3.s + + +#### + +D2: d2.o + ld65 -o D2 d2.o -C $(LINKER_SCRIPTS)/apple2_6000.inc + +d2.o: d2.s \ + zp.inc hardware.inc \ + mA2E_2.s mA2E_3.s \ + mockingboard_init.s play_frame.s \ + tracker_init.s ay3_write_regs.s + ca65 -o d2.o d2.s -l d2.lst + +#### + +text_to_tiny: text_to_tiny.o + $(CC) -o text_to_tiny text_to_tiny.o -lm + +text_to_tiny.o: text_to_tiny.c + $(CC) $(CFLAGS) -c text_to_tiny.c + +#### + +clean: + rm -f *~ *.o *.lst D2 HELLO text_to_tiny mA2E_2.s diff --git a/music/tiny_tracker5/ay3_write_regs.s b/music/tiny_tracker5/ay3_write_regs.s new file mode 100644 index 00000000..63eee339 --- /dev/null +++ b/music/tiny_tracker5/ay3_write_regs.s @@ -0,0 +1,34 @@ + ;===================== + ;===================== + ;===================== + ; ay3 write regs + ;===================== + ;===================== + ;===================== + ; write all 14 registers at AY_REGS + +ay3_write_regs: + + ldx #13 +ay3_write_reg_loop: + + lda #MOCK_AY_LATCH_ADDR ; latch_address for PB1 ; 2 + ldy #MOCK_AY_INACTIVE ; go inactive ; 2 + + stx MOCK_6522_ORA1 ; put address on PA1 ; 4 + sta MOCK_6522_ORB1 ; latch_address on PB1 ; 4 + sty MOCK_6522_ORB1 ; 4 + + ; value + lda AY_REGS,X + sta MOCK_6522_ORA1 ; put value on PA1 ; 4 + lda #MOCK_AY_WRITE ; ; 2 + sta MOCK_6522_ORB1 ; write on PB1 ; 4 + sty MOCK_6522_ORB1 ; 4 + + dex + bpl ay3_write_reg_loop + +; rts + + diff --git a/music/tiny_tracker5/d2.s b/music/tiny_tracker5/d2.s new file mode 100644 index 00000000..abb894ea --- /dev/null +++ b/music/tiny_tracker5/d2.s @@ -0,0 +1,57 @@ +; Apple II graphics/music in 1k + +; by deater (Vince Weaver) + +; Zero Page + .include "zp.inc" + .include "hardware.inc" + +; for a 256 entry we need to fit in 252 bytes + +; 310 bytes -- initial +; 268 bytes -- strip out interrupts +; 262 bytes -- simplify init +; 261 bytes -- optimize init more +; 253 bytes -- optimize var init +; 252 bytes -- bne vs jmp +; 250 bytes -- song only has 16 notes so can never be negative +; 249 bytes -- make terminating value $80 instead of $FF +; 247 bytes -- combine note loop. makes song a bit faster +; 245 bytes -- try to optimize writing out volume +; 255 bytes -- add in some visualization +; 252 bytes -- re-arrange decode code +d2: + + ;=================== + ; music Player Setup + +tracker_song = peasant_song + + ; assume mockingboard in slot#4 + + ; inline mockingboard_init + +.include "mockingboard_init.s" + +.include "tracker_init.s" + + jsr SETGR ; enable lo-res graphics + +game_loop: + ; typically A=0, X=FF, Y=0 here + + ; play a frame of music + +.include "play_frame.s" +.include "ay3_write_regs.s" + + ; delay 20Hz, or 1/20s = 50ms + + lda #140 + jsr WAIT + + beq game_loop + +; music +.include "mA2E_2.s" + diff --git a/music/tiny_tracker5/hardware.inc b/music/tiny_tracker5/hardware.inc new file mode 100644 index 00000000..e68b75fd --- /dev/null +++ b/music/tiny_tracker5/hardware.inc @@ -0,0 +1,92 @@ +;; HARDWARE LOCATIONS + +KEYPRESS = $C000 +KEYRESET = $C010 + +;; SOFT SWITCHES +CLR80COL = $C000 ; PAGE0/PAGE1 normal +SET80COL = $C001 ; PAGE0/PAGE1 switches PAGE0 in Aux instead +EIGHTYCOLOFF = $C00C +EIGHTYCOLON = $C00D +SPEAKER = $C030 +SET_GR = $C050 +SET_TEXT = $C051 +FULLGR = $C052 +TEXTGR = $C053 +PAGE1 = $C054 +PAGE2 = $C055 +LORES = $C056 ; Enable LORES graphics +HIRES = $C057 ; Enable HIRES graphics +AN3 = $C05E ; Annunciator 3 + +PADDLE_BUTTON0 = $C061 +PADDL0 = $C064 +PTRIG = $C070 + +;; BASIC ROUTINES + +NORMAL = $F273 +HGR2 = $F3D8 +HCLR = $F3F2 +HPOSN = $F411 ; (Y,X),(A) (values stores in HGRX,XH,Y) +XDRAW0 = $F65D + + +;; MONITOR ROUTINES + +HLINE = $F819 ;; HLINE Y,$2C at A +VLINE = $F828 ;; VLINE A,$2D at Y +CLRSCR = $F832 ;; Clear low-res screen +CLRTOP = $F836 ;; clear only top of low-res screen +SETGR = $FB40 ;; GR +SETCOL = $F864 ;; COLOR=A +TEXT = $FB36 +TABV = $FB5B ;; VTAB to A +BELL = $FBDD ;; ring the bell +BASCALC = $FBC1 ;; +VTAB = $FC22 ;; VTAB to CV +HOME = $FC58 ;; Clear the text screen +WAIT = $FCA8 ;; delay 1/2(26+27A+5A^2) us +CROUT1 = $FD8B +SETINV = $FE80 ;; INVERSE +SETNORM = $FE84 ;; NORMAL +COUT = $FDED ;; output A to screen +COUT1 = $FDF0 ;; output A to screen + + + + + + +COLOR_BLACK = 0 +COLOR_RED = 1 +COLOR_DARKBLUE = 2 +COLOR_PURPLE = 3 +COLOR_DARKGREEN = 4 +COLOR_GREY = 5 +COLOR_MEDIUMBLUE = 6 +COLOR_LIGHTBLUE = 7 +COLOR_BROWN = 8 +COLOR_ORANGE = 9 +COLOR_GREY2 = 10 +COLOR_PINK = 11 +COLOR_LIGHTGREEN = 12 +COLOR_YELLOW = 13 +COLOR_AQUA = 14 +COLOR_WHITE = 15 + +COLOR_BOTH_BLACK = $00 +COLOR_BOTH_RED = $11 +COLOR_BOTH_DARKBLUE = $22 +COLOR_BOTH_DARKGREEN = $44 +COLOR_BOTH_GREY = $55 +COLOR_BOTH_MEDIUMBLUE = $66 +COLOR_BOTH_LIGHTBLUE = $77 +COLOR_BOTH_BROWN = $88 +COLOR_BOTH_ORANGE = $99 +COLOR_BOTH_PINK = $BB +COLOR_BOTH_LIGHTGREEN = $CC +COLOR_BOTH_YELLOW = $DD +COLOR_BOTH_AQUA = $EE +COLOR_BOTH_WHITE = $FF + diff --git a/music/tiny_tracker5/hello.bas b/music/tiny_tracker5/hello.bas new file mode 100644 index 00000000..52c966ed --- /dev/null +++ b/music/tiny_tracker5/hello.bas @@ -0,0 +1,6 @@ +5 HOME +10 PRINT "HGR DEMO2 - 1K DEMO AT DEMOSPLASH 2021" +20 PRINT CHR$(4);"CATALOG" +25 PRINT:PRINT "AUTOMATICALLY STARTING" +30 PRINT "]BRUN D2" +40 PRINT CHR$(4);"BRUN D2" diff --git a/music/tiny_tracker5/mA2E_2.txt b/music/tiny_tracker5/mA2E_2.txt new file mode 100644 index 00000000..d3a5eb5a --- /dev/null +++ b/music/tiny_tracker5/mA2E_2.txt @@ -0,0 +1,212 @@ +'' TITLE: Mockingboard Tune 2 - 2021 +' AUTHOR: mA2E / dSr +' COMMENTS: +' +' LOOP: 640 +' +' BPM: 250 +' TEMPO: 6 +' FREQ: 1000000 +' IRQ: 50 +' +' LYRICS: 0 +' +' ENDHEADER +------- +' 0 +0 F 3-- C 5-- C 4-- +1 ----- ----- ----- +2 ----- ----- ----- +3 ----- ----- ----- +4 ----- ----- F 4-- +5 ----- ----- ----- +6 ----- ----- ----- +7 ----- ----- ----- +8 F 4-- F 5-- G 4-- +9 ----- ----- ----- +A ----- ----- ----- +B ----- ----- ----- +C ----- ----- G#4-- +D ----- ----- ----- +E ----- ----- ----- +F ----- ----- ----- +10 F 3-- G 5-- C 4-- +11 ----- ----- ----- +12 ----- ----- ----- +13 ----- ----- ----- +14 ----- ----- F 4-- +15 ----- ----- ----- +16 ----- ----- ----- +17 ----- ----- ----- +18 F 4-- G#5-- G 4-- +19 ----- ----- ----- +1A ----- ----- ----- +1B ----- ----- ----- +1C ----- ----- G#4-- +1D ----- ----- ----- +1E ----- ----- ----- +1F ----- ----- ----- +20 F 3-- G 5-- C 4-- +21 ----- ----- ----- +22 ----- ----- ----- +23 ----- ----- ----- +24 ----- ----- F 4-- +25 ----- ----- ----- +26 ----- ----- ----- +27 ----- ----- ----- +28 F 4-- G#5-- G 4-- +29 ----- ----- ----- +2A ----- ----- ----- +2B ----- ----- ----- +2C ----- ----- G#4-- +2D ----- ----- ----- +2E ----- ----- ----- +2F ----- ----- ----- +30 F 3-- C 6-- C 4-- +31 ----- ----- ----- +32 ----- ----- ----- +33 ----- ----- ----- +34 ----- ----- F 4-- +35 ----- ----- ----- +36 ----- ----- ----- +37 ----- ----- ----- +38 F 4-- ----- G 4-- +39 ----- ----- ----- +3A ----- ----- ----- +3B ----- ----- ----- +3C ----- ----- G#4-- +3D ----- ----- ----- +3E ----- ----- ----- +3F ----- ----- ----- +' 1 +0 D#3-- A#5-- D#4-- +1 ----- ----- ----- +2 ----- ----- ----- +3 ----- ----- ----- +4 ----- ----- G 4-- +5 ----- ----- ----- +6 ----- ----- ----- +7 ----- ----- ----- +8 D#4-- G#5-- G#4-- +9 ----- ----- ----- +A ----- ----- ----- +B ----- ----- ----- +C ----- ----- A#4-- +D ----- ----- ----- +E ----- ----- ----- +F ----- ----- ----- +10 D#3-- G 5-- D#4-- +11 ----- ----- ----- +12 ----- ----- ----- +13 ----- ----- ----- +14 ----- ----- G 4-- +15 ----- ----- ----- +16 ----- ----- ----- +17 ----- ----- ----- +18 D#4-- F 5-- G#4-- +19 ----- ----- ----- +1A ----- ----- ----- +1B ----- ----- ----- +1C ----- ----- A#4-- +1D ----- ----- ----- +1E ----- ----- ----- +1F ----- ----- ----- +20 D#3-- G 5-- D#4-- +21 ----- ----- ----- +22 ----- ----- ----- +23 ----- ----- ----- +24 ----- ----- G 4-- +25 ----- ----- ----- +26 ----- ----- ----- +27 ----- ----- ----- +28 D#4-- G#5-- G#4-- +29 ----- ----- ----- +2A ----- ----- ----- +2B ----- ----- ----- +2C ----- ----- A#4-- +2D ----- ----- ----- +2E ----- ----- ----- +2F ----- ----- ----- +30 D#3-- G 5-- D#4-- +31 ----- ----- ----- +32 ----- ----- ----- +33 ----- ----- ----- +34 ----- ----- G 4-- +35 ----- ----- ----- +36 ----- ----- ----- +37 ----- ----- ----- +38 D#4-- ----- G#4-- +39 ----- ----- ----- +3A ----- ----- ----- +3B ----- ----- ----- +3C ----- ----- A#4-- +3D ----- ----- ----- +3E ----- ----- ----- +3F ----- ----- ----- +' 2 +0 C#3-- G 5-- C#4-- +1 ----- ----- ----- +2 ----- ----- ----- +3 ----- ----- ----- +4 ----- ----- F 4-- +5 ----- ----- ----- +6 ----- ----- ----- +7 ----- ----- ----- +8 C#4-- ----- G 4-- +9 ----- ----- ----- +A ----- ----- ----- +B ----- ----- ----- +C ----- ----- G#4-- +D ----- ----- ----- +E ----- ----- ----- +F ----- ----- ----- +10 C#3-- G#5-- C#4-- +11 ----- ----- ----- +12 ----- ----- ----- +13 ----- ----- ----- +14 ----- ----- F 4-- +15 ----- ----- ----- +16 ----- ----- ----- +17 ----- ----- ----- +18 C#4-- F 5-- G 4-- +19 ----- ----- ----- +1A ----- ----- ----- +1B ----- ----- ----- +1C ----- ----- G#4-- +1D ----- ----- ----- +1E ----- ----- ----- +1F ----- ----- ----- +20 C#3-- ----- C#4-- +21 ----- ----- ----- +22 ----- ----- ----- +23 ----- ----- ----- +24 ----- ----- F 4-- +25 ----- ----- ----- +26 ----- ----- ----- +27 ----- ----- ----- +28 C#4-- ----- G 4-- +29 ----- ----- ----- +2A ----- ----- ----- +2B ----- ----- ----- +2C ----- ----- G#4-- +2D ----- ----- ----- +2E ----- ----- ----- +2F ----- ----- ----- +30 C#3-- ----- C#4-- +31 ----- ----- ----- +32 ----- ----- ----- +33 ----- ----- ----- +34 ----- ----- F 4-- +35 ----- ----- ----- +36 ----- ----- ----- +37 ----- ----- ----- +38 C#4-- ----- G 4-- +39 ----- ----- ----- +3A ----- ----- ----- +3B ----- ----- ----- +3C ----- ----- A#4-- +3D ----- ----- ----- +3E ----- ----- ----- +3F ----- ----- ----- + + diff --git a/demos/l/music_256/mA2E_3.txt b/music/tiny_tracker5/mA2E_3.txt similarity index 100% rename from demos/l/music_256/mA2E_3.txt rename to music/tiny_tracker5/mA2E_3.txt diff --git a/music/tiny_tracker5/mockingboard_init.s b/music/tiny_tracker5/mockingboard_init.s new file mode 100644 index 00000000..dfe014a6 --- /dev/null +++ b/music/tiny_tracker5/mockingboard_init.s @@ -0,0 +1,77 @@ +; Mockingboad programming: +; + Has two 6522 I/O chips connected to two AY-3-8910 chips +; + Optionally has some speech chips controlled via the outport on the AY +; + Often in slot 4 +; TODO: how to auto-detect? +; References used: +; http://macgui.com/usenet/?group=2&id=8366 +; 6522 Data Sheet +; AY-3-8910 Data Sheet + +;======================== +; Mockingboard card +; Essentially two 6522s hooked to the Apple II bus +; Connected to AY-3-8910 chips +; PA0-PA7 on 6522 connected to DA0-DA7 on AY +; PB0 on 6522 connected to BC1 +; PB1 on 6522 connected to BDIR +; PB2 on 6522 connected to RESET + + +; left speaker +MOCK_6522_ORB1 = $C400 ; 6522 #1 port b data +MOCK_6522_ORA1 = $C401 ; 6522 #1 port a data +MOCK_6522_DDRB1 = $C402 ; 6522 #1 data direction port B +MOCK_6522_DDRA1 = $C403 ; 6522 #1 data direction port A +MOCK_6522_T1CL = $C404 ; 6522 #1 t1 low order latches +MOCK_6522_T1CH = $C405 ; 6522 #1 t1 high order counter +MOCK_6522_T1LL = $C406 ; 6522 #1 t1 low order latches +MOCK_6522_T1LH = $C407 ; 6522 #1 t1 high order latches +MOCK_6522_T2CL = $C408 ; 6522 #1 t2 low order latches +MOCK_6522_T2CH = $C409 ; 6522 #1 t2 high order counters +MOCK_6522_SR = $C40A ; 6522 #1 shift register +MOCK_6522_ACR = $C40B ; 6522 #1 auxilliary control register +MOCK_6522_PCR = $C40C ; 6522 #1 peripheral control register +MOCK_6522_IFR = $C40D ; 6522 #1 interrupt flag register +MOCK_6522_IER = $C40E ; 6522 #1 interrupt enable register +MOCK_6522_ORANH = $C40F ; 6522 #1 port a data no handshake + + +; right speaker +MOCK_6522_ORB2 = $C480 ; 6522 #2 port b data +MOCK_6522_ORA2 = $C481 ; 6522 #2 port a data +MOCK_6522_DDRB2 = $C482 ; 6522 #2 data direction port B +MOCK_6522_DDRA2 = $C483 ; 6522 #2 data direction port A + +; AY-3-8910 commands on port B +; RESET BDIR BC1 +MOCK_AY_RESET = $0 ; 0 0 0 +MOCK_AY_INACTIVE = $4 ; 1 0 0 +MOCK_AY_READ = $5 ; 1 0 1 +MOCK_AY_WRITE = $6 ; 1 1 0 +MOCK_AY_LATCH_ADDR = $7 ; 1 1 1 + + + ;======================== + ;======================== + ; Mockingboard Init + ;======================== + ;======================== + ; Left channel only + +mockingboard_init: + + ;========================= + ; Initialize the 6522s + ; Reset Left AY-3-8910 + ;=========================== + + ldx #$FF + stx MOCK_6522_DDRB1 + stx MOCK_6522_DDRA1 + + inx ; #MOCK_AY_RESET $0 + stx MOCK_6522_ORB1 + ldx #MOCK_AY_INACTIVE ; $4 + stx MOCK_6522_ORB1 + diff --git a/music/tiny_tracker5/play_frame.s b/music/tiny_tracker5/play_frame.s new file mode 100644 index 00000000..9be4ecf9 --- /dev/null +++ b/music/tiny_tracker5/play_frame.s @@ -0,0 +1,79 @@ +play_frame: + + ;============================ + ; see if still counting down + + lda SONG_COUNTDOWN + bpl done_update_song + +set_notes_loop: + + ;================== + ; load next byte + + ldy SONG_OFFSET + lda tracker_song,Y + + ;================== + ; see if hit end + + ; this song only 16 notes so valid notes always positive +; cmp #$80 + bpl not_end + + ;==================================== + ; if at end, loop back to beginning + + asl ; reset song offset to 0 + sta SONG_OFFSET + beq set_notes_loop + +not_end: + + ; NNNNNECC -- c=channel, e=end, n=note + + pha ; save note + + and #3 + tax + ldy #$0E + sty AY_REGS+8,X ; $08 set volume A,B,C + + asl + tax ; put channel offset in X + + + pla ; restore note + tay + and #$4 + sta SONG_COUNTDOWN ; always 4 long? + + tya + lsr + lsr + lsr ; get note in A + + tay ; lookup in table + + lda frequencies_high,Y + sta AY_REGS+1,X +; sta $500,X + + lda frequencies_low,Y + sta AY_REGS,X ; set proper register value + + ; visualization +blah_urgh: + sta $400,Y + inc blah_urgh+1 + + + ;============================ + ; point to next + + ; assume less than 256 bytes + inc SONG_OFFSET + +done_update_song: + dec SONG_COUNTDOWN + bmi set_notes_loop diff --git a/music/tiny_tracker5/text_to_tiny.c b/music/tiny_tracker5/text_to_tiny.c new file mode 100644 index 00000000..e06c41e5 --- /dev/null +++ b/music/tiny_tracker5/text_to_tiny.c @@ -0,0 +1,511 @@ +/* make music for tiny_music player */ + +#define VERSION "1.0" + +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +static int octave_adjust=0; + +static int notes_used[64]; +static int allocated_notes[64]; +static int notes_allocated=0; +static int total_len=0; + + +unsigned short frequencies[]={ +//C C# D D# E F F# G G# A A# B +//0x7A3,0x735,0x6CD,0x66C,0x60F,0x5B8,0x566,0x518,0x4CF,0x48A,0x449,0x40B, +0x3D1,0x39A,0x366,0x336,0x307,0x2DC,0x2B3,0x28C,0x267,0x245,0x224,0x205, +0x1E8,0x1CD,0x1B3,0x19B,0x183,0x16E,0x159,0x146,0x133,0x122,0x112,0x102, //3 +0x0F4,0x0E6,0x0D9,0x0CD,0x0C1,0x0B7,0x0AC,0x0A3,0x099,0x091,0x089,0x081, //4 +0x07A,0x073,0x06C,0x066,0x060,0x05B,0x056,0x051,0x04C,0x048,0x044,0x040, //5 +//0x03D,0x039,0x036,0x033,0x030,0x02D,0x02B,0x028,0x026,0x024,0x022,0x020, //6 +//0x01E,0x01C,0x01B,0x019,0x018,0x016,0x015,0x014,0x013,0x012,0x011,0x010, +//0x00F,0x00E,0x00D,0x00C,0x00C,0x00B,0x00A,0x00A,0x009,0x009,0x008,0x008, +}; + +// CLLNNNN + +int note_to_ed(char note, int flat, int sharp, int octave) { + + int offset; + + switch(note) { + case 'C': offset=0; break; + case 'D': offset=2; break; + case 'E': offset=4; break; + case 'F': offset=5; break; + case 'G': offset=7; break; + case 'A': offset=9; break; + case 'B': offset=11; break; + + case 'R': offset=12; flat=0; sharp=0; octave=3; break; + + default: + fprintf(stderr,"Unknown note %c\n",note); + return -1; + } + if (flat==1) offset--; + if (sharp==1) offset++; + if (sharp==2) offset+=2; + + + offset=((((octave+octave_adjust)-3)&0x3)*12)+offset; +// offset=((((octave+octave_adjust)-3)&0x3)<<4)|offset; + + return offset; +} + + + +static int debug=0; + +static int line=0; + +static int header_version=0; + + +struct note_type { + unsigned char which; + unsigned char note; + int sharp,flat; + int octave; + int len; + int enabled; + int freq; + int length; + int left; + int ed_freq; + int offset; +}; + + +static int allocate_note(int which) { + + int i; + for(i=0;isharp=0; + n->flat=0; + n->ed_freq=-1; + n->note=ch; + sp++; + if (string[sp]==' ') ; + else if (string[sp]=='#') n->sharp=1; + else if (string[sp]=='-') n->flat=1; + else if (string[sp]=='=') n->flat=2; + else { + fprintf(stderr,"Unknown note modifier %c, line %d:%d\n", + string[sp],line,sp); + fprintf(stderr,"String: %s\n",string); + } +// printf("Sharp=%d Flat=%d\n",n->sharp,n->flat); + sp++; + n->octave=string[sp]-'0'; + sp++; + sp++; + n->len=string[sp]-'0'; + sp++; + + + if (n->note!='-') { + + freq=note_to_ed(n->note,n->flat,n->sharp, + n->octave); + + n->enabled=1; + n->length=0; + n->ed_freq=freq; + } + + if (header_version==2) sp+=6; + + return sp; +} + +static int get_string(char *string, char *key, char *output, int strip_linefeed) { + + char *found; + + found=strstr(string,key); + found=found+strlen(key); + + /* get rid of leading whitespace */ + while(1) { + if ((*found==' ') || (*found=='\t')) found++; + else break; + } + + strcpy(output,found); + + /* remove trailing linefeed */ + if (strip_linefeed) output[strlen(output)-1]=0; + + return 0; + +} + +static void print_help(int just_version, char *exec_name) { + + printf("\ntext_to_ed version %s by Vince Weaver \n\n",VERSION); + if (just_version) exit(0); + + printf("This created Electric Duet files\n\n"); + + printf("Usage:\n"); + printf("\t%s [-h] [-v] [-d] [-o X] [-i X] textfile outbase\n\n", + exec_name); + printf("\t-h: this help message\n"); + printf("\t-v: version info\n"); + printf("\t-d: print debug messages\n"); + printf("\t-o: Offset octave by X\n"); + printf("\t-i: set second instrument to X\n"); + + exit(0); +} + + +static int write_note(int *a_last,int *b_last,int *c_last,int *total_len) { + + unsigned char temp_value; + + if (*a_last>=0) { + temp_value=(*a_last<<3)|0; + if ((*b_last<0) && (*c_last<0)) { + temp_value|=4; + } + printf("\t.byte $%02X ; A=%d L=%d\n", + temp_value, + *a_last,(*b_last<0)||(*c_last<0)); + (*total_len)++; + *a_last=-1; + } + + if (*b_last>=0) { + temp_value=(*b_last<<3)|1; + if (*c_last<0) temp_value|=4; + + printf("\t.byte $%02X ; B=%d L=%d\n", + temp_value, + *b_last,(*c_last<0)); + (*total_len)++; + *b_last=-1; + } + + if (*c_last>=0) { + printf("\t.byte $%02X ; C=%d L=%d\n", + (unsigned char)(*c_last<<3)|4|2, + *c_last,1); + (*total_len)++; + *c_last=-1; + } + + return 0; +} + + +int main(int argc, char **argv) { + + char string[BUFSIZ]; + char *result; + char *in_filename; + char temp[BUFSIZ]; + FILE *in_file=NULL; + //int attributes=0; + int loop=0; + int sp,external_frequency,irq; + struct note_type a,b,c; + int copt; + + char song_name[BUFSIZ]; + char author_name[BUFSIZ]; + char comments[BUFSIZ]; + char *comments_ptr=comments; + +// unsigned char sharp_char[]=" #-="; + + /* Parse command line arguments */ + while ((copt = getopt(argc, argv, "dhvo:i:"))!=-1) { + switch (copt) { + case 'd': + /* Debug messages */ + printf("Debug enabled\n"); + debug=1; + break; + case 'h': + /* help */ + print_help(0,argv[0]); + break; + case 'v': + /* version */ + print_help(1,argv[0]); + break; + case 'o': + /* octave offset */ + octave_adjust=atoi(optarg); + break; + default: + print_help(0,argv[0]); + break; + } + } + + if (argv[optind]!=NULL) { + /* Open the input file */ + if (argv[optind][0]=='-') { + in_file=stdin; + } + else { + in_filename=strdup(argv[optind]); + in_file=fopen(in_filename,"r"); + if (in_file==NULL) { + fprintf(stderr,"Couldn't open %s\n",in_filename); + return -1; + } + } + } + + + /* Get the info for the header */ + + while(1) { + result=fgets(string,BUFSIZ,in_file); + if (result==NULL) break; + line++; + if (strstr(string,"ENDHEADER")) break; + if (strstr(string,"HEADER:")) { + get_string(string,"HEADER:",temp,1); + header_version=atoi(temp); + printf("Found header version %d\n",header_version); + } + if (strstr(string,"TITLE:")) { + get_string(string,"TITLE:",song_name,1); + } + if (strstr(string,"AUTHOR:")) { + get_string(string,"AUTHOR:",author_name,1); + } + if (strstr(string,"COMMENTS:")) { + get_string(string,"COMMENTS:",comments_ptr,0); + comments_ptr=&comments[strlen(comments)]; + } + + if (strstr(string,"FREQ:")) { + get_string(string,"FREQ:",temp,1); + external_frequency=atoi(temp); + } + if (strstr(string,"IRQ:")) { + get_string(string,"IRQ:",temp,1); + irq=atoi(temp); + } + if (strstr(string,"LOOP:")) { + get_string(string,"LOOP:",temp,1); + loop=atoi(temp); + } + + } + + a.which='A'; b.which='B'; c.which='C'; + + + // algorithm + // get A,B,C + + +// int first=1; +// int a_last=0,b_last=0,same_count=0; +// int a_len=0,b_len=0,a_freq=0,b_freq=0; + int current_length=0; + int first=1; + int a_last=-1,b_last=-1,c_last=-1; + +printf("peasant_song:\n"); +printf("; register init\n"); +//printf("\t.byte $00,$00,$00,$00,$00,$00 ; $00: A/B/C fine/coarse\n"); +//printf("\t.byte $00 ; $06\n"); +//printf("\t.byte $38 ; $07 mixer (ABC on)\n"); +//printf("\t.byte $0E,$0C,$0C ; $08 volume A/B/C\n"); +//printf("\t.byte $00,$00,$00,$00 ; $09\n"); +printf("\n"); + + while(1) { + result=fgets(string,BUFSIZ,in_file); + if (result==NULL) break; + line++; + + a.ed_freq=-1; + b.ed_freq=-1; + c.ed_freq=-1; + a.length=0; + b.length=0; + c.length=0; + + /* skip comments */ + if (string[0]=='\'') continue; + if (string[0]=='-') continue; + if (string[0]=='*') continue; + + sp=0; + + /* Skip line number */ + while((string[sp]!=' ' && string[sp]!='\t')) sp++; + + sp=get_note(string,sp,&a,line); + if (sp!=-1) sp=get_note(string,sp,&b,line); + if (sp!=-1) sp=get_note(string,sp,&c,line); + + if (a.ed_freq>=0) { + a.offset=allocate_note(a.ed_freq); + notes_used[a.ed_freq]++; + printf("; A: %d\n",a.ed_freq); + } + if (b.ed_freq>=0) { + b.offset=allocate_note(b.ed_freq); + notes_used[b.ed_freq]++; + printf("; B: %d\n",b.ed_freq); + } + if (c.ed_freq>=0) { + c.offset=allocate_note(c.ed_freq); + notes_used[c.ed_freq]++; + printf("; C: %d\n",c.ed_freq); + } + + if ((a.ed_freq>=0)||(b.ed_freq>=0)||(c.ed_freq>=0)) { + printf("; none: a=%d c=%d len=%d\n",a_last,c_last,current_length); + //(old) NNNNNLLC + //(new) NNNNNLCC + // L=Last + + if (!first) { + write_note(&a_last,&b_last,&c_last,&total_len); + } + current_length=0; + + //if (!first) { +// printf("\t.byte $%02X ; L = %d\n", +// current_length|0xc0,current_length); +// printf("\n"); +// current_length=0; +// total_len++; + //} + + + first=0; + } + + + if (a.ed_freq>=0) { + a_last=a.offset; + } + if (b.ed_freq>=0) { + b_last=b.offset; + } + if (c.ed_freq>=0) { + c_last=c.offset; + } + + current_length++; + + + } + + + printf("; last: a=%d c=%d len=%d\n",a_last,c_last,current_length); + write_note(&a_last,&b_last,&c_last,&total_len); + +// printf("\t.byte $FF ; end\n"); + + /* assume 16 notes or fewer */ + printf("\t.byte $80 ; end\n"); + total_len++; + + int o,n; + + for(o=0;o<4;o++) { + printf("; Octave %d : ",o); + for(n=0;n<12;n++) { + printf("%d ",notes_used[(o*12)+n]); + } + printf("\n"); + } + + printf("; %d notes allocated\n",notes_allocated); + + printf(";.byte "); + for(n=0;n>8)); + if (n!=(notes_allocated-1)) printf(","); + total_len++; + } + printf("\n"); + + + + printf("; total len=%d\n",total_len); + + (void) irq; + (void) loop; + (void) external_frequency; + + return 0; +} + diff --git a/demos/l/music_256/tracker_init.s b/music/tiny_tracker5/tracker_init.s similarity index 100% rename from demos/l/music_256/tracker_init.s rename to music/tiny_tracker5/tracker_init.s diff --git a/music/tiny_tracker5/zp.inc b/music/tiny_tracker5/zp.inc new file mode 100644 index 00000000..598dfecb --- /dev/null +++ b/music/tiny_tracker5/zp.inc @@ -0,0 +1,32 @@ +; zero page + +; pre-defined applesoft vars + +CH = $24 +CV = $25 +GBASL = $26 +GBASH = $27 +BASL = $28 +BASH = $29 + +AY_REGS = $70 + +SONG_L = $80 +SONG_H = $81 +SONG_OFFSET = $82 +SONG_COUNTDOWN = $83 +OCTAVE = $84 +REGISTER = $85 + +OUR_ROT = $A5 + +HGR_X = $E0 +HGR_XH = $E1 +HGR_Y = $E2 +HGR_COLOR = $E4 +HGR_PAGE = $E6 +HGR_SCALE = $E7 + +COUNT = $FB +FRAME = $FC +