1
0
mirror of https://github.com/sehugg/8bitworkshop.git synced 2024-11-23 06:32:11 +00:00
8bitworkshop/presets/apple2/tb_6502.s
2019-09-08 18:59:08 -05:00

4303 lines
74 KiB
ArmAsm

; From: http://www.deater.net/weave/vmwprod/tb1/tb_6502.html
.segment "INIT"
.segment "ONCE"
; don't use $800-$1fff :(
.segment "STARTUP"
jmp Start
; use $4000- :)
.segment "CODE"
.define EQU =
; blt = bcc
; bge = bcs
;; ZERO PAGE
CH EQU $24
CV EQU $25
BASL EQU $28
BASH EQU $29
H2 EQU $2C
COLOR EQU $30
YSAV EQU $34
YSAV1 EQU $35
RANDOM_SEED EQU $43
;; Our Zero Page Allocations
PADDLE_STATUS EQU $CA
HISCORE_1 EQU $CB
HISCORE_2 EQU $CC
HISCORE_3 EQU $CD
HISCORE_H EQU $CE
HISCORE_L EQU $CF
BOSS_X EQU $D0
BOSS_XADD EQU $D1
BOSS_COUNT EQU $D2
BOSS_SMOKE EQU $D3
BOSS_EXPLODING EQU $D4
BOSS_WAITING EQU $D5
BOSS_HITS EQU $D6
BOSS_SHOOTING EQU $D7
ENEMIES_SPAWNED EQU $D8
ENEMY_TYPE EQU $D9
ENEMY_WAVE EQU $DA
CURRENT_INIT_X EQU $DB
CURRENT_ENEMY_KIND EQU $DC
TOTAL_ENEMIES_OUT EQU $DD
SCROLL EQU $DE
SOUND_ON EQU $DF
SHIPX EQU $E0
SHIPXADD EQU $E1
ENEMY_PL EQU $E2
ENEMY_PH EQU $E3
MISSILE_PL EQU $E4
MISSILE_PH EQU $E5
GR_PAGE EQU $E6
LEVEL EQU $E7
SHIELDS EQU $E8
SCOREL EQU $E9
SCOREH EQU $EA
BONUS_FLAGS EQU $EB
BCD_BYTEH EQU $EC
BCD_BYTE EQU $ED
COL_X1 EQU $EC
COL_X2 EQU $ED
COL_X3 EQU $EE
COL_X4 EQU $EF
ENEMY_EXPLODING EQU $F0
ENEMY_KIND EQU $F1
ENEMY_X EQU $F2
ENEMY_Y EQU $F3
ENEMY_XADD EQU $F4
ENEMY_YADD EQU $F5
ENEMY_XMIN EQU $F6
ENEMY_XMAX EQU $F7
BETWEEN_DELAY EQU $F8
ENEMY_WAIT EQU $F9
STRINGL EQU $FA
STRINGH EQU $FB
PARAM2 EQU $FC
RESULT EQU $FD
LASTKEY EQU $FE
TEMP EQU $FF
;; VECTORS
BASIC EQU $3D0 ;; VECTOR for return to Applesoft
KEYPRESS EQU $C000
KEYRESET EQU $C010
SPEAKER EQU $C030
;; SOFT SWITCHES
GR EQU $C050
TEXT EQU $C051
FULLGR EQU $C052
TEXTGR EQU $C053
PAGE0 EQU $C054
PAGE1 EQU $C055
LORES EQU $C056
HIRES EQU $C057
PADDLE_BUTTON0 EQU $C061
PADDL0 EQU $C064
PTRIG EQU $C070
;; MONITOR ROUTINES
HLINE EQU $F819 ;; HLINE Y,$2C at A
VLINE EQU $F828 ;; VLINE A,$2D at Y
CLRSCR EQU $F832 ;; Clear low-res screen
CLRTOP EQU $F836 ;; clear only top of low-res screen
SETCOL EQU $F864 ;; COLOR=A
BASCALC EQU $FBC1 ;;
HOME EQU $FC58 ;; Clear the text screen
WAIT EQU $FCA8 ;; delay 1/2(26+27A+5A^2) us
SETINV EQU $FE80 ;; INVERSE
SETNORM EQU $FE84 ;; NORMAL
COUT1 EQU $FDF0 ;; output A to screen
;; GAME PARAMETERS
NUM_MISSILES EQU 2
NUM_ENEMIES EQU 6
UP_SHIELDS EQU 32
WAVE_SIZE EQU 16
WAVES_TILL_BOSS EQU 5
;; BONUS_FLAGS
PERFECT_SHIELDS EQU $80
PERFECT_KILLS EQU $40
PERFECT_AIM EQU $1
;==========================================================
; MAIN()
;==========================================================
;==============================
; back up part of the zero page
;==============================
Start:
lda #>zero_page_save
sta BASH
lda #<zero_page_save
sta BASL
ldy #0
save_zp_loop:
lda $C8,Y ; copy $C8-$FF
sta (BASL),Y ; to (zero_page_save)
iny
cpy #$38 ; we copy 56 bytes
bne save_zp_loop
lda #$ff
sta SOUND_ON
;==========================
; set graphics mode, page 0
;==========================
jsr HOME
jsr set_page0_gr
;=================
; clear the screen
;=================
jsr CLRTOP
;====================
; setup the high score
;=====================
lda #$01
sta HISCORE_H
lda #$00
sta HISCORE_L
sta PADDLE_STATUS
; modestly make me the high scorer
lda #$D6
sta HISCORE_1
lda #$CD
sta HISCORE_2
lda #$D7
sta HISCORE_3
lda #>(score_string+31)
sta STRINGH
lda #<(score_string+31)
sta STRINGL
jsr print_high_score
;=============
; put vmw logo
;=============
lda #$7 ; y=7
sta CV
lda #$8 ; x=8
sta CH
lda #>vmw_sprite
sta STRINGH
lda #<vmw_sprite
sta STRINGL
jsr blit ; blit the vmw sprite
;==================================
; Write "A VMW SOFTWARE PRODUCTION"
;==================================
lda #>vmw_string ; string = vmw_string
sta STRINGH
lda #<vmw_string
sta STRINGL
jsr print_text_xy ; print the string
;======================
; wait until keypressed
;======================
jsr wait_until_keypressed
;======================
; Opening Graphic
;======================
opener:
;=================
; clear the screen
;=================
jsr set_page0_gr
jsr HOME ; clear the screen
jsr CLRTOP
;=======================
; put opening graphic up
;=======================
lda #$0 ; y=0
sta CV
sta CH
lda #>opener_sprite
sta STRINGH
lda #<opener_sprite
sta STRINGL
jsr blit ; blit the opener sprite
lda #$A ; y=10 (we had to split in two)
sta CV
lda #>opener_sprite_2
sta STRINGH
lda #<opener_sprite_2
sta STRINGL
jsr blit ; blit the second half of sprite
;================================================
; Write "MERCILESS MALICIOUS MARAUDING MARKETERS"
;================================================
lda #>mercy_string ; string = MMMM
sta STRINGH
lda #<mercy_string
sta STRINGL
jsr print_text_xy ; print the string
jsr wait_until_keypressed ; wait until key pressed
;==========================================================
; The MAIN MENU
;==========================================================
main_menu:
jsr set_page0_text ; enter text mode
jsr HOME ; clear screen
lda #>help_string ; string = "H FOR HELP"
sta STRINGH
lda #<help_string
sta STRINGL
jsr print_text_xy ; print the string
lda #$5
sta PARAM2
jsr do_menu ; do the menu
lda RESULT ; load menu item from page0
cmp #$0 ; is it zero?
bne check_1 ; if not move ahead
jmp do_new_game ; yes, start new game
check_1:
cmp #$1 ; is it one?
bne check_2 ; if not, move ahead
jmp do_about ; else, show about into
check_2:
cmp #$2 ; is it two?
bne check_3 ; if not, move ahead
jmp do_story ; else, do the story info
check_3:
cmp #$3 ; is it three?
bne check_4 ; if not, move ahead
jmp do_hi_score ; else, show hi-score
check_4:
cmp #$4 ; is it four?
bne check_5 ; if not move ahead
jmp exit ; if so exit
check_5:
jsr do_help
jmp main_menu ; return to main_menu loop
;==========================================================
; EXIT back to BASIC
;==========================================================
exit:
jsr SETNORM ; NORMAL text
jsr set_page0_text
;======================
; restore the zero page
;======================
lda #>zero_page_save
sta BASH
lda #<zero_page_save
sta BASL
ldy #0
restore_zp_loop:
lda (BASL),Y ; copy (zero_page_save)
sta $C8,Y ; back to $C8-$FF
iny
cpy #$38 ; we copy 56 bytes
bne restore_zp_loop
jsr HOME ; clear screen
jmp (BASIC) ; return to BASIC
;==========================================================
; do_menu
;==========================================================
; draw a text-mode menu
do_menu:
lda #$0 ; start with menu=0
sta RESULT
menu:
lda #$8 ; y=8
sta CV
lda #>new_game_string ; string starts at "NEW GAME"
sta STRINGH
lda #<new_game_string
sta STRINGL
ldy #$0 ; clear string offset pointer
ldx #$0 ; clear line count
next_line:
lda #$8D ; carriage return
jsr COUT1
lda #$10 ; x=16
sta CH
sty TEMP ; save Y as NORM/INV changes it
jsr SETNORM ; set text style to normal
cpx RESULT ; are we at the selected entry?
bne menu_restore_y ; if not, move on
jsr SETINV ; if so, reverse text
menu_restore_y:
ldy TEMP ; restore Y we saved earlier
menu_char:
lda (STRINGL),Y ; get char from strings
beq done_line ; if 0, done this line
jsr COUT1 ; monitor char output line
iny ; indexY ++
jmp menu_char ; loop till done line
done_line:
inx ; move to next line
iny ; pointer after null
cpx PARAM2 ; are we past end of menu?
bne next_line ; if not, print next line
menu_wait:
jsr wait_until_keypressed
lda LASTKEY
cmp #('Q')
bne check_h
lda #$4
sta RESULT
rts
check_h:
cmp #('H')
bne check_i
lda #$5 ; if help, set RESULT=5
sta RESULT
rts
check_i:
cmp #('I')
bne check_m
dec RESULT
jmp check_menu_values
check_m:
cmp #('M')
bne check_cr
inc RESULT
jmp check_menu_values
check_cr:
cmp #(13)
bne menu_wait
rts
check_menu_values:
lda RESULT
bpl check_menu_over
lda PARAM2
sbc #$1
sta RESULT
check_menu_over:
cmp PARAM2
bne menu
lda #$0
sta RESULT
jmp menu
;==========================================================
; do_about
;==========================================================
; print the "about" info
do_about:
bit GR
jsr CLRTOP
lda #$0 ; x,y= 12,0
sta CV
lda #$C
sta CH
lda #>vince_sprite
sta STRINGH
lda #<vince_sprite
sta STRINGL
jsr blit ; blit the pic of vince
lda #>about_lines
sta STRINGH
lda #<about_lines
sta STRINGL
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr wait_until_keypressed
lda #$6 ; only want to clear bottom 3 lines
sta BASH ; so set pointer to $6D0
lda #$D0
sta BASL
ldx #$01 ; set X already to 1
jsr bottom_y ; and jump midway into clear_bottom
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr wait_until_keypressed
jmp opener
;==========================================================
; do_help
;==========================================================
; print the "help" info
do_help:
jsr set_page0_text
jsr HOME
lda #>help_lines
sta STRINGH
lda #<help_lines
sta STRINGL
ldx #$14
jsr print_x_strings ; print text
jsr wait_until_keypressed
rts
;==========================================================
; do_story
;==========================================================
; the story routine
do_story:
bit GR
jsr CLRTOP
;
; draw the mars map
;
lda #$0 ; y=0
sta CV
sta CH
lda #>phobos_sprite
sta STRINGH
lda #<phobos_sprite
sta STRINGL
jsr blit ; blit the phobos sprite
lda #>story_lines
sta STRINGH
lda #<story_lines
sta STRINGL
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr wait_until_keypressed
;
; animate the droid launch
;
jsr clear_bottom
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr print_text_xy ; print text
lda #$12 ; x=18
sta CH
lda #$7
sta CV ; y=7
phobos_loop:
lda #>evil_ship_sprite
sta STRINGH
lda #<evil_ship_sprite
sta STRINGL
jsr blit ; blit the droid sprite
; delay
; wait 1/2 (26+27A+5A^2) us
lda #$8B ; 107 = 30080us = 30ms
jsr WAIT
lda #$0 ; set color to zero
jsr SETCOL
; HLINE H1,H2 at V1
lda CH ; get x value
tay ; move it to H1
adc #$3 ; add 3
sta H2 ; move it to H2
lda #$e ; set V1
jsr HLINE
ldy CH ; reset H1
lda #$f ; set V1 to bottom value
jsr HLINE
; move ship
inc CH ; incrememnt X
lda CH
cmp #$25 ; have we reached 37? (edge)
bne phobos_loop
jsr wait_until_keypressed
;
; "you are tom..."
;
jsr clear_bottom
jsr CLRTOP
lda #$1 ; y=0
sta CV
lda #$C ; x=12
sta CH
lda #>tom_sprite
sta STRINGH
lda #<tom_sprite
sta STRINGL
jsr blit ; blit the tom sprite
lda #>you_are_tom
sta STRINGH
lda #<you_are_tom
sta STRINGL
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr wait_until_keypressed
jmp opener
;==========================================================
; do_ending
;==========================================================
; the ending you get after level 1
do_ending:
jsr set_page0_gr ; set graphics mode
jsr CLRTOP
jsr clear_bottom
;
; draw the earth
;
lda #$7 ; y=7
sta CV
lda #$F ; x=15
sta CH
lda #>earth_sprite
sta STRINGH
lda #<earth_sprite
sta STRINGL
jsr blit ; blit the earth sprite
lda #>ending_lines
sta STRINGH
lda #<ending_lines
sta STRINGL
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr print_text_xy ; print text
ldx #30
jsr wait_X_100msec ; pause for 3 seconds
bit KEYRESET ; clear keyboard
jsr wait_until_keypressed
;
; second page
;
jsr clear_bottom
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr print_text_xy ; print text
jsr wait_until_keypressed
;
; third page
;
jsr CLRTOP
jsr clear_bottom
;
; draw Susie
;
lda #$7 ; y=7
sta CV
lda #$A ; x=10
sta CH
lda #>susie_sprite
sta STRINGH
lda #<susie_sprite
sta STRINGL
jsr blit ; blit the susie sprite
lda #>susie_lines
sta STRINGH
lda #<susie_lines
sta STRINGL
jsr print_text_xy ; print text
jsr wait_until_keypressed
;
; fourth page
;
jsr CLRTOP
jsr clear_bottom
;
; draw Tom's head
;
lda #$7 ; y=7
sta CV
lda #$F ; x=15
sta CH
lda #>tom_head_sprite
sta STRINGH
lda #<tom_head_sprite
sta STRINGL
jsr blit ; blit Tom's head
lda #>tom_sigh
sta STRINGH
lda #<tom_sigh
sta STRINGL
jsr print_text_xy ; print text
jsr wait_until_keypressed
jmp no_ending
;==========================================================
; START NEW GAME
;==========================================================
do_new_game:
;; set up struct pointers
lda #>missile_0 ; clear the missile struct
sta MISSILE_PH ; should make this clear all BSS
lda #<missile_0 ; at some point
sta MISSILE_PL
lda #>enemy_0
sta ENEMY_PH
lda #<enemy_0
sta ENEMY_PL
;; Clear BSS
ldy #$0
lda #$0
clear_bss:
sta start_bss,Y
iny
cpy #end_bss-start_bss
bne clear_bss
;; Init one-time vars
lda #$8
sta SHIELDS ; shields start at 8
jsr update_shields
lda #$12
sta SHIPX ; shipx at start is 18
lda #$1
sta LEVEL ; start at level 1
lda #$0
sta SCOREL
sta SCOREH
jsr print_score
new_level:
;========================
; Setup various variables
;========================
lda #(PERFECT_AIM|PERFECT_SHIELDS|PERFECT_KILLS)
sta BONUS_FLAGS ; set perfect shot/shield/enemies
lda #$14
sta ENEMY_WAIT
lda #$0
sta BETWEEN_DELAY
sta SHIPXADD ; clear shipxadd
sta ENEMY_WAVE
sta TOTAL_ENEMIES_OUT
sta ENEMIES_SPAWNED
sta ENEMY_TYPE
;=======================
; Print "LEVEL X"
;=======================
jsr set_page0_text
jsr HOME
;===================
; set level to level
;===================
lda #>(level_string+9)
sta STRINGH
lda #<(level_string+9)
sta STRINGL
lda #0
sta BCD_BYTEH
lda LEVEL
sta BCD_BYTE
jsr print_bcd_byte
;======================
; Print level on screen
;======================
lda #>(level_string_xy)
sta STRINGH
lda #<(level_string_xy)
sta STRINGL
jsr print_text_xy
ldx #20
jsr wait_X_100msec ; pause for 3 seconds
bit KEYRESET ; clear keyboard
;==================================
; Enter graphics mode, clear screen
;==================================
jsr set_page0_gr ; set graphics mode
jsr clear_screen ; clear screen
draw_stars:
;=====================
; Setup star field
;=====================
lda #>(star_field)
sta STRINGH
lda #<(star_field)
sta STRINGL
ldy #$0
star_init:
jsr random_number
and #$9f
clc
adc #$4
sta (STRINGL),Y
iny
lda #$0
sta (STRINGL),Y
iny
bne star_init
lda #$0
sta SCROLL
;/========================\
;+ +
;+ MAIN GAME LOOP +
;+ +
;\========================/
main_game_loop:
jsr clear_screen ; clear screen
jsr show_stars
done_scrolling:
; ================================
; put out new enemies (if needed)
; ================================
inc BETWEEN_DELAY ; inc how long we've delayed
lda BETWEEN_DELAY ; load it in
cmp ENEMY_WAIT ; have we waited long enough?
beq reset_delay
jmp move_enemies ; if not, go on to movement
reset_delay:
; delay==wait, so attempt to put out new enemy
lda BETWEEN_DELAY
and #$1
sta BETWEEN_DELAY ; reset delay
; special case for boss
lda #$9 ; if boss, don't keep track of
cmp ENEMY_TYPE ; how many enemies were spawned
bne not_boss_dont_clear
lda #$1 ; store 1 so we don't increment wave
sta ENEMIES_SPAWNED
not_boss_dont_clear:
; see if we are at a new wave
; basically, if 16 have been spawned, change
lda ENEMIES_SPAWNED
and #$0f
bne same_enemy_type ; if not 16 gone by, move on
;=======================
; change the enemy type
inc ENEMIES_SPAWNED
jsr random_number
and #$7 ; get a random number 0-7
sta ENEMY_TYPE
inc ENEMY_WAVE
lda ENEMY_WAVE ; have we gone enough waves to reach boss?
cmp #WAVES_TILL_BOSS
bne not_boss_yet
lda #$8
sta ENEMY_TYPE
not_boss_yet:
; set various constants
; these may be overriden later
lda #20
sec
sbc LEVEL
sta ENEMY_WAIT ; enemy_wait=20-level
; set kind and init x to be random by default
lda #$ff
sta CURRENT_ENEMY_KIND
sta CURRENT_INIT_X
same_enemy_type:
; find empty enemy slot
ldy #$0 ; point to enemies[0]
tya
find_empty_enemy:
pha
lda (ENEMY_PL),Y ; get enemy[y].out
beq add_enemy
pla
clc
adc #$9
tay
cpy #(NUM_ENEMIES*9)
bne find_empty_enemy
jmp move_enemies ; no empty, slots, move on
add_enemy:
pla
;==============================================
; First see if we must wait for enemy to clear
; types 2 and 8
lda ENEMY_TYPE
cmp #$2
bne check_type_8
lda TOTAL_ENEMIES_OUT
beq change_to_type_3
jmp move_enemies
change_to_type_3:
lda #$3
sta ENEMY_TYPE
jsr random_number
and #$8
sta CURRENT_ENEMY_KIND
jsr random_number
and #$1F ; mask off so 0-31
clc
adc #$2
asl A
sta CURRENT_INIT_X
jmp setup_enemy_defaults
check_type_8:
cmp #$8
beq before_boss_stuff
jmp check_type_9
before_boss_stuff:
;======================
; before boss stuff
lda TOTAL_ENEMIES_OUT
beq prepare_for_boss
jmp move_enemies
prepare_for_boss:
;===============
; HANDLE BONUSES
;===============
; Set text mode
jsr set_page0_text
jsr HOME
; Print "BONUS POINTS"
lda #>bonus_string
sta STRINGH
lda #<bonus_string
sta STRINGL
jsr print_text_xy ; print text
; Check to see if we had any bonuses
lda #$C1
bit BONUS_FLAGS
beq no_bonus
; Check if we had no hits on shields
perfect_shields:
bpl perfect_kills
jsr score_plus_50
lda #>bonus_shields
sta STRINGH
lda #<bonus_shields
sta STRINGL
jsr print_text_xy ; print text
; See if we killed all the enemies
perfect_kills:
bit BONUS_FLAGS
bvc perfect_aim
jsr score_plus_50
lda #>bonus_kills
sta STRINGH
lda #<bonus_kills
sta STRINGL
jsr print_text_xy ; print text
; See if no missiles missed
perfect_aim:
lda #$01
bit BONUS_FLAGS
beq done_bonus
jsr score_plus_50
lda #>bonus_aim
sta STRINGH
lda #<bonus_aim
sta STRINGL
jsr print_text_xy ; print text
jmp done_bonus
; we had no bonuses
no_bonus:
lda #>no_bonus_string
sta STRINGH
lda #<no_bonus_string
sta STRINGL
jsr print_text_xy ; print text
; Wait until a keypress, and return to graphics mode
done_bonus:
ldx #30
jsr wait_X_100msec ; pause for 3 seconds
bit KEYRESET ; clear keyboard
jsr wait_until_keypressed
jsr set_page0_gr
;======================
; setup boss
lda #$0C
sta BOSS_X ; boss_x = 13
lda #$1
sta BOSS_XADD ; boss_xadd=1
sta BOSS_WAITING ; boss_waiting=1
jsr random_number
and #$1f
sta BOSS_COUNT ; boss_count = rand%32
lda #$0
sta BOSS_SMOKE ; boss_smoke=0
sta BOSS_EXPLODING ; boss_exploding=0
sta BOSS_SHOOTING ; boss_shooting=0
lda LEVEL
asl A
clc
adc #$10
sta BOSS_HITS ; boss_hits=(level*2)+20
lda #$9
sta ENEMY_TYPE ; enemy_type=9
jmp move_enemies
check_type_9:
; if boss, and he's waiting,
; don't produce enemies
cmp #$9
bne setup_enemy_defaults
lda BOSS_WAITING
beq setup_enemy_defaults
jmp move_enemies
;========================
; setup enemy defaults
setup_enemy_defaults:
inc ENEMIES_SPAWNED
inc TOTAL_ENEMIES_OUT
lda #$1
sta (ENEMY_PL),Y ; enemy[i].out=1
lda #$0
iny ; exploding
sta (ENEMY_PL),Y ; enemy[i].exploding=0
iny ; kind
lda CURRENT_ENEMY_KIND ; if kind <0 then random
bpl store_enemy_kind
jsr random_number
and #$38
jmp store_enemy_kind
store_enemy_kind:
sta (ENEMY_PL),Y
; determine enemy _x
; if < 0, make random between 2->34
lda CURRENT_INIT_X
bpl store_init_x
jsr random_number
and #$1f
clc
adc #$2
asl
store_init_x:
iny ; X
sta (ENEMY_PL),Y
; enemy_y is always 0 by default
iny ; Y
lda #$0
sta (ENEMY_PL),Y
lda #$0
iny
sta (ENEMY_PL),Y ; xadd
iny
sta (ENEMY_PL),Y ; yadd
lda #$2
iny
sta (ENEMY_PL),Y ; xmin
iny
lda #$24
sta (ENEMY_PL),Y ; ymin
dey ; xmin
dey ; yadd
dey ; xadd
;===========================================
; Enemy specific inits
lda ENEMY_TYPE
beq enemy_type_0
cmp #$1
beq enemy_type_1
jmp enemy_type_2
enemy_type_0:
enemy_type_1:
;================================
; ENEMY TYPE 0 and 1
; diagonal, no wait
; movement proportional to level
lda LEVEL ; xadd = level
sta (ENEMY_PL),Y
iny
lsr A
ora #$1
sta (ENEMY_PL),Y ; yadd = level/2
jmp move_enemies
enemy_type_2:
;=====================
; Enemy Type 2
; just a place-holder
; waits for enemies to die then moves on to 3
cmp #$2
bne enemy_type_3
jmp move_enemies
enemy_type_3:
cmp #$3
bne enemy_type_4
;======================
; Enemy type 3
lda #$1
sta (ENEMY_PL),Y ; xadd=1
iny
lda LEVEL
sta (ENEMY_PL),Y ; yadd=level
jmp move_enemies
enemy_type_4:
cmp #$4
bne enemy_type_5
;=========================
; Enemy Type 4
; Horizontal, then fall
lda #$2
sta (ENEMY_PL),Y ; xadd = 2
iny
jsr random_number
ora #$80 ; set negative
sta (ENEMY_PL),y ; yadd = -(random%128)
; this means bop back and forth a random
; time, then drop
jmp move_enemies
enemy_type_5:
cmp #$5
bne enemy_type_6
;========================
; Enemy Type 5
; "wiggle"
lda #$1
sta (ENEMY_PL),y ; xadd=1
iny
lda LEVEL
sta (ENEMY_PL),y ; yadd=2
iny
jsr random_number
and #$0f
clc
adc #$2
sta (ENEMY_PL),y ; xmin=(rand%16)+2
dey ; yadd
dey ; xadd
dey ; y
dey ;x
asl A
sta (ENEMY_PL),y
iny ;y
iny ; xadd
iny ; yadd
iny ; xmin
jsr random_number
and #$0f
clc
adc (ENEMY_PL),Y
adc #$02
iny
sta (ENEMY_PL),Y ; xmax = xmin+(rand%16)+2
jmp move_enemies
enemy_type_6:
cmp #$6
beq enemy_type_7
cmp #$7
beq enemy_type_7
jmp enemy_type_8
enemy_type_7:
;=====================
; Enemy Types 6+7
; "Rain"
jsr random_number
and #6
bne no_use_own_x
dey ; y
dey ; x
lda SHIPX
cmp #$2
bpl shipx_ok
lda #$2 ; stupid bug where gets stuck is < xmin
shipx_ok:
asl A
sta (ENEMY_PL),Y ; one-in-four chance we use shipx as X
iny ; y
iny ; xadd
no_use_own_x:
lda #$0
sta (ENEMY_PL),Y ; xadd=0
iny
lda #$1
sta (ENEMY_PL),Y ; yadd = 1
jmp move_enemies
enemy_type_8:
enemy_type_9:
;======================
; Things flung by boss
dey ; y
dey ; x
lda BOSS_X
clc
adc #$5
asl A
sta (ENEMY_PL),Y ; enemy_x=boss_x+5
iny
lda #$3
asl A
asl A
sta (ENEMY_PL),Y ; enemy_y=3
iny
lda #$0
sta (ENEMY_PL),Y ; xadd=0
iny
lda #$2
sta (ENEMY_PL),Y ; yadd=2
move_enemies:
;==============================================
; Move Enemies! (first thing, if no new added)
;==============================================
ldy #$0 ; point to enemies[0]
handle_enemies:
tya
pha ; store y on stack
lda (ENEMY_PL),Y ; get enemy[y].out
bne load_enemy_zero_page ; if enemy.out then we are good
jmp skip_to_next_enemy ; enemy is not out, so skip to next
;==========================================
; load this enemy stuff into zero page for
; easier access
;==========================================
load_enemy_zero_page:
ldx #ENEMY_EXPLODING
load_to_zero_page:
iny ; point to exploding
lda (ENEMY_PL),Y
sta 0,X ; store to zero page
inx
cpx #(ENEMY_XMAX+1) ; see if reached end
bne load_to_zero_page ; if not keep copying
;================================
; skip all movement and collision
; if exploding
;================================
lda ENEMY_EXPLODING
beq move_enemy_x
jmp draw_enemy
;================================
; Start the enemy movement engine
;================================
;========
; Move X
;========
move_enemy_x:
clc
lda ENEMY_X ; X
adc ENEMY_XADD ; x+=xadd
sta ENEMY_X
lsr A
cmp ENEMY_XMIN ; are we less than xmin?
bmi switch_dir_enemy_x ; if so, switch direction
cmp ENEMY_XMAX ; are we greater than xmax?
bpl switch_dir_enemy_x ; if so, switch direction
jmp move_enemy_y
switch_dir_enemy_x:
; switch X direction
lda #$0 ; load zero
sec
sbc ENEMY_XADD ; 0 - ENEMY_XADD
sta ENEMY_XADD ; store it back out, negated
jmp move_enemy_x ; re-add it in
;========
; Move Y
;========
move_enemy_y:
lda #$0 ; load in zero
cmp ENEMY_YADD ; compare to YADD
bmi no_y_special_case ; if minus, we have special case
inc ENEMY_YADD
bne done_enemy_y
lda #$0
sta ENEMY_XADD
lda #$2
sta ENEMY_YADD
; increment y
; is it > 0?
; if not keep going
; if so, yadd=level*2
jmp done_enemy_y
no_y_special_case:
clc
lda ENEMY_Y ; get Y
adc ENEMY_YADD ; y+=yadd
sta ENEMY_Y ; store back out
lsr A
lsr A
cmp #$12 ; is y<=12?
bmi done_enemy_y ; if so no need to do anything
beq done_enemy_y
; off screen
pla ; pop saved Y off stack
tay
pha ; push y back on stack
lda #$0
sta (ENEMY_PL),Y ; set enemy[i].out=0
dec TOTAL_ENEMIES_OUT
lda BONUS_FLAGS
and #<(~PERFECT_KILLS)
sta BONUS_FLAGS
jmp skip_to_next_enemy ; skip to next enemy
done_enemy_y:
;===============
; Done Movement
;===============
;======================
; Check for Collisions
;======================
;==================================
; Check ENEMY <> MISSILE collision
;==================================
check_enemy_missile_collision:
ldy #$0
sty YSAV
check_missile_loop:
lda (MISSILE_PL),Y
beq missile_not_out
iny ; point to missile.x
lda (MISSILE_PL),Y ; load missile.x
sta COL_X1
sta COL_X2
lda ENEMY_X
lsr A
sta COL_X3
clc
adc #3
sta COL_X4
jsr check_inside
bcc missile_done
x_in_range:
iny
lda (MISSILE_PL),Y ; load missile.y
sta COL_X3
clc
adc #2
sta COL_X4
lda ENEMY_Y
lsr A
lsr A
sta COL_X1
clc
adc #1
sta COL_X2
jsr check_inside
bcc missile_done
horrible_explosion:
; clear missile
ldy YSAV
lda #$0
sta (MISSILE_PL),Y
; clear enemy
lda #$1
sta ENEMY_EXPLODING
lda #$40
sta ENEMY_KIND
jsr inc_score
jmp draw_enemy
missile_done:
missile_not_out:
ldy YSAV
iny
iny
iny
sty YSAV
cpy #(NUM_MISSILES*3)
bne check_missile_loop
;=================================
; Done missile <> enemy collision
;=================================
;====================================
; check for ship <-> enemy collision
;====================================
lda SHIPX
sta COL_X3
clc
adc #8
sta COL_X4 ; big check is shipx - shipx+8
lda ENEMY_X
lsr A
sta COL_X1
clc
adc #2
sta COL_X2 ; small check enemy_x - enemy_x+2
jsr check_inside ; check if overlap
bcc draw_enemy ; if not, move ahead
lda #16
sta COL_X3
lda #18
sta COL_X4 ; big check is 16 - 18
lda ENEMY_Y
lsr A
lsr A
sta COL_X1
clc
adc #$1
sta COL_X2 ; little check is enemy_y - enemy_y+1
jsr check_inside ; check if overlap
bcc draw_enemy ; if not, move ahead
; make the enemy explode
lda #$1
sta ENEMY_EXPLODING
lda #$40
sta ENEMY_KIND
dec SHIELDS
jsr update_shields ; move shields down
lda #<(~PERFECT_SHIELDS) ; (~PERFECT_SHIELDS)
and BONUS_FLAGS ; remove perfect shield bonus
sta BONUS_FLAGS
;=====================================
; Done ship <> enemy collision detect
;=====================================
draw_enemy:
; See if the enemy is currently exploding
; if so, do explosion stuff
check_enemy_explode:
lda ENEMY_EXPLODING ; load enemy[i].exploding
beq not_exploding ; if 0 then not exploding
handle_exploding:
jsr click ; make some noise
clc
lda ENEMY_KIND ; move to next step in explosion
adc #$4
sta ENEMY_KIND
cmp #$58 ; have we cycles through explosion?
bne draw_enemy_sprite ; if not, we are still exploding
dec TOTAL_ENEMIES_OUT ; total_enemies_out--
pla
tay ; load y
pha
lda #$0 ; enemy[i].out=0
sta (ENEMY_PL),Y
jmp skip_to_next_enemy
; point to enemies_x
; goto enemies_xy
not_exploding:
draw_enemy_sprite:
; point to proper sprite
lda #>enemy_sprite0 ; point to the missile sprite
sta STRINGH
lda #<enemy_sprite0
sta STRINGL
lda ENEMY_KIND ; get kind
and #$F8
clc
adc STRINGL
sta STRINGL
lda #0
adc STRINGH
sta STRINGH
enemies_xy:
lda ENEMY_X ; get X
lsr A
sta CH
lda ENEMY_Y ; load it
lsr A
lsr A
sta CV
jsr blit ; blit the missile sprite
save_zp_enemy_back:
; save zero page copy back to RAM
ldx #ENEMY_EXPLODING
pla
tay ; restore y pointer
pha
save_from_zero_page:
iny ; point to exploding
lda 0,X
sta (ENEMY_PL),Y
inx
cpx #(ENEMY_XMAX+1) ; see if reached end
bne save_from_zero_page ; if not keep copying
skip_to_next_enemy:
pla ; get saved value of Y
clc
adc #$9 ; add 9 to point to next
tay
cpy #NUM_ENEMIES*9 ; have we looped through them all?
beq draw_the_boss ; if not, loop
jmp handle_enemies
;===================================================
;===================================================
; BOSS STUFF
draw_the_boss:
;=======================
; if enemy_type==9
; we have a boss out
lda ENEMY_TYPE
cmp #$9
beq check_boss_exploding
jmp done_with_boss
check_boss_exploding:
;================================
; if not exploding, draw the boss
lda BOSS_EXPLODING
bne skip_draw_boss
lda #$0 ; boss_y=0
sta CV
lda BOSS_X
sta CH ; boss_x
lda #>boss_sprite
sta STRINGH
lda #<boss_sprite
sta STRINGL
jsr blit ; blit the ship sprite
skip_draw_boss:
;================================
; Draw Smoke
lda BOSS_SMOKE
beq skip_draw_smoke
; point to proper sprite
lda #>smoke_sprite0 ; point to the missile sprite
sta STRINGH
lda #<smoke_sprite0
sta STRINGL
lda BOSS_SMOKE ; get kind
and #$fc ; mask off bottom 2 bits
clc
adc STRINGL
sta STRINGL
lda #0
adc STRINGH
sta STRINGH
lda BOSS_X ; get X
clc
adc #$5
sta CH
lda #$3 ; load it
sta CV
jsr blit ; blit the missile sprite
lda BOSS_X
clc
adc #$5
sec
sbc BOSS_XADD
sta CH
lda #$4
sta CV
jsr blit
dec BOSS_SMOKE
skip_draw_smoke:
;======================
; BOSS Laser Shoot
lda BOSS_SHOOTING
beq skip_boss_shooting
dec BOSS_SHOOTING
jsr click
ldx #$0
boss_shoot_loop:
txa
pha
; point to proper sprite
lda #>laser_sprite0 ; point to the missile sprite
sta STRINGH
lda #<laser_sprite0
sta STRINGL
lda BOSS_SHOOTING ; get shooting
and #$1
beq got_right_laser
clc
lda #$5
adc STRINGL
sta STRINGL
lda #0
adc STRINGH
sta STRINGH
got_right_laser:
lda BOSS_X
sta CH
txa
asl
clc
adc #$3
sta CV
jsr blit ; blit the laser sprite
lda BOSS_X
clc
adc #$C
sta CH
pla
pha
asl
clc
adc #$3
sta CV
jsr blit
pla
tax
inx
cpx #$8
bne boss_shoot_loop
jsr click
skip_boss_shooting:
;=============================
; boss is dead
lda BOSS_EXPLODING
beq boss_is_not_exploding
lda #$1
sta BOSS_WAITING
ldx #$20
big_explosion:
txa
pha
; point to proper sprite
lda #>smoke_sprite0 ; point to the missile sprite
sta STRINGH
lda #<smoke_sprite0
sta STRINGL
lda BOSS_EXPLODING ; get kind
lsr A
lsr A
and #$0c ; mask off bottom 2 bits
clc
adc STRINGL
sta STRINGL
lda #0
adc STRINGH
sta STRINGH
jsr random_number
and #$03
sta CV
jsr random_number
and #$07
clc
adc BOSS_X
sta CH
jsr blit ; blit the missile sprite
pla
tax
dex
bne big_explosion
jsr click
dec BOSS_EXPLODING
bne not_dead_yet
lda LEVEL ; only show ending after level 1
cmp #$1
bne no_ending
jmp do_ending
no_ending:
inc SCOREH ; add 100 to score
jsr print_score
lda LEVEL
cmp #$7 ; level can't be higher than 7
beq start_new_level
inc LEVEL
inc SHIELDS
inc SHIELDS
lda #$A
cmp SHIELDS
bpl start_new_level
sta SHIELDS
start_new_level:
jsr update_shields
jmp new_level
not_dead_yet:
jmp move_boss
boss_is_not_exploding:
dec BOSS_COUNT
bne move_boss
;=========================================
; Toggle boss waiting state if count is up
lda BOSS_WAITING
beq make_boss_wait
stop_boss_waiting:
lda #$0
sta BOSS_WAITING ; boss_waiting=0
jsr random_number
sta BOSS_COUNT ; boss_count=rand%256
jmp move_boss
make_boss_wait:
lda #$1
sta BOSS_WAITING
jsr random_number
and #$01f
clc
adc #$30
sta BOSS_COUNT
lda #$20
sta BOSS_SHOOTING
move_boss:
lda BOSS_WAITING
bne laser_collision
lda BOSS_X
clc
adc BOSS_XADD
sta BOSS_X
cmp #26
bpl boss_reverse
boss_under:
cmp #$0
bpl laser_collision
boss_reverse:
lda #$0 ; load zero
sec
sbc BOSS_XADD ; 0 - ENEMY_XADD
sta BOSS_XADD ; store it back out, negated
jmp move_boss ; re-add it in
laser_collision:
;================================
; Collision detection for lasers
;
lda BOSS_SHOOTING
beq done_with_boss
left_laser:
lda BOSS_X
sta COL_X1
sta COL_X2
lda SHIPX
sta COL_X3
lda #$6
clc
adc SHIPX
sta COL_X4
jsr check_inside
bcs laser_hit
right_laser:
lda BOSS_X
clc
adc #$C
sta COL_X1
sta COL_X2
jsr check_inside
bcc done_with_boss
laser_hit:
lda BOSS_SHOOTING
and #$3 ; only take damage 1/8 the time
bne done_with_boss
dec SHIELDS
jsr update_shields
done_with_boss:
ldy #$0 ; point to missile[0]
move_missiles:
lda (MISSILE_PL),Y ; get missile[y]
beq loop_move_missiles ; if missile.out==0 skip
iny ; move to missile.y
iny
clc ; clear carry
lda (MISSILE_PL),Y ; get missile.y
adc #$FF ; move up (subtract 1)
sta (MISSILE_PL),Y ; store missile.y
bpl missile_collision_detection ; if not off screen, contine
dey ; back up to missile.out
dey
lda #$0
sta (MISSILE_PL),Y ; set missile.out=0
lda #<(~PERFECT_AIM) ; shot missed!
and BONUS_FLAGS ; clear perfect shot flag
sta BONUS_FLAGS
jmp loop_move_missiles ; continue
missile_collision_detection:
check_missile_boss:
lda ENEMY_TYPE
cmp #$9
bne done_missile_collision
dey ; missile x
lda (MISSILE_PL),Y
iny ; fix y
sta COL_X1
sta COL_X2
lda BOSS_X
sta COL_X3
clc
adc #$0d
sta COL_X4
jsr check_inside
bcc done_missile_collision
check_boss_y:
lda (MISSILE_PL),Y
sta COL_X1
clc
adc #$2
sta COL_X2
lda #$0
sta COL_X3
lda #$3
sta COL_X4
jsr check_inside
bcc done_missile_collision
hit_the_boss:
dey
dey
lda #$0
sta (MISSILE_PL),Y ; missile_out=0
lda #$B
sta BOSS_SMOKE
dec BOSS_HITS
bne loop_move_missiles
lda #$2F
sta BOSS_EXPLODING
lda #$0
sta BOSS_SHOOTING
jmp loop_move_missiles
done_missile_collision:
jmp loop_move_at_y
loop_move_missiles:
iny
iny
loop_move_at_y:
iny
cpy #NUM_MISSILES*3 ; have we checked all missiles?
bne move_missiles ; if not, loop
done_move_missiles:
ldy #$0 ; point to missiles[0]
draw_missiles:
lda (MISSILE_PL),Y ; get missile[y]
beq loop_draw_missiles ; if missile.out==0 skip
iny ; point to missile.x
lda (MISSILE_PL),Y ; load it
sta CH
iny ; point to missile.y
lda (MISSILE_PL),Y ; load it
sta CV
sty YSAV1 ; save Y
lda #>missile_sprite ; point to the missile sprite
sta STRINGH
lda #<missile_sprite
sta STRINGL
jsr blit ; blit the missile sprite
ldy YSAV1 ; restore Y
jmp loop_draw_missiles_noadd
loop_draw_missiles:
iny
iny
loop_draw_missiles_noadd:
iny
cpy #NUM_MISSILES*3 ; have we looped through them all?
bne draw_missiles ; if not, loop
game_read_keyboard:
jsr get_key
lda LASTKEY
bne game_q
jmp move_ship ; if no keypressed, move the ship
game_q:
cmp #'Q'
bne game_j
;; call verify_quit
jmp done_game
;; jmp set_pause_flag
game_j:
cmp #'J'
bne game_k
lda SHIPXADD ; load xadd
beq game_j_sub
bpl game_j_0 ; is switch dir, then 0 it
game_j_sub:
dec SHIPXADD ; else, dec XADD
jmp move_ship
game_j_0:
lda #$0
sta SHIPXADD
jmp move_ship
game_k:
cmp #'K'
bne game_c
lda SHIPXADD ; load xadd
bmi game_j_0 ; if we are switching dirs, set to zero
inc SHIPXADD ; else xadd++
jmp move_ship
game_c:
cmp #'C'
bne game_p
lda PADDLE_STATUS
eor #$80
sta PADDLE_STATUS
jmp move_ship
game_p:
cmp #'P'
bne game_s
bit KEYRESET
jsr wait_until_keypressed
jmp move_ship
game_s:
cmp #'S'
bne game_h
lda SOUND_ON
eor #$ff
sta SOUND_ON
jmp move_ship
game_h:
cmp #'H'
bne game_space
jsr do_help
jsr set_page0_gr
game_space:
cmp #' '+128 ; +128 because of get_key 'feature'
bne game_unknown
ldy #$0 ; point to missile[y]
fire_missiles:
lda (MISSILE_PL),Y ; get missile[y].out
bne end_fire_loop ; if not out, skip ahead
lda #$1 ; set missile[y].out=1
sta (MISSILE_PL),Y
iny ; point to missile[y].x
lda #$3
clc
adc SHIPX ; missile[y].x=shipx+3
sta (MISSILE_PL),Y
iny ; point to missile[y].y
lda #$10 ; set to 16
sta (MISSILE_PL),Y
jmp done_fire_missiles
end_fire_loop:
iny
iny
iny
cpy #NUM_MISSILES*3 ; see if we have more missiles
bne fire_missiles ; if so, loop
done_fire_missiles:
jmp move_ship
game_unknown:
move_ship:
clc ; Clear carry
lda SHIPX ; load ship_x
adc SHIPXADD ; ship_x+=xadd
sta SHIPX ; store it back
check_x_under:
bpl check_x_over ; if positive, keep going
lda #$0 ; we were below zero
sta SHIPX ; so shipx=0
sta SHIPXADD ; xadd=0
jmp blit_ship ; go to blit
check_x_over:
cmp #$21 ; are we over 33?
bmi blit_ship ; if not, blit ship
lda #$0 ; shipxadd=0
sta SHIPXADD
lda #$21 ; shipx=33
sta SHIPX
blit_ship:
lda #$10 ; shipy=16
sta CV
lda SHIPX
sta CH ; load shipx
lda #>ship_sprite
sta STRINGH
lda #<ship_sprite
sta STRINGL
jsr blit ; blit the ship sprite
jsr clear_bottom
lda #>shields_string
sta STRINGH
lda #<shields_string
sta STRINGL
clc
lda #$2
adc GR_PAGE
sta BASH
lda #$50
sta BASL
ldy #$0
shield_print_loop:
;=============================
; CHECK TO SEE IF GAME IS OVER
;=============================
lda SHIELDS
bmi done_game
lda (STRINGL),Y
sta (BASL),Y
iny
cpy #$1E ; string is 30 long
bne shield_print_loop
inc BASH ; move to line 23
lda #>score_string
sta STRINGH
lda #<score_string
sta STRINGL
ldy #$0
score_print_loop:
lda (STRINGL),Y
sta (BASL),Y
iny
cpy #$26 ; string is 38 long
bne score_print_loop
clc
lda #$80
adc BASL
sta BASL ; move to line 24
;OPTIMIZE
lda #>level_string
sta STRINGH
lda #<level_string
sta STRINGL
ldy #$0
level_print_loop:
lda (STRINGL),Y
sta (BASL),Y
iny
cpy #$C ; string is 12 long
bne level_print_loop
;==========
; Flip Pages
lda #$4
bit GR_PAGE
bne gr_page_1
gr_page_0:
bit PAGE1 ; switch to page 0
lda #$4
jmp write_out_gr_page
gr_page_1:
bit PAGE0 ; switch to page 1
lda #$8
write_out_gr_page:
sta GR_PAGE
; lda #$8B ; 107 = 30080us = 30ms
lda #$65
jsr WAIT
jmp main_game_loop
done_game:
ldx #30
jsr wait_X_100msec ; pause for 3 seconds
bit KEYRESET ; clear keyboard
jsr set_page0_text
jsr HOME
lda #>game_over_string
sta STRINGH
lda #<game_over_string
sta STRINGL
jsr print_text_xy
jsr wait_until_keypressed
jsr set_page0_text
see_if_new_hi_score:
sed
lda SCOREH
cmp HISCORE_H
cld
beq too_close
bpl new_high
jmp do_hi_score
too_close:
sed
lda SCOREL
cmp HISCORE_L
cld
bpl new_high
jmp do_hi_score
new_high:
; Actually set the high score
lda SCOREL
sta HISCORE_L
lda SCOREH
sta HISCORE_H
lda #>(score_string+31)
sta STRINGH
lda #<(score_string+31)
sta STRINGL
jsr print_high_score
jsr HOME
; print new high score message
lda #>new_high_score_string
sta STRINGH
lda #<new_high_score_string
sta STRINGL
jsr print_text_xy
jsr print_text_xy
; set high score to AAA
lda #$C1
sta HISCORE_1
sta HISCORE_2
sta HISCORE_3
; load initials address
lda #$06
sta STRINGH
lda #$12
sta STRINGL
; load pointer address
lda #$06
sta BASH
lda #$92
sta BASL
ldy #$0
ldx #$0
initials_loop:
; erase old pointer
ldy #$0
lda #$A0
sta (BASL),Y
iny
sta (BASL),Y
iny
sta (BASL),Y
txa
tay
; draw pointer
lda #$DE ; '^'
sta (BASL),Y
ldy #$0
; draw initials
lda HISCORE_1
sta (STRINGL),Y
iny
lda HISCORE_2
sta (STRINGL),Y
iny
lda HISCORE_3
sta (STRINGL),Y
in_key:
jsr wait_until_keypressed
lda LASTKEY
beq in_key
cmp #$D
beq do_hi_score
cmp #'I'
bne in_down
inc HISCORE_1,X
in_down:
cmp #'M'
bne in_left
dec HISCORE_1,X
in_left:
cmp #'K'
bne in_right
inx
in_right:
cmp #'J'
bne fix_limits
dex
fix_limits:
; Make sure X is between 0 and 2
cpx #$0
bpl x_high
ldx #$2
x_high:
cpx #$3
bmi x_lo
ldx #$0
x_lo:
jmp initials_loop
do_hi_score:
jsr HOME
lda #>high_score_string
sta STRINGH
lda #<high_score_string
sta STRINGL
jsr print_text_xy
; go to screen co-ords 15x12
lda #$06
sta STRINGH
lda #$37
sta STRINGL
; put the initials up
lda HISCORE_1
ldy #$0
sta (STRINGL),Y
lda HISCORE_2
iny
sta (STRINGL),Y
lda HISCORE_3
iny
sta (STRINGL),Y
; Print score to screen
;
lda #$06
sta STRINGH
lda #$3D
sta STRINGL
jsr print_high_score
jsr wait_until_keypressed
jmp opener
;=========================================================
; CLICK
;=========================================================
click:
lda SOUND_ON
beq no_click
bit SPEAKER
no_click:
rts
;==========================================================
; Wait X 100 msec
;==========================================================
;
wait_X_100msec:
lda #$86 ; constant ~ 100msec
jsr WAIT
dex
bne wait_X_100msec
rts
;==========================================================
; Update Shields
;==========================================================
;
update_shields:
lda #>shields_string
sta STRINGH
lda #<shields_string
sta STRINGL
ldy #0
lda SHIELDS
bne normal_shields
flash_shields:
lda #$7f
and (STRINGL),Y
sta (STRINGL),Y
iny
cpy #$7
bne flash_shields
jmp shields_line
normal_shields:
lda #$80
ora (STRINGL),Y
sta (STRINGL),Y
iny
cpy #$7
bne normal_shields
shields_line:
ldy #$A
ldx #$0
shield_line_loop:
cpx SHIELDS
bmi shield_box
lda #'_'+128
jmp shield_char
shield_box:
lda #' '
shield_char:
sta (STRINGL),Y
iny
iny
inx
cpx #$A
bne shield_line_loop
rts
;==========================================================
; Random Number Generator
;==========================================================
; from dlyons@Apple.COM (David A Lyons)
; posting to comps.sys.apple2 24 November 1992
; algorithm from Pop Science ~1980s
; when seeded with non-zero, will generate all 255 values
; before repeating
random_number:
lda RANDOM_SEED
bne random_not_zero
lda #$D
random_not_zero:
asl A
bcc random_num_done
eor #$87
random_num_done:
sta RANDOM_SEED
rts
;==========================================================
; Wait until keypressed
;==========================================================
;
wait_until_keypressed:
inc RANDOM_SEED ; seed random num counter
lda KEYPRESS ; check if keypressed
bpl wait_until_keypressed ; if not, loop
jmp figure_out_key
;==========================================================
; Get Key
;==========================================================
;
get_key:
check_paddle_button:
; check for paddle button
bit PADDLE_BUTTON0
bpl no_button
lda #' '+128
jmp save_key
no_button:
lda KEYPRESS
bpl no_key
figure_out_key:
cmp #' '+128 ; the mask destroys space
beq save_key ; so handle it specially
and #$5f ; mask, to make upper-case
check_right_arrow:
cmp #$15
bne check_left_arrow
lda #'K'
check_left_arrow:
cmp #$08
bne check_up_arrow
lda #'J'
check_up_arrow:
cmp #$0B
bne check_down_arrow
lda #'I'
check_down_arrow:
cmp #$0A
bne check_escape
lda #'M'
check_escape:
cmp #$1B
bne save_key
lda #'Q'
jmp save_key
no_key:
bit PADDLE_STATUS
bpl no_key_store
; check for paddle action
; code from http://web.pdx.edu/~heiss/technotes/aiie/tn.aiie.06.html
inc PADDLE_STATUS
lda PADDLE_STATUS
and #$03
beq check_paddles
jmp no_key_store
check_paddles:
lda PADDLE_STATUS
and #$80
sta PADDLE_STATUS
ldx #$0
LDA PTRIG ;TRIGGER PADDLES
LDY #0 ;INIT COUNTER
NOP ;COMPENSATE FOR 1ST COUNT
NOP
PREAD2: LDA PADDL0,X ;COUNT EVERY 11 uSEC.
BPL RTS2D ;BRANCH WHEN TIMED OUT
INY ;INCREMENT COUNTER
BNE PREAD2 ;CONTINUE COUNTING
DEY ;COUNTER OVERFLOWED
RTS2D: ;RETURN W/VALUE 0-255
cpy #96
bmi paddle_left
cpy #160
bmi no_key_store
lda #'K'
jmp save_key
paddle_left:
lda #'J'
jmp save_key
no_key_store:
lda #0 ; no key, so save a zero
save_key:
sta LASTKEY ; save the key to our buffer
bit KEYRESET ; clear the keyboard buffer
rts
;==========================================================
; score_plus_50
;==========================================================
;
score_plus_50:
sed ; enter decimal mode
clc
lda SCOREL
adc #$50
sta SCOREL ; score+=50
lda SCOREH
adc #$0
sta SCOREH ; carry into high byte if needed
cld ; leave decimal mode
jmp print_score
;==========================================================
; inc_score
;==========================================================
;
inc_score:
sed ; enter decimal mode
clc
lda SCOREL
adc #$5
sta SCOREL ; score+=5
lda SCOREH
adc #$0
sta SCOREH ; carry into high byte if needed
cld ; leave decimal mode
lda SCOREL ; if score /100 =0 then inc shields
bne print_score
lda SHIELDS
cmp #$0A ; don't raise shields higher than 10
bpl print_score
inc SHIELDS
jsr update_shields
print_score:
lda #>(score_string+7)
sta STRINGH
lda #<(score_string+7)
sta STRINGL
tya
pha ; save Y on stack
ldy #$0
lda SCOREH
sta BCD_BYTEH
lda SCOREL
sta BCD_BYTE
jsr print_bcd_word
pla ; restore Y
tay
rts
;======================
; print high_score
;======================
; location to output to in STRINGH/STRINGL
print_high_score:
tya
pha ; save Y on stack
ldy #$0
lda HISCORE_H
sta BCD_BYTEH
lda HISCORE_L
sta BCD_BYTE
jsr print_bcd_word
pla ; restore Y
tay
rts
;==========================================================
; print_bcd_word
;==========================================================
; string to output in STRINGH/STRINGL
; byte to output in BCD_BYTE
print_bcd_word:
lda BCD_BYTEH
lsr A
lsr A
lsr A
lsr A
and #$f ; mask low nybble
bne to_ascii_thou
lda #$A0 ; load a space
jmp write_thousands
to_ascii_thou:
adc #$B0 ; covert to ascii
write_thousands:
sta (STRINGL),Y ; store output
iny
lda BCD_BYTEH
and #$f
bne to_ascii_hun
cmp BCD_BYTEH
bne to_ascii_hun
lda #$A0
jmp write_hundreds
to_ascii_hun:
adc #$B0
write_hundreds:
sta (STRINGL),Y
iny
print_bcd_byte:
lda BCD_BYTE
lsr A
lsr A
lsr A
lsr A
and #$f ; mask low nybble
bne to_ascii_tens ; if not zero, convert to ascii
cmp BCD_BYTEH
bne to_ascii_tens
lda #$A0
jmp write_tens
to_ascii_tens:
adc #$B0 ; covert to ascii
write_tens:
sta (STRINGL),Y ; store output
iny ; point one lower
lda BCD_BYTE
clc
and #$f
adc #$B0
sta (STRINGL),Y
rts
;==========================================================
; check inside
;==========================================================
; Simple collision detection. Have small line x1<->x2
; Want to see if it overlaps long line x3<---------->x4
; so:
; if ((x1>x3)&&(x1<x4)) || ((x2>x3) && (x2<x4)) inside
; else outside
check_inside:
lda COL_X1
cmp COL_X3
bmi check_higher
cmp COL_X4
bmi inside
check_higher:
lda COL_X2
cmp COL_X3
bmi outside
cmp COL_X4
bpl outside
inside:
sec
rts
outside:
clc
rts
;==========================================================
; set_page0_text
;==========================================================
;
set_page0_text:
bit PAGE0 ; set page0
bit TEXT ; set text mode
rts
;==========================================================
; set_page0_gr
;==========================================================
;
set_page0_gr:
lda #4
sta GR_PAGE
bit PAGE0 ; set page 0
bit LORES ; Lo-res graphics
bit TEXTGR ; mixed gr/text mode
bit GR ; set graphics
rts
;==========================================================
; clear bottom
;==========================================================
;
clear_bottom:
lda #$2
clc
adc GR_PAGE
sta BASH ; point to line 21
lda #$50
sta BASL
ldx #$00 ; line counter
bottom_y:
lda #' '+128 ; want to put a space
ldy #$0 ; column count
bottom_loop:
sta (BASL),Y ; store a space
iny ; incrememnt
cpy #$28 ; are we > 40?
bmi bottom_loop ; if not, loop
lda #$80 ; go to next line [they are $80 apart]
clc
adc BASL ; increment base
sta BASL ; store it out
lda #$0 ; load 0 into A
adc BASH ; carry into top byte if need be
sta BASH ; and store out
inx ; increment line count
cpx #$4 ; have we done 4
bcc bottom_y ; if not, loop
rts
;==========================================================
; print X strings
;==========================================================
;
;
print_x_strings:
stx TEMP
jsr print_text_xy
ldx TEMP
dex
bne print_x_strings
rts
;==========================================================
; Print text x,y
;==========================================================
; x=ch y=cv
; string=string_addr
print_text_xy:
ldy #$0 ; clear IY
lda (STRINGL),Y ; load x from memory
sta CH ; store to CH
iny ; point to next value
lda (STRINGL),Y ; load y from memory
sta CV ; store to CV
; point to beginning of string
clc
lda #$2
adc STRINGL
sta STRINGL
lda #$0
adc STRINGH
sta STRINGH
print_text:
ldy #$0
lda CV
jsr BASCALC ; get the address of y in BASH:BASL
clc ; clear the carry
lda BASL ; load BASL
adc CH ; add x
sta BASL ; store BASL back out
output_loop:
lda (STRINGL),Y ; load char from string_addr+y
beq print_done ; if null terminated, done
sta (BASL),Y ; store to BASH:BASL
iny ; IY++
jmp output_loop ; loop
print_done:
iny
tya ; transfer y to accumulator
adc STRINGL ; add y and stringl
sta STRINGL ; and store it out
lda #$0 ; clear accumulator
adc STRINGH ; add with carry from prev stringh
sta STRINGH ; and save it
rts
;==========================================================
; clear_screen
;==========================================================
;
clear_screen:
ldx #$0
clear_0:
cpx #$0
bne clear_1
lda GR_PAGE
sta BASH
lda #$0
sta BASL
ldy #$78
jmp clear_it
clear_1:
cpx #$1
bne clear_2
lda #$80
sta BASL
ldy #$78
jmp clear_it
clear_2:
cpx #$2
bne clear_3
clc
lda #$1
adc GR_PAGE
sta BASH
ldy #$78
jmp clear_it
clear_3:
cpx #$3
bne clear_4
lda #$0
sta BASL
ldy #$78
jmp clear_it
clear_4:
cpx #$4
bne clear_5
clc
lda #$2
adc GR_PAGE
sta BASH
ldy #$50
jmp clear_it
clear_5:
cpx #$5
bne clear_6
lda #$80
sta BASL
ldy #$50
jmp clear_it
clear_6:
cpx #$6
bne clear_7
clc
lda #$3
adc GR_PAGE
sta BASH
ldy #$50
jmp clear_it
clear_7:
cpx #$7
bne clear_8
lda #$0
sta BASL
ldy #$50
jmp clear_it
clear_8:
rts
clear_it:
lda #$00
clear_loop:
dey
sta (BASL),Y
bne clear_loop
inx
jmp clear_0
;==========================================================
; show_stars
;==========================================================
;
show_stars:
lda #>star_field ; Load the star offsets
sta STRINGH ; array into
lda #<star_field ; STRINGH/STRINGL
sta STRINGL
lda BETWEEN_DELAY
and #$1
beq star_7
dec SCROLL ; decrement the scroll count
star_7:
ldy SCROLL ; load the scroll offset into Y
ldx #$8 ; We initially repeat 8 times
lda #$0 ; And have L of 0
sta TEMP ; This is stupid interlace stuff
sta COL_X1
star_4:
lda GR_PAGE ; we loop here, it resets
sta BASH ; to the proper page
star_0:
lda (STRINGL),Y ; get star offset
beq star_5
pha ; save for later
bmi light_star ; check color
lda #$5
jmp done_star
light_star:
lda #$A
done_star:
sta COL_X2 ; save color for later
pla ; restore offset
clc
and #$7f ; clear color flag
adc TEMP ; add in offset
sta BASL ; store as offset
tya ; save y on stack
pha ;
lda COL_X2
ldy #$0
sta (BASL),Y ; plot star
pla ; restore y from stack
tay
star_5:
iny ; increase y (offset pointer)
lda (STRINGL),Y ; get star offset
beq star_6
pha ; save for later
bmi light_star2 ; check color
lda #$50
jmp done_star2
light_star2:
lda #$A0
done_star2:
sta COL_X2 ; save color for later
pla ; restore offset
clc
and #$7f ; clear color flag
adc TEMP ; add in offset
sta BASL ; store as offset
tya ; save y on stack
pha ;
lda COL_X2
ldy #$0
sta (BASL),Y ; plot star
pla ; restore y from stack
tay
star_6:
iny ; increase y (offset pointer)
lda TEMP ; get offset
clc
adc #$80 ; increment to next line
sta TEMP ; and save back
bcc overflow_line ; if overflow, increase BASH
inc BASH
overflow_line:
dex ; repeat X times
bne star_0
inc COL_X1 ; check if first time through
lda #$1
cmp COL_X1
star_1:
bne star_2 ; reload for second 8 lines
ldx #$8
lda #$28
sta TEMP
jmp star_4
star_2: ; check if second time through
lda #$2
cmp COL_X1
bne star_3
ldx #$4 ; reload for another 4 lines
lda #$50
sta TEMP
jmp star_4
star_3:
rts
;==========================================================
; blit
;==========================================================
; x=ch y=cv
; string=addr of sprite
; only works on sprites < 256 bytes!!!
blit:
ldy #$00 ; clear IY
big_blit_loop:
lda CV ; Load y
jsr BASCALC ; we only do 40x20 so just use text mode
lda #$4
bit GR_PAGE
bne keep_blitting
clc
adc BASH
sta BASH
keep_blitting:
clc ; clear the carry
lda BASL ; load BASL
adc CH ; add x
sta BASL ; store BASL back out
bcc blit_loop ; if carry we overflowed
inc BASH ; add overflow into BASH
blit_loop:
lda (STRINGL),Y ; load char from string_addr+y
beq blit_x_done ; if null terminated, done this row
sty YSAV ; save IY
pha ; save run/color on stack
jsr SETCOL ; make low/high both col
; and save in in COLOR
pla ; restore run/color
lsr
lsr
lsr
lsr ; shift A right by 4 to get run-length
tax ; store in IX
ldy #$0 ; clear IY
lda COLOR ; load back COLOR
bne rle_loop ; if not zero, plot it
skip_color:
inc BASL ; increment pointer
dex ; count down IX
bne skip_color ; and skip ahead
jmp rle_loop_done ; done
rle_loop:
sta (BASL),Y ; store our color to the line
inc BASL ; increment output pointer
dex ; decrement count
bne rle_loop ; until done
rle_loop_done:
ldy YSAV ; restore SPRITE pointer
iny ; IY++
jmp blit_loop ; loop until row done
blit_x_done:
iny ; point past 0
lda (STRINGL),Y ; get next
beq blit_done ; if 0 as well, done with sprite
inc CV ; otherwise y++
jmp big_blit_loop ; and loop
blit_done:
rts
;; *********************
;; BSS
;; *********************
.bss
start_bss:
missile_0: .res NUM_MISSILES*3
enemy_0: .res NUM_ENEMIES*9
end_bss:
star_field: .res 255
zero_page_save: .res 56
;; *********************
;; DATA
;; *********************
.data
; SHIELDS: plus 22 spaces (30 total)
shields_string:
.byte $D3,$C8,$C9,$C5,$CC,$C4,$D3,$BA
.byte $A0,$A0,$A0,$A0,$A0,$A0,$A0,$A0,$A0,$A0
.byte $A0,$A0,$A0,$A0,$A0,$A0,$A0,$A0,$A0,$A0
.byte $A0,$A0
; SCORE and HISCORE: (38 Total)
score_string:
.byte $D3,$C3,$CF,$D2,$C5,$BA,$A0,$A0,$A0,$A0
.byte $A0
.byte $A0,$A0,$A0,$A0,$A0,$A0,$A0,$A0,$A0
.byte $C8,$C9,$D3,$C3,$CF,$D2,$C5,$BA,$A0,$A0,$A0,$A0
.byte $B5,$B0,$B0,$A0,$A0,$A0
; LEVEL:
level_string_xy:
.byte $0E,$0A
level_string:
.byte $CC,$C5,$D6,$C5,$CC,$BA,$A0,$A0,$A0,$A0,$B1,$A0,$00
; A VMW SOFTWARE PRODUCTION
vmw_string: .byte $07,$14
.byte $C1,$A0,$D6,$CD,$D7,$A0,$D3,$CF,$C6,$D4,$D7,$C1,$D2
.byte $C5,$A0,$D0,$D2,$CF,$C4,$D5,$C3,$D4,$C9,$CF,$CE,$00
; MERCILESS MARAUDING MALICIOUS MARKETERS
mercy_string: .byte $00,$14
.byte $CD,$C5,$D2,$C3,$C9,$CC,$C5,$D3,$D3,$A0,$CD,$C1,$D2
.byte $C1,$D5,$C4,$C9,$CE,$C7,$A0,$CD,$C1,$CC,$C9,$C3,$C9
.byte $CF,$D5,$D3,$A0,$CD,$C1,$D2,$CB,$C5,$D4,$C5,$D2,$D3
.byte $00
; GAME OVER
game_over_string:
.byte $0F,$0A
.byte $C7,$C1,$CD,$C5,$A0,$CF,$D6,$C5,$D2,$00
; NEW GAME
new_game_string: .byte $CE,$C5,$D7,$A0,$C7,$C1,$CD,$C5,$00
; ABOUT
about_string: .byte $C1,$C2,$CF,$D5,$D4,$00
; STORY
story_string: .byte $D3,$D4,$CF,$D2,$D9,$00
; HISCORE
hiscore_string: .byte $C8,$C9,$A0,$D3,$C3,$CF,$D2,$C5,$00
; QUIT
quit_string: .byte $D1,$D5,$C9,$D4,$00
; H FOR HELP
help_string: .byte $0,$14
.byte $C8,$A0,$C6,$CF,$D2,$A0,$C8,$C5,$CC,$D0,$00
; QUIT? ARE YOU SURE?
quit_conf_string: .byte $D1,$D5,$C9,$D4,$BF,$A0,$C1,$D2,$C5,$A0,$D9,$CF,$D5
.byte $A0,$D3,$D5,$D2,$C5,$BF,$00
; YES
yes_string: .byte $D9,$C5,$D3,$00
; NO
no_string: .byte $CE,$CF,$00
about_lines:
; TOM BOMBEM BY VINCE WEAVER (PIC ABOVE)
.byte $0,$14
.byte $D4,$CF,$CD,$A0,$C2,$CF,$CD,$C2,$C5,$CD
.byte $A0,$A0,$C2,$D9,$A0,$D6,$C9,$CE,$C3,$C5
.byte $A0,$D7,$C5,$C1,$D6,$C5,$D2,$A0,$A8,$D0
.byte $C9,$C3,$A0,$C1,$C2,$CF,$D6,$C5,$A9,$00
; BASED ON LINUX TB_ASM - APPLE II FOREVER
.byte $0,$16
.byte $C2,$C1,$D3,$C5,$C4,$A0,$CF,$CE,$A0,$CC
.byte $C9,$CE,$D5,$D8,$A0,$D4,$C2,$DF,$C1,$D3
.byte $CD,$A0,$AD,$A0,$C1,$D0,$D0,$CC,$C5,$A0
.byte $C9,$C9,$A0,$C6,$CF,$D2,$C5,$D6,$C5,$D2,$00
; VINCE@DEATER.NET
.byte $0,$16
.byte $D6,$C9,$CE,$C3,$C5,$C0,$C4,$C5,$C1,$D4
.byte $C5,$D2,$AE,$CE,$C5,$D4,$00
; HTTP://WWW.DEATER.NET/WEAVE/VMWPROD/TB1/
.byte $0,$17
.byte $C8,$D4,$D4,$D0,$BA,$AF,$AF,$D7,$D7,$D7
.byte $AE,$C4,$C5,$C1,$D4,$C5,$D2,$AE,$CE,$C5
.byte $D4,$AF,$D7,$C5,$C1,$D6,$C5,$AF,$D6,$CD
.byte $D7,$D0,$D2,$CF,$C4,$AF,$D4,$C2,$B1,$AF,$00
ending_lines:
; *** MESSAGE FROM EARTH ***
.byte $7,$14
.byte $AA,$AA,$AA,$A0,$CD,$C5,$D3,$D3,$C1,$C7
.byte $C5,$A0,$C6,$D2,$CF,$CD,$A0,$C5,$C1,$D2
.byte $D4,$C8,$A0,$AA,$AA,$AA,$00
; CONGRATULATIONS TOM, YOU'VE DESTROYED
.byte $0,$16
.byte $C3,$CF,$CE,$C7,$D2,$C1,$D4,$D5,$CC,$C1
.byte $D4,$C9,$CF,$CE,$D3,$A0,$D4,$CF,$CD,$AC
.byte $A0,$D9,$CF,$D5,$A7,$D6,$C5,$A0,$C4,$C5
.byte $D3,$D4,$D2,$CF,$D9,$C5,$C4,$00
; THE MENACE!
.byte $2,$17
.byte $D4,$C8,$C5,$A0,$CD,$C5,$CE,$C1,$C3,$C5
.byte $A1,$00
; BUT WAIT! SENSORS ARE DETECTING NEW
.byte $0,$14
.byte $C2,$D5,$D4,$A0,$D7,$C1,$C9,$D4,$A1,$A0
.byte $A0,$D3,$C5,$CE,$D3,$CF,$D2,$D3,$A0,$C1
.byte $D2,$C5,$A0,$C4,$C5,$D4,$C5,$C3,$D4,$C9
.byte $CE,$C7,$A0,$CE,$C5,$D7,$00
; INCOMING TARGETS.
.byte $2,$15
.byte $C9,$CE,$C3,$CF,$CD,$C9,$CE,$C7,$A0,$D4
.byte $C1,$D2,$C7,$C5,$D4,$D3,$AE,$00
; REMEMBER, YOU DON'T GET PAID UNTIL ALL
.byte $0,$16
.byte $D2,$C5,$CD,$C5,$CD,$C2,$C5,$D2,$AC,$A0
.byte $D9,$CF,$D5,$A0,$C4,$CF,$CE,$A7,$D4,$A0
.byte $C7,$C5,$D4,$A0,$D0,$C1,$C9,$C4,$A0,$D5
.byte $CE,$D4,$C9,$CC,$A0,$C1,$CC,$CC,$00
; ARE DESTROYED
.byte $2,$17
.byte $C1,$D2,$C5,$A0,$C4,$C5,$D3,$D4,$D2,$CF
.byte $D9,$C5,$C4,$00
susie_lines:
; PS. YOUR PET GUINEA PIG IS DOING FINE
.byte $0,$14
.byte $D0,$D3,$AE,$A0,$A0,$D9,$CF,$D5,$D2,$A0
.byte $D0,$C5,$D4,$A0,$C7,$D5,$C9,$CE,$C5,$C1
.byte $A0,$D0,$C9,$C7,$A0,$C9,$D3,$A0,$C4,$CF
.byte $C9,$CE,$C7,$A0,$C6,$C9,$CE,$C5,$00
tom_sigh:
; TOM: *SIGH*
.byte $0,$14
.byte $D4,$CF,$CD,$BA,$A0,$AA,$D3,$C9,$C7,$C8,$AA,$00
story_lines:
; IT IS THE YEAR 2025.
.byte $0,$14
.byte $C9,$D4,$A0,$C9,$D3,$A0,$D4,$C8,$C5,$A0
.byte $D9,$C5,$C1,$D2,$A0,$B2,$B0,$B2,$B5,$AE,$00
; ALL TELEMARKETERS AND UNSOLICITED
.byte $2,$15
.byte $C1,$CC,$CC,$A0,$D4,$C5,$CC,$C5,$CD,$C1
.byte $D2,$CB,$C5,$D4,$C5,$D2,$D3,$A0,$C1,$CE
.byte $C4,$A0,$D5,$CE,$D3,$CF,$CC,$C9,$C3,$C9
.byte $D4,$C5,$C4,$00
; BULK E-MAILERS HAVE BEEN EXILED
.byte $4,$16
.byte $C2,$D5,$CC,$CB,$A0,$C5,$AD,$CD,$C1,$C9
.byte $CC,$C5,$D2,$D3,$A0,$C8,$C1,$D6,$C5,$A0
.byte $C2,$C5,$C5,$CE,$A0,$C5,$D8,$C9,$CC,$C5
.byte $C4,$00
; TO PHOBOS.
.byte $6,$17
.byte $D4,$CF,$A0,$D0,$C8,$CF,$C2,$CF,$D3,$AE,$00
; RIGHT BEFORE BEING TRAPPED FOREVER
.byte $0,$14
.byte $D2,$C9,$C7,$C8,$D4,$A0,$C2,$C5,$C6,$CF
.byte $D2,$C5,$A0,$C2,$C5,$C9,$CE,$C7,$A0,$D4
.byte $D2,$C1,$D0,$D0,$C5,$C4,$A0,$C6,$CF,$D2
.byte $C5,$D6,$C5,$D2,$00
; THEY MANAGE TO LAUNCH ONE LAST
.byte $2,$15
.byte $D4,$C8,$C5,$D9,$A0,$CD,$C1,$CE,$C1,$C7
.byte $C5,$A0,$D4,$CF,$A0,$CC,$C1,$D5,$CE,$C3
.byte $C8,$A0,$CF,$CE,$C5,$A0,$CC,$C1,$D3,$D4,$00
; MARKETING DROID.
.byte $4,$16
.byte $CD,$C1,$D2,$CB,$C5,$D4,$C9,$CE,$C7,$A0
.byte $C4,$D2,$CF,$C9,$C4,$AE,$00
help_lines:
; TOM BOMBEM
.byte $A,$0
.byte $D4,$CF,$CD,$A0,$C2,$CF,$CD,$C2,$C5,$CD,$00
; BY
.byte $E,$1
.byte $C2,$D9,$00
; VINCE WEAVER
.byte $9,$2
.byte $D6,$C9,$CE,$C3,$C5,$A0,$D7,$C5,$C1,$D6
.byte $C5,$D2,$00
; KEY BINDINGS:
.byte $0,$4
.byte $CB,$C5,$D9,$A0,$C2,$C9,$CE,$C4,$C9,$CE
.byte $C7,$D3,$BA,$00
; UP OR 'I' : MOVE MENU UP
.byte $2,$6
.byte $D5,$D0,$A0,$CF,$D2,$A0,$A7,$C9,$A7,$A0
.byte $A0,$A0,$A0,$BA,$A0,$CD,$CF,$D6,$C5,$A0
.byte $CD,$C5,$CE,$D5,$A0,$D5,$D0,$00
; DOWN OR 'M' : MOVE MENU DOWN
.byte $2,$7
.byte $C4,$CF,$D7,$CE,$A0,$CF,$D2,$A0,$A7,$CD
.byte $A7,$A0,$A0,$BA,$A0,$CD,$CF,$D6,$C5,$A0
.byte $CD,$C5,$CE,$D5,$A0,$C4,$CF,$D7,$CE,$00
; ENTER : SELECTS CURRENT OPTION
.byte $2,$8
.byte $C5,$CE,$D4,$C5,$D2,$00
.byte $F,$8
.byte $BA,$A0,$D3,$C5,$CC,$C5,$C3,$D4,$D3,$A0
.byte $C3,$D5,$D2,$D2,$C5,$CE,$D4,$A0,$CF,$D0
.byte $D4,$C9,$CF,$CE,$00
; RIGHT OR 'K' : MOVES SHIP RIGHT
.byte $2,$A
.byte $D2,$C9,$C7,$C8,$D4,$A0,$CF,$D2,$A0,$A7
.byte $CB,$A7,$A0,$BA,$A0,$CD,$CF,$D6,$C5,$D3
.byte $A0,$D3,$C8,$C9,$D0,$A0,$D2,$C9,$C7,$C8
.byte $D4,$00
; LEFT OR 'J' : MOVES SHIP LEFT
.byte $2,$B
.byte $CC,$C5,$C6,$D4,$A0,$CF,$D2,$A0,$A7,$CA
.byte $A7,$A0,$A0,$BA,$A0,$CD,$CF,$D6,$C5,$D3
.byte $A0,$D3,$C8,$C9,$D0,$A0,$CC,$C5,$C6,$D4,$00
; SPACEBAR : SHOOTS
.byte $2,$C
.byte $D3,$D0,$C1,$C3,$C5,$C2,$C1,$D2,$A0,$A0
.byte $A0,$A0,$A0,$BA,$A0,$D3,$C8,$CF,$CF,$D4
.byte $D3,$00
; 'H' : DISPLAYS HELP
.byte $2,$E
.byte $A7,$C8,$A7,$00
.byte $F,$E
.byte $BA,$A0,$C4,$C9,$D3,$D0,$CC,$C1,$D9,$D3
.byte $A0,$C8,$C5,$CC,$D0,$00
; ESC OR 'Q' : QUITS
.byte $2,$F
.byte $C5,$D3,$C3,$A0,$CF,$D2,$A0,$A7,$D1,$A7
.byte $A0,$A0,$A0,$BA,$A0,$D1,$D5,$C9,$D4,$D3,$00
; 'P' : PAUSES
.byte $2,$10
.byte $A7,$D0,$A7,$00
.byte $F,$10
.byte $BA,$A0,$D0,$C1,$D5,$D3,$C5,$D3,$00
; 'S' : TOGGLES SOUND
.byte $2,$11
.byte $A7,$D3,$A7,$00
.byte $F,$11
.byte $BA,$A0,$D4,$CF,$C7,$C7,$CC,$C5,$D3,$A0
.byte $D3,$CF,$D5,$CE,$C4,$00
; 'C' : ENABLES PADDLES
.byte $2,$12
.byte $A7,$C3,$A7,$00
.byte $F,$12
.byte $BA,$A0,$C5,$CE,$C1,$C2,$CC,$C5,$D3,$A0
.byte $D0,$C1,$C4,$C4,$CC,$C5,$D3,$00
you_are_tom:
; YOU ARE TOM BOMBEM.
.byte $0,$14
.byte $D9,$CF,$D5,$A0,$C1,$D2,$C5,$A0,$D4,$CF
.byte $CD,$A0,$C2,$CF,$CD,$C2,$C5,$CD,$AE,$00
; YOU DREW THE SHORT STRAW
.byte $1,$15
.byte $D9,$CF,$D5,$A0,$C4,$D2,$C5,$D7,$A0,$D4
.byte $C8,$C5,$A0,$D3,$C8,$CF,$D2,$D4,$A0,$D3
.byte $D4,$D2,$C1,$D7,$00
; IT IS UP TO YOU TO DESTROY THE EVIL
.byte $2,$16
.byte $C9,$D4,$A0,$C9,$D3,$A0,$D5,$D0,$A0,$D4
.byte $CF,$A0,$D9,$CF,$D5,$A0,$D4,$CF,$A0,$C4
.byte $C5,$D3,$D4,$D2,$CF,$D9,$A0,$D4,$C8,$C5
.byte $A0,$C5,$D6,$C9,$CC,$00
; AND RESTORE PEACE TO THE SOL SYSTEM.
.byte $3,$17
.byte $C1,$CE,$C4,$A0,$D2,$C5,$D3,$D4,$CF,$D2
.byte $C5,$A0,$D0,$C5,$C1,$C3,$C5,$A0,$D4,$CF
.byte $A0,$D4,$C8,$C5,$A0,$D3,$CF,$CC,$A0,$D3
.byte $D9,$D3,$D4,$C5,$CD,$AE,$00
bonus_string:
; BONUS POINTS
.byte $E,$0
.byte $C2,$CF,$CE,$D5,$D3,$A0,$D0,$CF,$C9,$CE
.byte $D4,$D3,$00
no_bonus_string:
; NONE
.byte $12,$2
.byte $CE,$CF,$CE,$C5,$00
bonus_shields:
; PERFECT SHIELDS +50
.byte $9,$2
.byte $D0,$C5,$D2,$C6,$C5,$C3,$D4,$A0,$D3,$C8
.byte $C9,$C5,$CC,$C4,$D3,$A0,$A0,$A0,$AB,$B5
.byte $B0,$00
bonus_aim:
; PERFECT AIM +50
.byte $9,$3
.byte $D0,$C5,$D2,$C6,$C5,$C3,$D4,$A0,$C1,$C9
.byte $CD,$A0,$A0,$A0,$A0,$A0,$A0,$A0,$AB,$B5
.byte $B0,$00
bonus_kills:
; PERFECT KILLS +50
.byte $9,$4
.byte $D0,$C5,$D2,$C6,$C5,$C3,$D4,$A0,$CB,$C9
.byte $CC,$CC,$D3,$A0,$A0,$A0,$A0,$A0,$AB,$B5
.byte $B0,$00
high_score_string:
; HIGH SCORE
.byte $F,$A
.byte $C8,$C9,$C7,$C8,$A0,$D3,$C3,$CF,$D2,$C5,$00
new_high_score_string:
; NEW HIGH SCORE
.byte $D,$0
.byte $CE,$C5,$D7,$A0,$C8,$C9,$C7,$C8,$A0,$D3
.byte $C3,$CF,$D2,$C5,$00
; USE ARROWS TO ENTER INITIALS
.byte $6,$2
.byte $D5,$D3,$C5,$A0,$C1,$D2,$D2,$CF,$D7,$D3
.byte $A0,$D4,$CF,$A0,$C5,$CE,$D4,$C5,$D2,$A0
.byte $C9,$CE,$C9,$D4,$C9,$C1,$CC,$D3,$00
vmw_sprite:
.byte $71,$14,$72,$14,$72,$00
.byte $71,$14,$72,$14,$72,$00
.byte $10,$51,$34,$52,$34,$52,$00
.byte $10,$51,$34,$52,$34,$52,$00
.byte $20,$31,$54,$32,$54,$32,$00
.byte $20,$31,$54,$32,$54,$32,$00
.byte $30,$11,$74,$12,$74,$12,$00
.byte $30,$11,$74,$12,$74,$12,$00
.byte $00
opener_sprite:
.byte $36,$10,$36,$10,$16,$30,$16,$10,$29,$20,$39,$10,$19,$30,$19,$10,$29,$20,$29,$10,$19,$30,$19,$00
.byte $10,$16,$20,$16,$10,$16,$10,$26,$10,$26,$10,$19,$10,$19,$10,$19,$10,$19,$10,$29,$10,$29,$10,$19,$10,$19,$10,$19,$20,$29,$10,$29,$00
.byte $10,$16,$20,$16,$10,$16,$10,$16,$10,$16,$10,$16,$10,$29,$20,$19,$10,$19,$10,$19,$10,$19,$10,$19,$10,$29,$20,$29,$10,$19,$10,$19,$10,$19,$00
.byte $10,$16,$20,$16,$10,$16,$10,$16,$30,$16,$10,$19,$10,$19,$10,$19,$10,$19,$10,$19,$30,$19,$10,$19,$10,$19,$10,$19,$20,$19,$30,$19,$00
.byte $10,$16,$20,$36,$10,$16,$30,$16,$10,$29,$20,$39,$10,$19,$30,$19,$10,$29,$20,$29,$10,$19,$30,$19,$00
.byte $F0,$F0,$00
.byte $F0,$A0,$1A,$A0,$1A,$00
.byte $30,$1A,$70,$1F,$F0,$40,$15,$00
.byte $F0,$F0,$00
.byte $50,$1A,$A0,$15,$C0,$1F,$00
.byte $00
opener_sprite_2:
.byte $50,$2A,$F0,$1A,$D0,$1E,$00
.byte $10,$1D,$30,$3A,$F0,$E0,$1E,$21,$00
.byte $10,$3D,$16,$8F,$36,$F0,$60,$1E,$29,$00
.byte $2D,$11,$19,$16,$1F,$11,$14,$16,$14,$16,$2F,$46,$F0,$40,$1E,$10,$29,$00
.byte $10,$2D,$19,$16,$1F,$11,$14,$16,$14,$16,$2F,$56,$60,$11,$D0,$29,$00
.byte $30,$1D,$16,$8A,$6F,$40,$1D,$4A,$A0,$29,$00
.byte $F0,$90,$11,$80,$1F,$40,$21,$00
.byte $40,$15,$50,$1A,$F0,$00
.byte $10,$1F,$F0,$1F,$90,$15,$00
.byte $F0,$F0,$80,$1A,$00
.byte $00
vince_sprite:
.byte $20,$15,$8F,$15,$00
.byte $10,$15,$1A,$4F,$1A,$4F,$2A,$00
.byte $15,$5A,$15,$1A,$1F,$2A,$3F,$1A,$00
.byte $1A,$15,$10,$15,$3A,$8F,$15,$00
.byte $1A,$10,$15,$CF,$1A,$00
.byte $15,$10,$1A,$7F,$1A,$1F,$2A,$25,$00
.byte $25,$9F,$25,$10,$15,$1A,$00
.byte $15,$1A,$4F,$1A,$3F,$1A,$10,$15,$10,$15,$1A,$00
.byte $1A,$25,$1A,$1F,$15,$10,$15,$1A,$1F,$1A,$40,$15,$00
.byte $10,$1A,$10,$15,$1F,$20,$15,$10,$15,$1F,$15,$30,$15,$00
.byte $10,$1A,$15,$1A,$1F,$15,$20,$1A,$2F,$1A,$30,$1A,$00
.byte $10,$1F,$15,$3F,$2A,$4F,$30,$1A,$00
.byte $10,$1F,$1A,$25,$1A,$5F,$35,$2A,$00
.byte $10,$2F,$15,$10,$1A,$4F,$1A,$10,$3A,$00
.byte $10,$2F,$1A,$15,$2A,$3F,$1A,$10,$2A,$15,$00
.byte $10,$2A,$10,$15,$20,$2A,$20,$1F,$15,$1A,$00
.byte $10,$15,$2F,$15,$2F,$1A,$20,$15,$2A,$15,$00
.byte $20,$1F,$3A,$1F,$2A,$25,$1F,$1A,$00
.byte $20,$1A,$4F,$25,$10,$15,$1F,$15,$00
.byte $30,$25,$50,$1A,$1F,$15,$00
.byte $00
phobos_sprite:
.byte $F0,$30,$1A,$F0,$00
.byte $F0,$F0,$60,$1F,$00
.byte $F0,$F0,$00
.byte $50,$1A,$4F,$F0,$00
.byte $40,$11,$69,$E0,$1F,$00
.byte $30,$11,$59,$11,$29,$F0,$00
.byte $30,$11,$89,$F0,$00
.byte $30,$11,$39,$11,$49,$50,$15,$18,$F0,$00
.byte $30,$11,$89,$50,$18,$F0,$00
.byte $30,$11,$59,$11,$29,$F0,$40,$1F,$00
.byte $40,$11,$69,$F0,$1A,$00
.byte $50,$1A,$4F,$F0,$00
.byte $F0,$F0,$00
.byte $F0,$F0,$00
.byte $30,$1F,$D0,$1A,$C0,$1A,$00
.byte $00
evil_ship_sprite:
.byte $17,$1E,$17,$00
.byte $00
tom_sprite:
.byte $20,$62,$46,$00
.byte $20,$22,$40,$25,$26,$00
.byte $20,$22,$40,$25,$26,$00
.byte $20,$22,$40,$25,$26,$00
.byte $20,$22,$40,$25,$26,$00
.byte $40,$42,$26,$00
.byte $82,$86,$00
.byte $22,$26,$42,$26,$29,$22,$26,$00
.byte $22,$26,$42,$46,$22,$26,$00
.byte $22,$26,$42,$46,$22,$26,$00
.byte $22,$26,$42,$46,$22,$26,$00
.byte $22,$26,$42,$46,$22,$26,$00
.byte $27,$2E,$22,$26,$22,$26,$27,$2E,$00
.byte $40,$22,$26,$22,$26,$00
.byte $40,$22,$26,$22,$26,$00
.byte $40,$22,$26,$22,$26,$00
.byte $40,$27,$2E,$27,$2E,$00
.byte $40,$25,$2A,$25,$2A,$00
.byte $00
ship_sprite:
.byte $20,$36,$00
.byte $10,$1A,$3F,$1A,$00
.byte $2A,$1F,$1A,$1F,$2A,$00
.byte $30,$1D,$00
.byte $00
missile_sprite:
.byte $1A,$00
.byte $1A,$00
.byte $1D,$00
.byte $00
enemy_sprites:
enemy_sprite0:
.byte $2F,$11,$00
.byte $3F,$00
.byte $00,$00,$00
enemy_sprite1:
.byte $16,$10,$16,$00
.byte $10,$16,$00
.byte $00
enemy_sprite2:
.byte $17,$28,$00
.byte $17,$28,$00
.byte $00,$00
enemy_sprite3:
.byte $10,$1D,$00
.byte $1D,$10,$1D,$00
.byte $00
enemy_sprite4:
.byte $19,$2F,$00
.byte $19,$2A,$00
.byte $00,$00
enemy_sprite5:
.byte $14,$1C,$14,$00
.byte $34,$00
; .byte $14,$1C,$14,$00
.byte $00,$00
enemy_sprite6:
.byte $1F,$20,$00
.byte $3C,$00
.byte $00,$00,$00
enemy_sprite7:
.byte $33,$00
.byte $13,$10,$13,$00
.byte $00,$00
explosion_sprites:
explosion_sprite0:
.byte $2D,$19,$00
.byte $19,$1D,$15,$00
.byte $00
explosion_sprite1:
.byte $2A,$11,$00
.byte $11,$1A,$00
.byte $00,$00
explosion_sprite2:
.byte $25,$00
.byte $10,$25,$00
.byte $00
smoke_sprites:
smoke_sprite0:
.byte $10,$15,$00
.byte $00
smoke_sprite1:
.byte $25,$1A,$00
.byte $00
smoke_sprite2:
.byte $2A,$11,$00
.byte $00
boss_sprite:
.byte $1F,$1A,$11,$79,$11,$1A,$1F,$00
.byte $1F,$15,$1A,$21,$17,$1E,$17,$21,$1A,$15,$1F,$00
.byte $1F,$10,$15,$1A,$17,$1E,$1D,$1E,$17,$1A,$15,$10,$1F,$00
.byte $00
laser_sprites:
laser_sprite0:
.byte $1B,$00
.byte $13,$00
.byte $00
laser_sprite1:
.byte $13,$00
.byte $1B,$00
.byte $00
tom_head_sprite:
.byte $10,$32,$26,$00
.byte $10,$12,$20,$15,$16,$00
.byte $10,$12,$20,$15,$16,$00
.byte $10,$12,$20,$15,$16,$00
.byte $10,$12,$20,$15,$16,$00
.byte $20,$22,$16,$00
.byte $42,$36,$00
.byte $00
earth_sprite:
.byte $10,$1A,$2F,$00
.byte $14,$2C,$26,$00
.byte $12,$1C,$36,$00
.byte $12,$16,$2C,$16,$00
.byte $12,$16,$1C,$26,$00
.byte $10,$1A,$2F,$00
.byte $00
susie_sprite:
.byte $BA,$15,$10,$2F,$10,$3A,$00
.byte $2A,$90,$15,$20,$15,$20,$2A,$00
.byte $1A,$F0,$20,$1A,$00
.byte $1A,$F0,$20,$1A,$00
.byte $1A,$F0,$3A,$00
.byte $2A,$C0,$5A,$00
.byte $4A,$35,$5A,$35,$4A,$00