dos33fsprogs/games/sb/hgr_sprite_big.s

206 lines
3.3 KiB
ArmAsm
Raw Normal View History

2023-06-02 16:37:01 -04:00
;===========================================
; hgr draw sprite (only at 7-bit boundaries)
;===========================================
2023-06-03 12:48:37 -04:00
; can handle sprites bigger than a 256 byte page
; Note this is optimized for blue/orange sprites
; it treats black0 as transparent
2023-06-02 16:37:01 -04:00
; SPRITE in INL/INH
; Location at SPRITE_X SPRITE_Y
; xsize, ysize in first two bytes
; sprite AT INL/INH
2023-06-03 12:48:37 -04:00
; orange = color5 1 101 0101 1 010 1010
2023-06-03 16:00:05 -04:00
hgr_draw_sprite_big:
2023-06-03 12:48:37 -04:00
lda SPRITE_X
ror
2023-06-03 16:00:05 -04:00
bcs hgr_draw_sprite_big_odd
2023-06-02 16:37:01 -04:00
2023-06-03 16:00:05 -04:00
hgr_draw_sprite_big_even:
2023-06-02 16:37:01 -04:00
ldy #0
lda (INL),Y ; load xsize
clc
adc SPRITE_X
2023-06-03 16:00:05 -04:00
sta big_sprite_width_end_smc+1 ; self modify for end of line
2023-06-02 16:37:01 -04:00
iny ; load ysize
lda (INL),Y
2023-06-03 16:00:05 -04:00
sta big_sprite_ysize_smc+1 ; self modify
2023-06-02 16:37:01 -04:00
; point smc to sprite
lda INL ; 16-bit add
2023-06-03 16:00:05 -04:00
sta big_sprite_smc1+1
2023-06-02 16:37:01 -04:00
lda INH
2023-06-03 16:00:05 -04:00
sta big_sprite_smc1+2
2023-06-02 16:37:01 -04:00
ldx #0 ; X is pointer offset
stx CURRENT_ROW ; actual row
ldx #2
2023-06-03 16:00:05 -04:00
hgr_big_sprite_yloop:
2023-06-02 16:37:01 -04:00
lda CURRENT_ROW ; row
clc
adc SPRITE_Y ; add in cursor_y
; calc GBASL/GBASH
tay ; get output ROW into GBASL/H
lda hposn_low,Y
sta GBASL
lda hposn_high,Y
clc
adc DRAW_PAGE
sta GBASH
ldy SPRITE_X
2023-06-03 16:00:05 -04:00
big_sprite_inner_loop:
2023-06-02 16:37:01 -04:00
2023-06-03 16:00:05 -04:00
big_sprite_smc1:
2023-06-02 16:37:01 -04:00
lda $f000,X ; load sprite data
beq big_sprite_transparent
2023-06-02 16:37:01 -04:00
sta (GBASL),Y ; store to screen
big_sprite_transparent:
2023-06-02 16:37:01 -04:00
inx ; increment sprite offset
; if > 1 page
2023-06-03 16:00:05 -04:00
bne big_sprite_no_page_cross
inc big_sprite_smc1+2
2023-06-02 16:37:01 -04:00
2023-06-03 16:00:05 -04:00
big_sprite_no_page_cross:
2023-06-02 16:37:01 -04:00
iny ; increment output position
2023-06-03 16:00:05 -04:00
big_sprite_width_end_smc:
2023-06-02 16:37:01 -04:00
cpy #6 ; see if reached end of row
2023-06-03 16:00:05 -04:00
bne big_sprite_inner_loop ; if not, loop
2023-06-02 16:37:01 -04:00
inc CURRENT_ROW ; row
lda CURRENT_ROW ; row
2023-06-03 16:00:05 -04:00
big_sprite_ysize_smc:
2023-06-02 16:37:01 -04:00
cmp #31 ; see if at end
2023-06-03 16:00:05 -04:00
bne hgr_big_sprite_yloop ; if not, loop
2023-06-02 16:37:01 -04:00
rts
2023-06-03 12:48:37 -04:00
2023-06-03 16:00:05 -04:00
hgr_draw_sprite_big_odd:
2023-06-03 12:48:37 -04:00
ldy #0
lda (INL),Y ; load xsize
clc
adc SPRITE_X
sta osprite_width_end_smc+1 ; self modify for end of line
iny ; load ysize
lda (INL),Y
sta osprite_ysize_smc+1 ; self modify
; point smc to sprite
lda INL ; 16-bit add
sta osprite_smc1+1
lda INH
sta osprite_smc1+2
ldx #0 ; X is pointer offset
stx CURRENT_ROW ; actual row
ldx #2
ohgr_sprite_yloop:
lda CURRENT_ROW ; row
clc
adc SPRITE_Y ; add in cursor_y
; calc GBASL/GBASH
tay ; get output ROW into GBASL/H
lda hposn_low,Y
sta GBASL
lda hposn_high,Y
clc
adc DRAW_PAGE
sta GBASH
ldy SPRITE_X
clc
php ; store 0 carry on stack
osprite_inner_loop:
osprite_smc1:
lda $f000,X ; load sprite data
bne osprite_not_transparent
; we can't just skip if 0 because we might shift a bit in
; from previous byte
plp
bcs osprite_oops
clc
php
bcc osprite_transparent_done
osprite_not_transparent:
2023-06-03 12:48:37 -04:00
plp ; restore carry from last
osprite_oops:
2023-06-03 12:48:37 -04:00
rol ; rotate in carry
asl ; one more time, bit6 in carry
php ; save on stack
sec ; assume blur/orange
ror ; rotate it back down
sta (GBASL),Y ; store to screen
osprite_transparent_done:
2023-06-03 12:48:37 -04:00
inx ; increment sprite offset
; if > 1 page
bne osprite_no_page_cross
inc osprite_smc1+2
osprite_no_page_cross:
iny ; increment output position
osprite_width_end_smc:
cpy #6 ; see if reached end of row
bne osprite_inner_loop ; if not, loop
plp ; restore stack
inc CURRENT_ROW ; row
lda CURRENT_ROW ; row
osprite_ysize_smc:
cmp #31 ; see if at end
bne ohgr_sprite_yloop ; if not, loop
rts