mirror of
https://github.com/deater/dos33fsprogs.git
synced 2025-01-17 18:29:55 +00:00
338 lines
6.0 KiB
ArmAsm
338 lines
6.0 KiB
ArmAsm
; Lo-res fire animation, size-optimized
|
|
|
|
; by deater (Vince Weaver) <vince@deater.net>
|
|
|
|
; based on code described here http://fabiensanglard.net/doom_fire_psx/
|
|
|
|
FIRE_YSIZE=20
|
|
|
|
fire_init:
|
|
lda #<fire_framebuffer
|
|
sta FIRE_FB_L
|
|
|
|
lda #>fire_framebuffer
|
|
sta FIRE_FB_H
|
|
|
|
ldx #FIRE_YSIZE
|
|
clear_fire_loop:
|
|
|
|
lda #0
|
|
ldy #39
|
|
clear_fire_line_loop:
|
|
sta (FIRE_FB_L),Y
|
|
dey
|
|
bpl clear_fire_line_loop
|
|
done_fire_line_loop:
|
|
|
|
clc
|
|
lda FIRE_FB_L
|
|
adc #40
|
|
sta FIRE_FB_L
|
|
lda FIRE_FB_H
|
|
adc #0
|
|
sta FIRE_FB_H
|
|
|
|
dex
|
|
bne clear_fire_loop
|
|
|
|
lda #7
|
|
jsr fire_setline
|
|
|
|
rts
|
|
|
|
|
|
|
|
;======================================
|
|
; set bottom line to color in A
|
|
;======================================
|
|
;
|
|
fire_setline:
|
|
|
|
ldx #<(fire_framebuffer+(FIRE_YSIZE-1)*40)
|
|
stx FIRE_FB_L
|
|
|
|
ldx #>(fire_framebuffer+(FIRE_YSIZE-1)*40)
|
|
stx FIRE_FB_H
|
|
|
|
ldy #39
|
|
set_fire_line:
|
|
sta (FIRE_FB_L),Y
|
|
dey
|
|
bpl set_fire_line
|
|
done_set_fire_line:
|
|
|
|
rts
|
|
|
|
|
|
;===============================
|
|
;===============================
|
|
; Draw fire frame
|
|
;===============================
|
|
;===============================
|
|
|
|
draw_fire_frame:
|
|
|
|
;===============================
|
|
; Update fire frame
|
|
;===============================
|
|
|
|
lda #<fire_framebuffer ; 2
|
|
sta fire_smc5_fb+1 ; 4
|
|
lda #>fire_framebuffer ; 2
|
|
sta fire_smc5_fb+2 ; 4
|
|
|
|
lda #<(fire_framebuffer+40) ; 2
|
|
sta fire_smc5_fb2+1 ; 4
|
|
|
|
lda #>(fire_framebuffer+40) ; 2
|
|
sta fire_smc5_fb2+2 ; 4
|
|
|
|
ldx #0 ; 2
|
|
|
|
fire_fb_update:
|
|
|
|
; FIXME: optimize
|
|
; original = ??? (complex)
|
|
; -X : move
|
|
|
|
ldy #39
|
|
; stx FIRE_X ; 2
|
|
fire_fb_update_loop:
|
|
|
|
; Get random 16-bit number
|
|
|
|
jsr random16 ; 40
|
|
|
|
; get random number Q
|
|
|
|
lda SEEDL ; 3
|
|
and #$3 ; 2
|
|
sta FIRE_Q ; 3
|
|
|
|
|
|
; 0..12 = A volume, 13-25 = B volume, 26-39 = C volume
|
|
|
|
lda A_VOLUME ; 3
|
|
cpy #13 ; 2
|
|
bcc fire_vol ; blt ; 2/3
|
|
|
|
lda B_VOLUME ; 3
|
|
cpy #26 ; 2
|
|
bcc fire_vol ; blt ; 2/3
|
|
|
|
lda C_VOLUME ; 3
|
|
|
|
fire_vol:
|
|
and #$f ; 2
|
|
|
|
; adjust fire height with volume
|
|
|
|
cmp #$8 ; 2
|
|
bcs fire_medium ; bge ; 2/3
|
|
fire_low:
|
|
; Q=1 3/4 of time
|
|
lda FIRE_Q ; 3
|
|
beq fire_height_done ; 2/3
|
|
fire_low_br:
|
|
lda #1 ; 2
|
|
jmp fire_height_done ; 3
|
|
|
|
fire_medium:
|
|
cmp #$d ; 2
|
|
bcs fire_high ; blt ; 2/3
|
|
|
|
; Q=1 1/2 of time
|
|
lda FIRE_Q ; 3
|
|
and #$1 ; 2
|
|
|
|
jmp fire_height_done ; 3
|
|
fire_high:
|
|
; Q=1 1/4 of time
|
|
lda FIRE_Q ; 3
|
|
cmp #1 ; 2
|
|
beq fire_height_done ; 2/3
|
|
fire_high_br:
|
|
lda #0 ; 2
|
|
|
|
fire_height_done:
|
|
sta FIRE_Q ; 3
|
|
|
|
sty FIRE_Y ; 3
|
|
|
|
; bounds check
|
|
|
|
cpy #0 ; 2
|
|
beq fire_r_same ; 2/3
|
|
cpy #39 ; 2
|
|
beq fire_r_same ; 2/3
|
|
|
|
lda #$2 ; 2
|
|
bit SEEDH ; 3
|
|
bne fire_r_same ; 2/3
|
|
lda SEEDH ; 3
|
|
and #$1 ; 2
|
|
beq r_up ; 2/3
|
|
r_down:
|
|
dey ; 2
|
|
jmp fire_r_same ; 3
|
|
r_up:
|
|
iny ; 2
|
|
fire_r_same:
|
|
|
|
; get next line color
|
|
fire_smc5_fb2:
|
|
lda $1234,Y ; 4+
|
|
|
|
ldy FIRE_Y ; 3
|
|
|
|
; adjust it
|
|
sec ; 2
|
|
sbc FIRE_Q ; 3
|
|
|
|
; saturate to 0
|
|
bpl fb_positive ; 2/3
|
|
lda #0 ; 2
|
|
fb_positive:
|
|
|
|
; store out
|
|
fire_smc5_fb:
|
|
sta $1234,Y ; store out ; 5
|
|
|
|
dey ; 2
|
|
bpl fire_fb_update_loop ; 2/3
|
|
|
|
done_fire_fb_update_loop:
|
|
|
|
; complicated adjustment
|
|
clc ; 2
|
|
lda fire_smc5_fb+1 ; 4
|
|
adc #40 ; 2
|
|
sta fire_smc5_fb+1 ; 4
|
|
lda fire_smc5_fb+2 ; 4
|
|
adc #0 ; 2
|
|
sta fire_smc5_fb+2 ; 4
|
|
|
|
clc ; 2
|
|
lda fire_smc5_fb2+1 ; 4
|
|
adc #40 ; 2
|
|
sta fire_smc5_fb2+1 ; 4
|
|
lda fire_smc5_fb2+2 ; 4
|
|
adc #0 ; 2
|
|
sta fire_smc5_fb2+2 ; 4
|
|
|
|
inx ; 2
|
|
cpx #(FIRE_YSIZE-1) ; 2
|
|
beq fire_update_done ; 2/3
|
|
jmp fire_fb_update ; 3
|
|
|
|
fire_update_done:
|
|
; ldx FIRE_X
|
|
|
|
;===============================
|
|
; copy framebuffer to low-res screen
|
|
;===============================
|
|
|
|
lda #<fire_framebuffer ; 2
|
|
sta fire_smc_fb+1 ; 5
|
|
lda #>fire_framebuffer ; 2
|
|
sta fire_smc_fb+2 ; 5
|
|
|
|
lda #<(fire_framebuffer+40) ; 2
|
|
sta fire_smc_fb2+1 ; 5
|
|
lda #>(fire_framebuffer+40) ; 2
|
|
sta fire_smc_fb2+2 ; 5
|
|
|
|
lda #16 ; 2
|
|
sta FIRE_FB_LINE ; 3
|
|
|
|
ldx #(FIRE_YSIZE/2) ; 2
|
|
|
|
fire_fb_copy:
|
|
|
|
ldy FIRE_FB_LINE ; 3
|
|
iny ; 2
|
|
iny ; 2
|
|
sty FIRE_FB_LINE ; 3
|
|
|
|
|
|
; Set up output, self-modifying code
|
|
lda gr_offsets,Y ; 4+
|
|
sta fire_smc_outl+1 ; 4
|
|
lda gr_offsets+1,Y ; 4+
|
|
clc ; 2
|
|
adc DRAW_PAGE ; 3
|
|
sta fire_smc_outl+2 ; 4
|
|
|
|
; FIXME: below, can we do this better?
|
|
; 50: original code
|
|
; 39: move to big lookup table
|
|
; 33: move save/restore X outside inner loop
|
|
; 30: self-modifying code
|
|
|
|
ldy #39 ; 2
|
|
stx FIRE_X ; 3
|
|
fire_fb_copy_loop:
|
|
; get top byte
|
|
; note, seems backwards, apple II LORES bottom byte is on top
|
|
fire_smc_fb2:
|
|
lda $1234,Y ; 4+
|
|
asl ; 2
|
|
asl ; 2
|
|
asl ; 2
|
|
; get bottom byte
|
|
fire_smc_fb:
|
|
ora $1234,Y ; 4+
|
|
tax ; 2
|
|
lda fire_colors,X ; 4+
|
|
fire_smc_outl:
|
|
sta $1234,Y ; store out ; 5
|
|
|
|
dey ; 2
|
|
bpl fire_fb_copy_loop ; 2/3
|
|
done_fire_fb_copy_loop:
|
|
ldx FIRE_X ; 3
|
|
|
|
|
|
; complicated adjustment
|
|
clc ; 2
|
|
lda fire_smc_fb+1 ; 4
|
|
adc #80 ; 2
|
|
sta fire_smc_fb+1 ; 5
|
|
lda fire_smc_fb+2 ; 4
|
|
adc #0 ; 2
|
|
sta fire_smc_fb+2 ; 5
|
|
|
|
clc ; 2
|
|
lda fire_smc_fb2+1 ; 4
|
|
adc #80 ; 2
|
|
sta fire_smc_fb2+1 ; 5
|
|
lda fire_smc_fb2+2 ; 4
|
|
adc #0 ; 2
|
|
sta fire_smc_fb2+2 ; 5
|
|
|
|
dex ; 2
|
|
bne fire_fb_copy ; 2/3
|
|
|
|
|
|
rts ; 6
|
|
|
|
;fire_colors_low: .byte $00,$00,$03,$02,$06,$07,$0E,$0F
|
|
;fire_colors_high: .byte $00,$00,$30,$20,$60,$70,$E0,$F0
|
|
|
|
fire_colors:
|
|
; 0 1 2 3 4 5 6 7
|
|
; 0 0 3 2 6 7 e f
|
|
.byte $00,$00,$03,$02,$06,$07,$0e,$0f ; 0
|
|
.byte $00,$00,$03,$02,$06,$07,$0e,$0f ; 0
|
|
.byte $30,$30,$33,$32,$36,$37,$3e,$3f ; 3
|
|
.byte $20,$20,$23,$22,$26,$27,$2e,$2f ; 2
|
|
.byte $60,$60,$63,$62,$66,$67,$6e,$6f ; 6
|
|
.byte $70,$70,$73,$72,$76,$77,$7e,$7f ; 7
|
|
.byte $e0,$e0,$e3,$e2,$e6,$e7,$ee,$ef ; e
|
|
.byte $f0,$f0,$f3,$f2,$f6,$f7,$fe,$ff ; f
|
|
|
|
; FIXME: just reserve space in our memory map
|
|
fire_framebuffer:
|
|
.res 40*FIRE_YSIZE, $00
|
|
|