1
0
mirror of https://github.com/sehugg/8bitworkshop.git synced 2025-01-07 11:34:04 +00:00
8bitworkshop/presets/apple2/deltamod.dasm
2020-07-16 10:25:28 -05:00

219 lines
4.4 KiB
Plaintext

processor 6502
;;;
;;; Apple ][ Delta Modulation Sample Playback
;;;
;;; This is the same technique used in the NES
;;; for DMC samples, but adapted for the Apple ][.
;;; Basically: Every 1 bit increases the signal
;;; by 1 step, every 0 bit decreases by 1 step.
;;; The signal is clipped on the top and bottom.
;;;
;;; The NES has a 6-bit DAC, but we only have a
;;; PWM-encoded signal. We have 13 different
;;; routines to generate pulses ranging from 4 to
;;; 14 microseconds long, with a 29-30 microsecond
;;; interval (when we reach 50% duty cycle we just
;;; invert the signal)
;;;
;;; Every 8 bits we have to fetch a new byte
;;; and increment the pointer. To avoid a high-
;;; frequency tone, we generate another pulse
;;; (the same width as the last) in this macro.
;;;
;;; also see http://michaeljmahon.com/RTSynth.html
;;; and https://github.com/oliverschmidt/Play-BTc
;;;
; uninitialized zero-page variables
seg.u ZEROPAGE
org $0
PWMBUF .ds 2
PWMLEN .ds 1
TRASH .ds 1
; speaker toggle
SPKR equ $c030
; max pulse width = PULWID*2+4 cycles
PULWID = 6
; sleep {1} cycles
; must be multiple of 2
MAC SLEEP2
.CYCLES SET {1}
IF .CYCLES == 1
ECHO "MACRO ERROR: 'SLEEP': Duration must be > 1"
ERR
ENDIF
IF .CYCLES >= 12
jsr HandyRTS
.CYCLES SET .CYCLES - 12
ENDIF
REPEAT .CYCLES / 2
nop
REPEND
ENDM
ENDM
; pulse macro
; flip speaker, wait N*2 cycles
; flip again, wait (MAX-N)*2 cycles
; total = 8 + PULWID*2 cycles
MAC PULSE
sta SPKR
SLEEP2 {1}*2
sta SPKR
SLEEP2 (PULWID-{1})*2
ENDM
; shift next bit into carry flag
; load next byte and increment
; pointer if needed
; {1} : index of previous routine
; {2} : index of current routine
; we also interleave a speaker pulse in this routine
; using macros (these are the IF/ENDIF blocks)
MAC NEXTPULSE
lsr ; next bit -> carry flag
dex ; bit count == 0?
bne .noinc ; skip inc/reload cycle
; total pulse = 15 + PULWID*2 cycles
IF {2} > 0
sta SPKR
ELSE
SLEEP2 8 ; no SPKR toggle, take up 8 cycles
ENDIF
IF {2} == 1
sta SPKR
ENDIF
; increment pointer lo byte
iny ; 2-1 = 1
IF {2} == 2
sta SPKR
ENDIF
bne .noinchi ; 4
; increment pointer hi byte
inc PWMBUF+1
dec PWMLEN
bne .noinchi
rts ; end of sound data
.noinchi
IF {2} == 3
sta SPKR
ENDIF
; reset X counter
ldx #8 ; 6
IF {2} == 4 || {2} == 5
sta SPKR
ENDIF
; fetch new byte
lda (PWMBUF),y ; 11
; a few NOPs to match the normal pulse width
; (inx, lsr, branch)
IF {2} == 6
sta SPKR
nop
ENDIF
IF {2} >= 7
nop
sta SPKR
ENDIF
; make up cycles we missed
SLEEP2 PULWID*2-6
.noinc
; carry flag still valid from lsr
bcc LVLS{1}
ENDM
;;; start of code
seg CODE
org $803 ; starting address
Start
lda #>(SAMPLE_END-SAMPLES)+1
sta PWMLEN
lda #<SAMPLES
sta PWMBUF
lda #>SAMPLES
sta PWMBUF+1
jsr Play
jmp Start
; handy RTS for wasting 6+6 cycles
HandyRTS
rts
; each pulse takes PULWID*2+18(-1) cycles
; every 8 bits, the last pulse is repeated
; 1022727 / 29.5 / (9/8) = 30817 Hz
; Y = PWMBUF lo address
; PWMBUF+0 = 0
Play
ldy PWMBUF
lda #0
sta PWMBUF
ldx #8
lda (PWMBUF),y
; each label emits a pulse at a duty cycle
; from 0 (silent) to 7
; if bit is 1, falls through to next highest level
; otherwise, jumps back to previous level
LVLS0
SLEEP2 PULWID*2+8 ; sleep, don't click
NEXTPULSE 0,0
LVLS1
PULSE 0
NEXTPULSE 0,1
LVLS2
PULSE 1
NEXTPULSE 1,2
LVLS3
PULSE 2
NEXTPULSE 2,3
LVLS4
PULSE 3
NEXTPULSE 3,4
LVLS5
PULSE 4
NEXTPULSE 4,5
LVLS6
LVLS6INV equ *+3 ; skip STA SPKR to invert signal
PULSE 5
NEXTPULSE 5,6
bcs LVLS7INV ; 1 = invert signal
LVLS7
LVLS7INV equ *+3 ; skip STA SPKR to invert signal
PULSE 4
NEXTPULSE 6INV,5 ; 0 = invert signal
LVLS8
PULSE 3
NEXTPULSE 7,4
LVLS9
PULSE 2
NEXTPULSE 8,3
LVLS10
PULSE 1
NEXTPULSE 9,2
LVLS11
PULSE 0
NEXTPULSE 10,1
LVLS12
SLEEP2 PULWID*2+8 ; sleep, don't click
NEXTPULSE 11,0
bcs LVLS12 ; if 1, stay at highest level
; delta-encoded samples
SAMPLES
.incbin "springchicken.dat12.bin" ; "Spring Chicken" by BryanTeoh
SAMPLE_END