******************************************************************** * Space Invaders: * * Translated the original 8080 CPU Instructions to 6809 CPU * * instructions and tweaked the hardware options to work with the * * TRS-80 Color Computer 3 * * * * Version 0.01 By Glen Hewlett * * * * Code is still the same as original in that the original monitor * * inside the Space Invaders cabinet was rotated. This should be * * fixed to look proper on a regular CoCo 3. Sound still needs to * * be added. * * * * Controls are typical MAME style controls: * * Insert Coin - 5 * * 1 Player Start - 1 * * 2 Player Start - 2 * * Space to Fire * * Left Arrow moves Left * * Right Arrow moves Right * ******************************************************************** **************************************************************** * Standard CoCo 3 Labels * **************************************************************** TEXT_SCREEN_START EQU $0400 TEXT_SCREEN_END EQU $0600 FIRQ_Jump_position EQU $FEF4 * Store $7E which is the JMP opcode FIRQ_Start_Address EQU $FEF5 * Store the address you want the IRQ to jumpt to at this address FIRQ_POINT EQU $FEF4 * $FEF4 is the usual CoCo3 SuperBasic FIRQ value IRQ_Jump_position EQU $FEF7 * Store $7E which is the JMP opcode IRQ_Start_Address EQU $FEF8 * Store the address you want the IRQ to jumpt to at this address IRQ_POINT EQU $FEF7 * $FEF7 is the usual CoCo3 SuperBasic IRQ value PIA0_Byte_0_KeyRow_Joy EQU $FF00 PIA0_Byte_1_HSYNC EQU $FF01 PIA0_Byte_2_KeyColumn_Joy EQU $FF02 PIA0_Byte_3_VSYNC EQU $FF03 PIA1_Byte_0_IRQ EQU $FF20 PIA1_Byte_1_IRQ EQU $FF21 PIA1_Byte_2 EQU $FF22 PIA1_Byte_3_IRQ_Ct_Snd EQU $FF23 Disk_Cont_Reg EQU $FF40 Disk_Comnd_Stat_Reg EQU $FF48 Disk_Track_Reg EQU $FF49 Disk_Sector_Reg EQU $FF4A Disk_Data_Reg EQU $FF4B INIT0_Register0 EQU $FF90 INIT1_Register1 EQU $FF91 IRQENR EQU $FF92 FIRQENR EQU $FF93 Timer_register_MSB EQU $FF94 Timer_register_LSB EQU $FF95 Video_Mode_Register EQU $FF98 Vid_Res_Reg EQU $FF99 Border_Register EQU $FF9A VidStart EQU $FF9D * Hor_Offset_Reg EQU $FF9F MMU_Reg_0 EQU $FFA0 MMU_Reg_1 EQU $FFA1 MMU_Reg_2 EQU $FFA2 MMU_Reg_3 EQU $FFA3 MMU_Reg_4 EQU $FFA4 *Screen Colour Settings Palette_Start EQU $FFB0 Palette_0 EQU $FFB0 Palette_1 EQU $FFB1 Palette_2 EQU $FFB2 Palette_3 EQU $FFB3 Palette_4 EQU $FFB4 Palette_5 EQU $FFB5 Palette_6 EQU $FFB6 Palette_7 EQU $FFB7 Palette_8 EQU $FFB8 Palette_9 EQU $FFB9 Palette_10 EQU $FFBA Palette_11 EQU $FFBB Palette_12 EQU $FFBC Palette_13 EQU $FFBD Palette_14 EQU $FFBE Palette_15 EQU $FFBF Regular_Speed EQU $FFD8 High_Speed_Mode EQU $FFD9 ROM_Mode_Enable EQU $FFDE RAM_Mode_Enable EQU $FFDF Screen_Start EQU $2400 Border_Colour EQU $00 * 00 - Black *Stack_Space RMB $100 * Example for making some stack space *********************************************************** ORG $4400 * Start address of our Program Stack_Space RMB $100 * Leave room for a our stack Stack_Pointer EQU START-1 START *********************************************************** System_EnableGraphics ORCC #$50 * Disable interrupts LDX #$2400 * Clear the graphics RAM area CLRA * Space Invaders uses 256 x 224 screen with two colours (black and white) CLRB * Our closest size screen available it 256 x 225 with two colours ! STD ,x++ * CMPX #$2400+$1C20 * Our screen size is 256 X 225 / 8 = 7200 = $1C20 BNE < * LDA #$7E * Write the JMP instruction LDX #IRQ_Interrupt * STA IRQ_Jump_position * STX IRQ_Start_Address * Point the IRQ to our interrupt routine IRQ_Interrupt LDA #%00110100 * STA PIA0_Byte_1_HSYNC * HSYNC IRQ Disabled, IRQ Polarity Flag falling Edge, Data Direction Normal, Select Line LSB = 0, HSYNC Flag = 0 STA PIA0_Byte_3_VSYNC * VSYNC IRQ Disabled, IRQ Polarity Flag falling Edge, Data Direction Normal, Select Line MSB = 0, VSYNC Flag = 0 STA PIA1_Byte_1_IRQ * CONTROL OF CD FIRQ* TO CPU DISABLED, IRQ Polarity Falling Edge of CD, CD Flag off STA PIA1_Byte_3_IRQ_Ct_Snd * CONTROL OF Cart FIRQ* TO CPU DISABLED, IRQ Polarity Falling Edge of Cart, Cart Flag off LDA #%01101100 * STA INIT0_Register0 * CoCo 3 Mode, MMU Enabled, GIME IRQ Enabled, GIME FIRQ Disabled, Vector RAM at FEXX enabled, Standard SCS Normal, ROM Map 16k Int, 16k Ext LDA #%00000000 * STA INIT1_Register1 * Mem Type 64k chips, 63.695 usec timer, MMU Task 0 - $FFa0-$FFA7 LDA #%10000000 * STA Video_Mode_Register * Graphics mode, Colour output, 60 hz, max vertical res LDA #$28 STA IRQENR * Timer interrupt Enabled and Vertical Border Sync (VBORD) Interrupt *********************************************************** * Set Hires Screen Resolution and the number of Colours * * Bit Pattern Rows Displayed * x00xxxxx 192 * x01xxxxx 200 * x10xxxxx *zero/infinite lines on screen (undefined) * x11xxxxx 225 * Bit Pattern Bytes/Row (Graphics) * xxx000xx 16 * xxx001xx 20 * xxx010xx 32 * xxx011xx 40 * xxx100xx 64 * xxx101xx 80 320 4 Colours 01 * xxx110xx 128 * xxx111xx 160 * Bit Pattern Colours Pixels/Byte * xxxxxx00 2 8 * xxxxxx01 4 4 * xxxxxxl0 16 2 * xxxxxx11 Undefined Undefined *********************************************************** * Most Common used settings (Uncomment the one you want to use) * LDA #%00001000 * 256 x 192 x 2 Colours requires 6,144 bytes = $1800 RAM * LDA #%00101000 * 256 x 200 x 2 Colours requires 6,400 bytes = $1900 RAM LDA #%01101000 * 256 x 225 x 2 Colours requires 7,200 bytes = $1C20 RAM * LDA #%00010001 * 256 x 192 x 4 Colours requires 12,288 bytes = $3000 RAM * LDA #%00110001 * 256 x 200 x 4 Colours requires 12,800 bytes = $3200 RAM * LDA #%01110001 * 256 x 225 x 4 Colours requires 14,400 bytes = $3840 RAM * LDA #%00011010 * 256 x 192 x 16 Colours requires 24,576 bytes = $6000 RAM * LDA #%00111010 * 256 x 200 x 16 Colours requires 25,600 bytes = $6400 RAM * LDA #%01111010 * 256 x 225 x 16 Colours requires 28,800 bytes = $7080 RAM * LDA #%00001100 * 320 x 192 x 2 Colours requires 7,680 bytes = $1E00 RAM * LDA #%00101100 * 320 x 200 x 2 Colours requires 8,000 bytes = $1F40 RAM * LDA #%01101100 * 320 x 225 x 2 Colours requires 10,240 bytes = $2800 RAM * LDA #%00010101 * 320 x 192 x 4 Colours requires 15,360 bytes = $3C00 RAM * LDA #%00110101 * 320 x 200 x 4 Colours requires 16,000 bytes = $3E80 RAM * LDA #%01110101 * 320 x 225 x 4 Colours requires 18,000 bytes = $4650 RAM * LDA #%00011110 * 320 x 192 x 16 Colours requires 30,720 bytes = $7800 RAM * LDA #%00111110 * 320 x 200 x 16 Colours requires 32,00 bytes = $7D00 RAM * LDA #%01111110 * 320 x 225 x 16 Colours requires 36,000 bytes = $8CA0 RAM * LDA #%00010100 * 640 x 192 x 2 Colours requires 15,360 bytes = $3C00 RAM * LDA #%00110100 * 640 x 200 x 2 Colours requires 16,000 bytes = $3E80 RAM * LDA #%01110100 * 640 x 225 x 2 Colours requires 18,000 bytes = $4650 RAM * LDA #%00011101 * 640 x 192 x 4 Colours requires 30,720 bytes = $7800 RAM * LDA #%00111101 * 640 x 200 x 4 Colours requires 32,000 bytes = $7D00 RAM * LDA #%01111101 * 640 x 225 x 4 Colours requires 36,000 bytes = $8CA0 RAM STA Vid_Res_Reg *********************************************************** * Border settings LDA #Border_Colour * STA Border_Register * Black border LDD #$003f STD Palette_Start * Black and White colours LDD #$E480 * Point video screen veiwer at $72400 (Hi-Res page start) - Space Invaders uses $2400 STD VidStart * which is at $72400 / 8 = $E480 * Start at Hi-Res page #1 (default page) * Load GIME pages $38 & $39 into CPU RAM $0000-$1FFF & $2000-$4000 * Not needed since we are using the default RAM for our HIRES screen ; LDA #$38 * ; STA $FFA0 * $0000k-$1FFFk ; LDA #$39 * ; STA $FFA1 * $2000k-$3FFFk STA RAM_Mode_Enable * Any value stored in $FFDF Turns on All RAM mode, top 32k is now RAM STA High_Speed_Mode * Put the CoCo into High Speed mode LDD #$0083 ; 131 * 63.695 usecs should get to around the middle of the screen 60 Hz Screen (131 * 63.695) = .008344 close to (1/60) / 2 = 0.008337 STD Timer_register_MSB ANDCC #$EF * Enable the IRQ Interupt LBRA SI_Start * Start the Space Invaders Code *********************************************************** * Space Invaders Transcode starts here... *********************************************************** * SI Basic Memory Map: * $0000 - $1A92 - executable code - Moved code to give us extra room, starts at $4400 * $1A93 - $23FF - graphic/character data and variables - Used this exact RAM location * $2400 - $4000 - Screen memory - Used the same location for our Screen memory (actually we have one line, 225 lines and use memory location upto $401F) * * Memory locations used as temporary register storage since the 8080 has more registers * then the 6809 and sometimes the 8080 registers are split up. For example * the register HL is quite often used as H and then L (similar to 6809 A and B) and then combined * and read as an address (which can't be done frome 6809 D - register). * Store them in low memory where the direct page (DP) is set to Zero * ORG $0000 H_Temp FCB $00 L_Temp FCB $00 D_Temp FCB $00 E_Temp FCB $00 B_Temp FCB $00 C_Temp FCB $00 A_Temp FCB $00 AlienCount FCB $00 Shift_Amount FCB $00 Shift_Count FCB $00 Cur_Shift_Value FDB $0000 Shift_Byte FCB $00 *Special hardware references IO_01_INP1 FCB $00 IO_02_INP2 FCB %00000011 ; Dip Switches - bits 0 & 1 are added to the value 03 to give you the number of ships to start with IO_03_SOUND1 FCB $00 IO_05_SOUND2 FCB $00 IO_06_WATCHDOG FCB $00 *Handle Space Invaders Shift hardware * Substitute IO_02_SHFTAMNT with the code below (Write only command) IO_02_SHFTAMNT: ; This is the amount of bits to shift to the left from our Cur_Shift_Value ; I believe we don't have to do the shift until it is read from 03(Read)_SHFT_IN STA Shift_Amount RTS * Substitute IO_03_SHFT_IN with the code below (Read only command) IO_03_SHFT_IN: ; Do the shift and return the value in A accumulator PSHS B LDA Shift_Amount STA Shift_Count LDD Cur_Shift_Value ShiftLoop: TST Shift_Count BEQ > ASLB ROLA DEC Shift_Count BRA ShiftLoop ! PULS B RTS * End loop * Substitute IO_04_SHFT_DATA with the code below (Write only command) IO_04_SHFT_DATA: ; * Input value in A PSHS B STA Shift_Byte LDD Cur_Shift_Value TFR A,B LDA Shift_Byte STD Cur_Shift_Value PULS B RTS *********************************************************** SI_Start: * Build a lookup table, SI screen data is backwards the bit order is 0,1,2,3,4,5,6,7 * where the 6809 and the CoCo uses bit order of 7,6,5,4,3,2,1,0 ;Invert bits - lookup buffer Table ReverseLookUp CLRA LDX #ReverseTable+$80 ! PSHS A CLRB RORA ROLB RORA ROLB RORA ROLB RORA ROLB RORA ROLB RORA ROLB RORA ROLB RORA ROLB STB ,X+ PULS A INCA CMPA #$80 BNE < LDX #ReverseTable ! PSHS A CLRB RORA ROLB RORA ROLB RORA ROLB RORA ROLB RORA ROLB RORA ROLB RORA ROLB RORA ROLB STB ,X+ PULS A INCA TSTA BNE < ; Reverse ShieldImage Data in order to fix 2 player double reversal problem LDX #ShieldImage LDU #ReverseLookUp ! LDA ,X LDA A,U STA ,X+ CMPX #ShieldImage+44 BNE < LBRA L18D4 * Go to Game initialization ReverseTable RMB $100 ; Space for our lookup table to flip the bits of the data ReverseLookUp EQU ReverseTable+$80 ; Execution begins here on power-up and reset. *Reset LBRA L18D4 ;0003 -JP $18D4 Continue startup at 18D4 ************************************************** *Interrupt * * The IRQ Interrupt is being used below, two points are used for the IRQ, one is a timer to estimate the scanline point is at line 96 or * halfway down the screen. The other is the Vertical Border interrupt, which means the screen has finished being drawn IRQ_Interrupt LDA IRQENR * IRQENR - Interrupt request enable register BITA #$20 * Check if the Timer Interrupt Occurred (TMR) LBNE ScanLine96 * Must be a Timer Interrupt - Go act on Timer Event we have just drawn the top half of the screen BITA #$08 * Check VBORD vertical border interrupt Occurred - went low BNE ScanLine224 * Must be a VBORD Interrupt - Go act on VBORD interrupt the bottom half of the screen has just been drawn RTI * Interrupt brings us here when the beam is at the end of the screen (line 224) when the VBLANK begins. ScanLine224: LDX H_Temp LDY D_Temp LDU B_Temp PSHS X,Y,U LDD #$0083 ; 131 * 63.695 usecs should get to around the middle of the 60 Hz Screen (131 * 63.695) = .008344 close to (1/60) / 2 = 0.008337 STD Timer_register_MSB LDA #$80 ;0014 -LD A,$80 Flag that tells objects ... STA vblankStatus ;0016 -LD (vblankStatus),A ... on the lower half of the screen to draw/move DEC isrDelay ;0019 -LD HL,$20C0 Decrement ... ;001C -DEC (HL) ... the general countdown (used for pauses) JSR CheckHandleTilt ;001D -CALL CheckHandleTilt Check and handle TILT LDX #PIA0_Byte_0_KeyRow_Joy * PIA0 - Read Coin switch LDA #%11011111 * We used 5 on keyboard for quarter inserted STA $02,X * Store bit 5 in FF02 LDA ,X * Read keyboard BITA #%00010000 * Check bit 4 BEQ L0067 * jump if bit 4 is set (coin inserted, 5 pressed) ;0023 -JP C,$0067 Yes ... note that switch is closed and ; continue at 3F with A=1 TST coinSwitch ;0026 -LD A,(coinSwitch) Switch is now open. Was it ... ;0029 -AND A ... closed last time? BEQ L0042 ;002A -JP Z,$0042 No ... skip registering the credit LDA numCoins ;002D -LD A,(numCoins) Number of credits in BCD CMPA #$99 ;0030 -CP $99 99 credits already? BEQ L003E ;0032 -JP Z,$003E Yes ... ignore this (better than rolling over to 00) ADDA #$01 ;0035 -ADD A,$01 Bump number of credits DAA ;0037 -DAA Make it binary coded decimal STA numCoins ;0038 -LD (numCoins),A New number of credits JSR DrawNumCredits ;003B -CALL DrawNumCredits Draw credits on screen L003E CLRA ;003E -XOR A Credit switch ... L003F STA coinSwitch ;003F -LD (coinSwitch),A ... has opened ; L0042 TST suspendPlay ;0042 -LD A,(suspendPlay) Are we moving ... ;0045 -AND A ... game objects? BEQ L0082 ;0046 -JP Z,$0082 No ... restore registers and out TST gameMode ;0049 -LD A,(gameMode) Are we in ... ;004C -AND A ... game mode? BNE L006F ;004D -JP NZ,$006F Yes ... go process game-play things and out TST numCoins ;0050 -LD A,(numCoins) Number of credits ;0053 -AND A Are there any credits (player standing there)? BNE L005D ;0054 -JP NZ,$005D Yes ... skip any ISR animations for the splash screens JSR ISRSplTasks ;0057 -CALL ISRSplTasks Process ISR tasks for splash screens BRA L0082 ;005A -JP $0082 Restore registers and out ; ; At this point no game is going and there are credits L005D TST waitStartLoop ;005D -LD A,(waitStartLoop) Are we in the ... ;0060 -AND A ... "press start" loop? BNE L0082 ;0061 -JP NZ,$0082 Yes ... restore registers and out LBRA WaitForStart ;0064 -JP WaitForStart Start the "press start" loop ; ; Mark credit as needing registering L0067 LDA #$01 ;0067 -LD A,$01 Remember switch ... STA coinSwitch ;0069 -LD (coinSwitch),A ... state for debounce BRA L003F ;006C -JP $003F Continue ; ; Main game-play timing loop L006F JSR TimeFleetSound ;006F -CALL TimeFleetSound Time down fleet sound and sets flag if needs new delay value L0072 LDA obj2TimerExtra ;0072 -LD A,(obj2TimerExtra) Use rolling shot's timer to sync ... STA shotSync ;0075 -LD (shotSync),A ... other two shots JSR DrawAlien ;0078 -CALL DrawAlien Draw the current alien (or exploding alien) JSR RunGameObjs ;007B -CALL RunGameObjs Process game objects (including player object) JSR TimeToSaucer ;007E -CALL TimeToSaucer Count down time to saucer ; NOP ;0081 -NOP ** Why are we waiting? L0082 PULS U,Y,X STX H_Temp STY D_Temp STU B_Temp RTI ;0087 -RET Return from interrupt ; Continues here at scanline 96 ; ScanLine96: LDX H_Temp LDY D_Temp LDU B_Temp PSHS X,Y,U CLRA ;008C -XOR A Flag that tells ... STA vblankStatus ;008D -LD (vblankStatus),A ... objects on the upper half of screen to draw/move LDA suspendPlay ;0090 -LD A,(suspendPlay) Are we moving ... TSTA ;0093 -AND A ... game objects? LBEQ L0082 ;0094 -JP Z,$0082 No ... restore and return LDA gameMode ;0097 -LD A,(gameMode) Are we in ... TSTA ;009A -AND A ... game mode? BNE L00A5 ;009B -JP NZ,$00A5 Yes .... process game objects and out LDA isrSplashTask ;009E -LD A,(isrSplashTask) Splash-animation tasks ASRA ;00A1 -RRCA If we are in demo-mode then we'll process the tasks anyway BCC L0082 ;00A2 -JP NC,$0082 Not in demo mode ... done ; L00A5 LDX #objStruct1 ;00A5 -LD HL,$2020 Game object table (skip player-object at 2010) JSR L024B ;00A8 -CALL $024B Process all game objects (except player object) JSR CursorNextAlien ;00AB -CALL CursorNextAlien Advance cursor to next alien (move the alien if it is last one) PULS U,Y,X STX H_Temp STY D_Temp STU B_Temp RTI ;BRA L0082 ;00AE -JP $0082 Restore and return InitRack: ; Initialize the player's rack of aliens. Copy the reference-location and deltas from the ; player's data bank. ; JSR GetAlRefPtr ;00B1 -CALL GetAlRefPtr 2xFC Get current player's ref-alien position pointer PSHS X ;00B4 -PUSH HL Hold pointer LDB ,X+ LDA ,X ;00B5 -LD A,(HL) Get player's ... a=peek(hl) LDX ,X ; INC X ;00B6 -INC HL ... ref-alien ... hl=hl+1 ; LDA ,X ;00B7 -LD H,(HL) ... h=peek(hl) ; TFR A,B ;00B8 -LD L,A ... coordinates poke hl,h STD refAlienYr ;00B9 -LD (refAlienYr),HL Set game's reference alien's X,Y LPOKE refAlienYr,hl STX refAlienYr STD alienPosLSB ;00BC -LD (alienPosLSB),HL Set game's alien cursor bit position LPOKE alienPosLSB,hl STX alienPosLSB PULS X ;00BF -POP HL Restore pointer LEAX -1,X ;00C0 -DEC HL 21FB or 22FB ref alien's delta (left or right) LDB ,X ;00C1 -LD A,(HL) Get ref alien's delta X CMPB #$03 ;00C2 -CP $03 If there is one alien it will move right at 3 LBNE L00C8 ;00C4 -JP NZ,$00C8 Not 3 ... keep it DECB ;00C7 -DEC A If it is 3, back it down to 2 until it switches again L00C8 STB refAlienDXr ;00C8 -LD (refAlienDXr),A Store alien deltaY LDA #$00 ;Moved up so CMPB & BNE works CMPB #$FE ;00CB -CP $FE Moving left? ; LDA #$00 ;00CD -LD A,$00 Value of 0 for rack-moving-right (not XOR so flags are unaffected) LBNE L00D3 ;00CF -JP NZ,$00D3 Not FE ... keep the value 0 for right INCA ;00D2 -INC A It IS FE ... use 1 for left L00D3 STA rackDirection ;00D3 -LD (rackDirection),A Store rack direction RTS ;00D6 -RET Done L00D7 LDA #$02 ;00D7 -LD A,$02 Set ... STA p1RefAlienDX ;00D9 -LD (p1RefAlienDX),A ... player 1 and 2 ... STA p2RefAlienDX ;00DC -LD (p2RefAlienDX),A ... alien delta to 2 (right 2 pixels) LBRA L08E4 ;00DF -JP $08E4 ; FCB $00 ;00E2 -0 00 00 00 00 00 00 00 00 ; FCB $00 ;00F0 -0 00 00 00 00 00 00 00 00 00 ; ; This is heavily patched from a previous version of the code. There was a test here to jump to a ; self-test routine on startup (based on a dip switch). Even the original code padded with zeros ; to make the next function begin at 0100. Room for expansion? DrawAlien: ; 2006 holds the index into the alien flag data grid. 2067 holds the MSB of the pointer (21xx or 22xx). ; If there is an alien exploding time it down. Otherwise draw the alien if it alive (or skip if ; it isn't). If an alien is drawn (or blank) then the 2000 alien-drawing flag is cleared. ; LDX #alienIsExploding ;0100 -LD HL,$2002 Is there an ... LDA ,X ;0103 -LD A,(HL) ... alien ... TSTA ;0104 -AND A ... exploding? LBNE AExplodeTime ;0105 -JP NZ,AExplodeTime Yes ... go time it down and out ; PSHS X ;0108 -PUSH HL 2002 on the stack ; LDA alienCurIndex ;0109 -LD A,(alienCurIndex) Get alien index ... A=peek(alienCurIndex) LDB alienCurIndex ; TFR A,B ;010C -LD L,A ... for the 21xx or 22xx pointer L=A ; LDA playerDataMSB ;010D -LD A,(playerDataMSB) Get MSB ... A=peek(playerDataMSB) LDA playerDataMSB ; TFR A,A ;0110 -LD H,A ... of data area (21xx or 22xx) H=A TFR D,X LDB alienCurIndex LDA playerDataMSB TFR D,X LDA ,X ;0111 -LD A,(HL) Get alien status flag a=peek(HL) A=peek(HL) ; TSTA Move down ;0112 -AND A Is the alien alive? A=A and A PULS X ;0113 -POP HL HL=2002 TSTA LBEQ L0136 ;0114 -JP Z,$0136 No alien ... skip drawing alien sprite (but flag done) LEAX 1,X ;0117 -INC HL HL=2003 Bump descriptor LEAX 1,X ;0118 -INC HL HL=2004 Point to alien's row LDA ,X ;0119 -LD A,(HL) Get alien type LEAX 1,X ;011A -INC HL HL=2005 Bump descriptor LDB ,X ;011B -LD B,(HL) Get animation number ANDA #$FE ;011C -AND $FE Translate row to type offset as follows: ... ASLA ;011E -RLCA ... 0,1 -> 32 (type 1) ... ASLA ;011F -RLCA ... 2,3 -> 16 (type 2) ... ASLA ;0120 -RLCA ... 4 -> 32 (type 3) on top row ; TFR A,E ;0121 -LD E,A Sprite offset LSB E=A ; LDD #$00 ;0122 -LD D,$00 MSB is 0 D=0 ; LDX #AlienSprA ;0124 -LD HL,$1C00 Position 0 alien sprites HL=$1C00 y=#$1C00 LDY #AlienSprA ; ADD X=X+Y ;0127 -ADD HL,DE Offset to sprite type HL=HL+DE y=y+A LEAY A,Y ; EXG Y,X ;0128 -EX DE,HL Sprite offset to DE HL=DE,DE=HL x=y,y=x TFR B,A ;0129 -LD A,B Animation frame number TSTA ;012A -AND A Is it position 0? BEQ L012E ;012B - No ... add 30 and use position 1 alien sprites JSR L013B ;**** -CALL NZ,$013B Added for 6809 worakaround L012E ; LDX alienPosLSB ;012E -LD HL,(alienPosLSB) Pixel position LDD alienPosLSB EXG A,B TFR D,X LDB #$10 ;0131 -LD B,$10 16 rows in alien sprites JSR DrawSprite ;0133 -CALL DrawSprite Draw shifted sprite ; L0136 CLRA ;0136 -XOR A Let the ISR routine ... STA waitOnDraw ;0137 -LD (waitOnDraw),A ... advance the cursor to the next alien RTS ;013A -RET Out ; ;L013B LDX #$0030 ;013B -LD HL,$0030 Offset sprite pointer ... ; ADD X=X+Y ;013E -ADD HL,DE ... to animation frame 1 sprites L013B ; LEAX $30,y ; EXG Y,X ;013F -EX DE,HL Back to DE LEAY $30,Y RTS ;0140 -RET Out CursorNextAlien: ; This is called from the mid-screen ISR to set the cursor for the next alien to draw. ; When the cursor moves over all aliens then it is reset to the beginning and the reference ; alien is moved to its next position. ; ; The flag at 2000 keeps this in sync with the alien-draw routine called from the end-screen ISR. ; When the cursor is moved here then the flag at 2000 is set to 1. This routine will not change ; the cursor until the alien-draw routine at 100 clears the flag. Thus no alien is skipped. ; LDA playerOK ;0141 -LD A,(playerOK) Is the player ... TSTA ;0144 -AND A ... blowing up? BNE > ;0145 -RET Z Yes ... ignore the aliens RTS ! * Jump here LDA waitOnDraw ;0146 -LD A,(waitOnDraw) Still waiting on ... TSTA ;0149 -AND A ... this alien to be drawn? BEQ > ;014A -RET NZ Yes ... leave cursor in place RTS ! * Jump here LDA playerDataMSB ;014B -LD A,(playerDataMSB) Load alien-data ... STA H_Temp ;014E -LD H,A ... MSB (either 21xx or 22xx) LDA alienCurIndex ;014F -LD A,(alienCurIndex) Load the xx part of the alien flag pointer LDB #$02 ;0152 -LD D,$02 When all are gone this triggers 1A1 to return from this stack frame STB D_Temp L0154 INCA ;0154 -INC A Have we drawn all aliens ... CMPA #$37 ;0155 -CP $37 ... at last position? BNE L015A ;0157 - Yes ... move the bottom/right alien and reset index to 0 JSR MoveRefAlien ;**** -CALL Z,MoveRefAlien Added for 6809 worakaround L015A STA L_Temp ;015A -LD L,A HL now points to alien flag LDB [H_Temp] ;015B -LD B,(HL) Is alien ... DECB ;015C -DEC B ... alive? LBNE L0154 ;015D -JP NZ,$0154 No ... skip to next alien STA alienCurIndex ;0160 -LD (alienCurIndex),A New alien index JSR GetAlienCoords ;0163 -CALL GetAlienCoords Calculate bit position and type for index ; LDA C_Temp ;0166 -LD H,C The calculation returns the MSB in C ; STA H_Temp ; LDX H_Temp ; STX alienPosLSB ;0167 -LD (alienPosLSB),HL Store new bit position ; LDA L_Temp ;016A -LD A,L Has this alien ... ; CMPA #$28 ;016B -CP $28 ... reached the end of screen? ; LBCS L1971 ;016D -JP C,$1971 Yes ... kill the player ; LDA L_Temp ;0170 -LD A,D This alien's ... ; STA alienRow ;0171 -LD (alienRow),A ... row index ; LDA #$01 ;0174 -LD A,$01 Set the wait-flag for the ... ; STA waitOnDraw ;0176 -LD (waitOnDraw),A ... draw-alien routine to clear ; RTS ;0179 -RET Done ;Reverse These LDB C_Temp LDA L_Temp STD alienPosLSB CMPA #$28 LBCS L1971 LDA D_Temp STA alienRow LDA #$01 STA waitOnDraw RTS GetAlienCoords: ; Convert alien index in L to screen bit position in C,L. ; Return alien row index (converts to type) in D. ; CLR D_Temp ;017A -LD D,$00 Row 0 LDA L_Temp ;017C -LD A,L Hold onto alien index LDX #refAlienYr ;017D -LD HL,$2009 Get alien X ... LDB ,X+ ;0180 -LD B,(HL) ... to B STB B_Temp ;0181 -INC HL Get alien y ... LDB ,X ;0182 -LD C,(HL) ... to C STB C_Temp L0183 CMPA #$0B ;0183 -CP $0B Can we take a full row off of index? LBMI L0194 ;0185 -JP M,$0194 No ... we have the row SBCA #$0B ;0188 -SBI A,$0B Subtract off 11 (one whole row) STA E_Temp ;018A -LD E,A Hold the new index LDA B_Temp ;018B -LD A,B Add ... ADDA #$10 ;018C -ADD A,$10 ... 16 to bit ... STA B_Temp ;018E -LD B,A ... position Y (1 row in rack) LDA E_Temp ;018F -LD A,E Restore tallied index INC D_Temp ;0190 -INC D Next row LBRA L0183 ;0191 -JP $0183 Keep skipping whole rows ; L0194 LDB B_Temp ;0194 -LD L,B We have the LSB (the row) STB L_Temp L0195 TSTA ;0195 -AND A Are we in the right column? BNE > ;0196 -RET Z Yes ... X and Y are right RTS ! * Jump here STA E_Temp ;0197 -LD E,A Hold index LDA C_Temp ;0198 -LD A,C Add ... ADDA #$10 ;0199 -ADD A,$10 ... 16 to bit ... STA C_Temp ;019B -LD C,A ... position X (1 column in rack) LDA E_Temp ;019C -LD A,E Restore index DECA ;019D -DEC A We adjusted for 1 column LBRA L0195 ;019E -JP $0195 Keep moving over column MoveRefAlien: ; The "reference alien" is the bottom left. All other aliens are drawn relative to this ; reference. This routine moves the reference alien (the delta is set elsewhere) and toggles ; the animation frame number between 0 and 1. ; DEC D_Temp ;01A1 -DEC D This decrements with each call to move LBEQ ReturnTwo ;01A2 -JP Z,ReturnTwo Return out of TWO call frames (only used if no aliens left) LDX #alienCurIndex ;01A5 -LD HL,$2006 Set current alien ... CLR ,X+ ;01A8 -LD (HL),$00 ... index to 0 ;01AA -INC HL Point to DeltaX LDA ,X STA C_Temp ;01AB -LD C,(HL) Load DX into C CLR ,X ;01AC -LD (HL),$00 Set DX to 0 JSR AddDelta ;01AE -CALL AddDelta Move alien LDX #alienanimframe ;01B1 -LD HL,$2005 Alien animation frame number LDA ,X ;01B4 -LD A,(HL) Toggle ... INCA ;01B5 -INC A ... animation ... ANDA #$01 ;01B6 -AND $01 ... number between ... STA ,X ;01B8 -LD (HL),A ... 0 and 1 ; CLRA ;01B9 -XOR A Alien index in A is now 0 LDX #playerDataMSB ;01BA -LD HL,$2067 Restore H ... LDA ,X ;01BD -LD H,(HL) ... to player data MSB (21 or 22) LDB #$67 STD H_Temp CLRA RTS ;01BE -RET Done ; FCB $00 ;01BF -RET ** Why? InitAliens: ; Initialize the 55 aliens from last to 1st. 1 means alive. ; LDX #StartAlienStr ;01C0 -LD HL,$2100 Start of alien structures (this is the last alien) L01C3 LDB #$37 ;01C3 -LD B,$37 Count to 55 (that's five rows of 11 aliens) LDA #$1 L01C5 STA ,X+ ;01C5 -LD (HL),$01 Bring alien to live ; LEAX 1,X ;01C7 -INC HL Next alien DECB ;01C8 -DEC B All done? LBNE L01C5 ;01C9 -JP NZ,$01C5 No ... keep looping RTS ;01CC -RET Done ReturnTwo: ; If there are no aliens left on the screen then MoveDrawAlien comes here which returns from the ; caller's stack frame. ; PULS X ;01CD -POP HL Drop return to caller RTS ;01CE -RET Return to caller's caller DrawBottomLine: ; Draw a 1px line across the player's stash at the bottom of the screen. ; ; LDA #$01 ;01CF -LD A,$01 Bit 1 set ... going to draw a 1-pixel stripe down left side **** Check graphics ; LDB #$E0 ;01D1 -LD B,$E0 All the way down the screen ; LDX #Screen_Start+2 ;01D3 -LD HL,$2402 Screen coordinates (3rd byte from upper left) (Screen starts at $8000) ; LBRA L14CC ;01D6 -JP $14CC Draw line down left side LDA #$80 ;01CF -LD A,$01 Flip to bit 7 set (graphics byte reversal) ... going to draw a 1-pixel stripe down left side **** Check graphics LDB #$E0 ;01D1 -LD B,$E0 All the way down the screen (224 lines) LDX #Screen_Start+2 ;01D3 -LD HL,$2402 Screen coordinates (3rd byte from upper left) (Screen starts at $8000) LBRA L14CC ;01D6 -JP $14CC Draw line down left side AddDelta: ; HL points to descriptor: DX DY XX YY except DX is already loaded in C ; ** Why the "already loaded" part? Why not just load it here? ; LEAX 1,X ;01D9 -INC HL We loaded delta-x already ... skip over it LDB ,X+ ;01DA -LD B,(HL) Get delta-y ; LEAX 1,X ;01DB -INC HL Skip over it LDA C_Temp ;01DC -LD A,C Add delta-x ... ADDA ,X ;01DD -ADD A,(HL) ... to x STA ,X+ ;01DE -LD (HL),A Store new x ; LEAX 1,X ;01DF -INC HL Skip to y TFR B,A ;01E0 -LD A,B Add delta-y ... ADDA ,X ;01E1 -ADD A,(HL) ... to y STA ,X ;01E2 -LD (HL),A Store new y RTS ;01E3 -RET Done CopyRAMMirror: ; Block copy ROM mirror 1B00-1BBF to initialize RAM at 2000-20BF. ; ; *Changed to copy from RAMInit to $2000 (for now) LDB #$C0 ;01E4 -LD B,$C0 Number of bytes ; ; Called From $18d9 and copies 256 bytes ; L01E6 LDY #RAMInit ;01E6 -LD DE,$1B00 RAM mirror in ROM LDX #waitOnDraw ;01E9 -LD HL,$2000 Start of RAM LBRA BlockCopy ;01EC -JP BlockCopy Copy [DE]->[HL] and return DrawShieldPl1: ; Draw the shields for player 1 (draws it in the buffer in the player's data area). ; LDX #Ply1ShieldBuff ;01EF -LD HL,$2142 Player 1 shield buffer (remember between games in multi-player) LBRA L01F8 ;01F2 -JP $01F8 Common draw point ; DrawShieldPl2: ; Draw the shields for player 1 (draws it in the buffer in the player's data area). ; LDX #Ply2ShieldBuff ;01F5 -LD HL,$2242 Player 2 shield buffer (remember between games in multi-player) ; L01F8 LDB #$04 ;01F8 -LD C,$04 Going to draw 4 shields LDY #ShieldImage ;01FA -LD DE,$1D20 Shield pixel pattern L01FD PSHS Y,B ;01FD -PUSH DE Hold the start for the next shield LDB #$2C ;01FE -LD B,$2C 44 bytes to copy JSR BlockCopy ;0200 -CALL BlockCopy Block copy DE to HL (B bytes) PULS B,Y ;0203 -POP DE Restore start of shield pattern DECB ;0204 -DEC C Drawn all shields? LBNE L01FD ;0205 -JP NZ,$01FD No ... go draw them all RTS ;0208 -RET Done RememberShields1: ; Copy shields on the screen to player 1's data area. ; LDA #$01 ;0209 -LD A,$01 Not zero means remember LBRA L021B ;020B -JP $021B Shuffle-shields player 1 RememberShields2: ; Copy shields on the screen to player 2's data area. ; LDA #$01 ;020E -LD A,$01 Not zero means remember LBRA L0214 ;0210 -JP $0214 Shuffle-shields player 2 RestoreShields2: ; Copy shields from player 2's data area to screen. ; CLRA ;0213 -XOR A Zero means restore L0214 LDY #Ply2ShieldBuff ;0214 -LD DE,$2242 Player 2 shield buffer (remember between games in multi-player) LBRA CopyShields ;0217 -JP CopyShields Shuffle-shields player 2 RestoreShields1: ; Copy shields from player 1's data area to screen. ; CLRA ;021A -XOR A Zero means restore L021B LDY #Ply1ShieldBuff ;021B -LD DE,$2142 Player 1 shield buffer (remember between games in multi-player) CopyShields: ; A is 1 for screen-to-buffer, 0 for to buffer-to-screen ; HL is screen coordinates of first shield. There are 23 rows between shields. ; DE is sprite buffer in memory. ; ; STA tmp2081 ;021E -LD (tmp2081),A Remember copy/restore flag ; LDU #$1602 ;0221 -LD BC,$1602 22 rows, 2 bytes/row (for 1 shield pattern) ;; STU B_Temp ;; LDB B_Temp ; LDX #Screen_Start+$406 ;0224 -LD HL,$2806 Screen coordinates ; LDB #$04 ;0227 -LD A,$04 Four shields to move ;L0229 PSHS B ;0229 -PUSH AF Hold shield count ; STU B_Temp ;022A -PUSH BC Hold sprite-size ; LDA tmp2081 ;022B -LD A,(tmp2081) Get back copy/restore flag ;; TSTA ;022E -AND A Not zero ... ; LBNE L0242 ;022F -JP NZ,$0242 ... means remember shidles ; JSR RestoreShields ;0232 -CALL RestoreShields Restore player's shields ;L0235 PULS B ;0235 -POP BC Get back sprite-size ;; PULS A,CC ;0236 -POP AF Get back shield count ; DECB ;0237 -DEC A Have we moved all shields? ; BNE > ;0238 -RET Z Yes ... out ; RTS ;! * Jump here ;; PSHS Y ;0239 -PUSH DE Hold sprite buffer ;; LDY #$02E0 ;023A -LD DE,$02E0 Add 2E0 (23 rows) to get to ... ;; ADD X=X+Y ;023D -ADD HL,DE ... next shield on screen ; LEAX $02E0,X ;; PULS Y ;023E -POP DE restore sprite buffer ; LBRA L0229 ;023F -JP $0229 Go back and do all STA tmp2081 ;021E -LD (tmp2081),A Remember copy/restore flag LDU #$1602 ;0221 -LD BC,$1602 22 rows, 2 bytes/row (for 1 shield pattern) LDX #Screen_Start+$406 ;0224 -LD HL,$2806 Screen coordinates LDA #$04 ;0227 -LD A,$04 Four shields to move L0229 PSHS A ;0229 -PUSH AF Hold shield count PSHS U ;022A -PUSH BC Hold sprite-size STU B_Temp LDA tmp2081 ;022B -LD A,(tmp2081) Get back copy/restore flag TSTA ;022E -AND A Not zero ... LBNE L0242 ;022F -JP NZ,$0242 ... means remember shidles JSR RestoreShields ;0232 -CALL RestoreShields Restore player's shields L0235 PULS U ;0235 -POP BC Get back sprite-size PULS A ;0236 -POP AF Get back shield count DECA ;0237 -DEC A Have we moved all shields? BNE > ;0238 -RET Z Yes ... out RTS ! * Jump here LEAX $02E0,X LBRA L0229 ;023F -JP $0229 Go back and do all ; L0242 JSR RememberShields ;0242 -CALL RememberShields Remember player's shields LBRA L0235 ;0245 -JP $0235 Continue with next shield RunGameObjs: ; Process game objects. Each game object has a 16 byte structure. The handler routine for the object ; is at xx03 and xx04 of the structure. The pointer to xx04 is pushed onto the stack before calling ; the handler. ; ; All game objects (except task 0 ... the player) are called at the mid-screen and end-screen renderings. ; Each object decides when to run based on its Y (not rotated) coordinate. If an object is on the lower ; half of the screen then it does its work when the beam is at the top of the screen. If an object is ; on the top of the screen then it does its work when the beam is at the bottom. This keeps the ; object from updating while it is being drawn which would result in an ugly flicker. ; ; ; The player is only processed at the mid-screen interrupt. I am not sure why. ; ; The first three bytes of the structure are used for status and timers. ; ; If the first byte is FF then the end of the game-task list has been reached. ; If the first byte is FE then the object is skipped. ; ; If the first-two bytes are non-zero then they are treated like a two-byte counter ; and decremented as such. The 2nd byte is the LSB (moves the fastest). ; ; If the first-two bytes are zero then the third byte is treated as an additional counter. It ; is decremented as such. ; ; When all three bytes reach zero the task is executed. ; ; The third-byte-counter was used as a speed-governor for the player's object, but evidently even the slowest ; setting was too slow. It got changed to 0 (fastest possible). ; ; LDX #firstgameobj ;0248 -LD HL,$2010 First game object (active player) ;L024B LDA ,X ;024B -LD A,(HL) Have we reached the ... ; CMPA #$FF ;024C -CP $FF ... end of the object list? ; BNE > ;024E -RET Z Yes ... done ; RTS ;! * Jump here ; CMPA #$FE ;024F -CP $FE Is object active? ; LBEQ L0281 ;0251 -JP Z,$0281 No ... skip it ;; LEAX 1,X ;0254 -INC HL xx01 ;; LDB ,X ;0255 -LD B,(HL) First byte to B ;; STA C_Temp ;0256 -LD C,A Hold 1st byte ;; ORA ,X ;0257 -OR B OR 1st and 2nd byte X points to B so ORA ,X works ;; PSHS CC ; Save CC as 6809 LDA changes CC ;; LDA C_Temp ;0258 -LD A,C Restore 1st byte ;; PULS CC ; 8080 LD or MOV does not affect CC (from Reg to Reg) ;; LBNE L0277 ;0259 -JP NZ,$0277 If word at xx00,xx02 is non zero then decrement it ;; ; LEAX 1,X ;025C -INC HL xx02 ; LDA ,X ;025D -LD A,(HL) Get byte counter ; TSTA ;025E -AND A Is it 0? ; LBNE L0288 ;025F -JP NZ,$0288 No ... decrement byte counter at xx02 ; LEAX 1,X ;0262 -INC HL xx03 ;; LDA ,X+ ;0263 -LD E,(HL) Get handler address LSB ;; STA E_Temp ;; LEAX 1,X ;0264 -INC HL xx04 ;; LDA ,X ;0265 -LD D,(HL) Get handler address MSB ;; STA D_Temp ;; LDY D_Temp ; LDY ,X+ ; PSHS X ;0266 -PUSH HL Remember pointer to MSB X point to Handler MSB ; EXG Y,X ;0267 -EX DE,HL Handler address to HL Y now has address of handler, X now has jump address ;; PSHS X ;0268 -PUSH HL Now to stack (making room for indirect call) ;; LDX #SPNT01 ;0269 -LD HL,$026F Return address to 026F ; LDU #SPNT01 ; U = address ahead that we will return to from jump ;; EXG S,X ;026C -EX (SP),HL Return address (026F) now on stack. Handler in HL. ; PSHS U Push the address on the stack ; PSHS Y ;026D -PUSH DE Push pointer to data struct (xx04) for handler to use ;; TFR HL,PC transfer HL to Program coun;026E -PCHL Run object''s code (will return to next line) ; JMP ,X jump to where x points ; After lots of debugging with the addresses of the jumps I rewrote this section to make it clear for myself ; the above code worked but it was too fragmented for me to understand LDX #firstgameobj L024B LDA ,X CMPA #$FF BNE > RTS ! * Jump here CMPA #$FE LBEQ L0281 LEAX 1,X LDB ,X TSTB BNE L0277 LEAX 1,X LDA ,X TSTA LBNE L0288 LEAX 1,X LDY ,X+ PSHS X EXG Y,X LDU #SPNT01 PSHS U PSHS Y JMP ,X SPNT01 PULS X ;026F -POP HL Restore pointer to xx04 ; LDY #$000C ;0270 -LD DE,$000C Offset to next ... LEAX $0C,X ;0273 -ADD HL,DE ... game task (C+4=10) LBRA L024B ;0274 -JP $024B Do next game task ; ; Word at xx00 and xx01 is non-zero. Decrement it and move to next task. ;L0277 DECB ;0277 -DEC B Decrement ... ; INCB ;0278 -INC B ... two ... ; LBNE L027D ;0279 -JP NZ,$027D ... byte ... ; DECA ;027C -DEC A ... value ... ;L027D DECB ;027D -DEC B ... at ... ;; STB ,X ;027E -LD (HL),B ... xx00 ... ; LEAX -1,X ;027F -DEC HL ... and ... ;; STA ,X ;0280 -LD (HL),A ... xx01 ; STD ,X ; Save decremented value at xx00 & xx01 L0277 SUBD #$0001 LEAX -1,X STD ,X ; L0281 ;LDY #$0010 ;0281 -LD DE,$0010 Next ... LEAX $10,X ;0284 -ADD HL,DE ... object descriptor LBRA L024B ;0285 -JP $024B Keep processing game objects ; ; Word at xx00 and xx01 is zero and byte at xx02 is non-zero. Decrement xx02 and ; move to next task. L0288 DEC ,X ;0288 -DEC (HL) Decrement the xx02 counter LEAX -1,X ;0289 -DEC HL Back up to ... LEAX -1,X ;028A -DEC HL ... start of game task LBRA L0281 ;028B -JP $0281 Next game task GameObj0: ; Game object 0: Move/draw the player ; ; This task is only called at the mid-screen ISR. It ALWAYS does its work here, even though ; the player can be on the top or bottom of the screen (not rotated). ; ; PULS X ; RTS PULS X ;028E -POP HL Get player object structure 2014 LEAX 1,X ;028F -INC HL Point to blow-up status LDA ,X ;0290 -LD A,(HL) Get player blow-up status CMPA #$FF ;0291 -CP $FF Player is blowing up? LBEQ L033B ;0293 -JP Z,$033B No ... go do normal movement ; ; Handle blowing up player LEAX 1,X ;0296 -INC HL Point to blow-up delay count DEC ,X ;0297 -DEC (HL) Decrement the blow-up delay BEQ > ;0298 -RET NZ Not time for a new blow-up sprite ... out RTS ! * Jump here TFR A,B ;0299 -LD B,A Hold sprite image number CLRA ;029A -XOR A 0 STA playerOK ;029B -LD (playerOK),A Player is NOT OK ... player is blowing up STA enableAlienFire ;029E -LD (enableAlienFire),A Alien fire is disabled LDA #$30 ;02A1 -LD A,$30 Reset count ... STA alienFireDelay ;02A3 -LD (alienFireDelay),A ... till alien shots are enabled TFR B,A ;02A6 -LD A,B Restore sprite image number (used if we go to 39B) ; ST#$05 ,X ;02A7 -LD (HL),$05 Reload time between blow-up changes LDB #$05 STB ,X+ ; LEAX 1,X ;02A9 -INC HL Point to number of blow-up changes DEC ,X ;02AA -DEC (HL) Count down blow-up changes LBNE DrawPlayerDie ;02AB -JP NZ,DrawPlayerDie Still blowing up ... go draw next sprite ; ; Blow up finished LDD playerYr ;02AE -LD HL,(playerYr) player's coordinates EXG A,B TFR D,X LDB #$10 ;02B1 -LD B,$10 16 Bytes JSR EraseSimpleSprite ;02B3 -CALL EraseSimpleSprite Erase simple sprite (the player) LDX #firstgameobj ;02B6 -LD HL,$2010 Restore player ... LDY #RAMInit+$10 ;02B9 -LD DE,$1B10 ... structure ... LDB #$10 ;02BC -LD B,$10 ... from ... JSR BlockCopy ;02BE -CALL BlockCopy ... ROM mirror LDB #$00 ;02C1 -LD B,$00 Turn off ... JSR SoundBits3Off ;02C3 -CALL SoundBits3Off ... all sounds LDA invaded ;02C6 -LD A,(invaded) Has rack reached ... TSTA ;02C9 -AND A ... the bottom of the screen? BEQ > ;02CA -RET NZ Yes ... done here RTS ! * Jump here LDA gameMode ;02CB -LD A,(gameMode) Are we in ... TSTA ;02CE -AND A ... game mode? BNE > ;02CF -RET Z No ... return to splash screens RTS ! * Jump here LDS #Stack_Pointer ;02D0 -LD SP,$2400 We aren't going to return ANDCC #$EF ;02D3 -EI Enable interrupts (we just dropped the ISR context) enable IRQ JSR DsableGameTasks ;02D4 -CALL DsableGameTasks Disable game tasks JSR L092E ;02D7 -CALL $092E Get number of ships for active player TSTA ;02DA -AND A Any left? LBEQ L166D ;02DB -JP Z,$166D No ... handle game over for player JSR L18E7 ;02DE -CALL $18E7 Get player-alive status pointer LDA ,X ;02E1 -LD A,(HL) Is player ... TSTA ;02E2 -AND A ... alive? LBEQ L032C ;02E3 -JP Z,$032C Yes ... remove a ship from player's stash and reenter game loop LDA twoPlayers ;02E6 -LD A,(twoPlayers) Multi-player game TSTA ;02E9 -AND A Only one player? LBEQ L032C ;02EA -JP Z,$032C Yes ... remove a ship from player's stash and reenter game loop L02ED LDA playerDataMSB ;02ED -LD A,(playerDataMSB) Player data MSB PSHS A,CC ;02F0 -PUSH AF Hold the MSB RORA ;02F1 -RRCA Player 1 is active player? LBCS L0332 ;02F2 -JP C,$0332 Yes ... go store player 1 shields and come back to 02F8 JSR RememberShields2 ;02F5 -CALL RememberShields2 No ... go store player 2 shields L02F8 JSR L0878 ;02F8 -CALL $0878 Get ref-alien info and pointer to storage ; STE ,X ;02FB -LD (HL),E Hold the ... ; LEAX 1,X ;02FC -INC HL ... ref-alien ... ; STD ,X ;02FD -LD (HL),D ... screen coordinates STY ,X ; ******* Maybe the registers are stored backwards intel byte reversal is a pain!!! This might cause an error... ; LEAX -1,X ;02FE -DEC HL Back up ... ; LEAX -1,X ;02FF -DEC HL .. to delta storage STB ,-X ;0300 -LD (HL),B Store ref-alien's delta (direction) ; NOP ;0301 -NOP ** Why? JSR CopyRAMMirror ;0302 -CALL CopyRAMMirror Copy RAM mirror (getting ready to switch players) PULS A,CC ;0305 -POP AF Restore active player MSB RORA ;0306 -RRCA Player 1? LDA #$21 ;0307 -LD A,$21 Player 1 data pointer **** Check value LDB #$00 ;0309 -LD B,$00 Cocktail bit=0 (player 1) LBCC L0312 ;030B -JP NC,$0312 It was player one ... keep data for player 2 LDB #$20 ;030E -LD B,$20 Cocktail bit=1 (player 2) LDA #$22 ;0310 -LD A,$22 Player 2 data pointer **** Check value L0312 STA playerDataMSB ;0312 -LD (playerDataMSB),A Change players JSR TwoSecDelay ;0315 -CALL TwoSecDelay Two second delay CLRA ;0318 -XOR A Clear the player-object ... STA obj0TimerLSB ;0319 -LD (obj0TimerLSB),A ... timer (player can move instantly after switching players) TFR B,A ;031C -LD A,B Cocktail bit to A STA IO_05_SOUND2 ;031D -OUT (SOUND2),A Set the cocktail mode INCA ;031F -INC A Fleet sound 1 (first tone) STA soundPort5 ;0320 -LD (soundPort5),A Set the port 5 hold JSR ClearPlayField ;0323 -CALL ClearPlayField Clear center window JSR RemoveShip ;0326 -CALL RemoveShip Remove a ship and update indicators LBRA L07F9 ;0329 -JP $07F9 Tell the players that the switch has been made ; L032C JSR RemoveShip ;032C -CALL RemoveShip Remove a ship and update indicators LBRA L0817 ;032F -JP $0817 Continue into game loop ; L0332 JSR RememberShields1 ;0332 -CALL RememberShields1 Remember the shields for player 1 LBRA L02F8 ;0335 -JP $02F8 Back to switching-players above ; FCB $00 ;0338 -y ** Why ; Player not blowing up ... handle inputs L033B LDX #playerOK ;033B -LD HL,$2068 Player OK flag ; ST#$01 ,X ;033E -LD (HL),$01 Flag 1 ... player is OK LDA #$01 STA ,X+ ; LEAX 1,X ;0340 -INC HL 2069 LDA ,X ;0341 -LD A,(HL) Alien shots enabled? TSTA ;0342 -AND A Set flags LBRA L03B0 ;0343 -JP $03B0 Continue L0346 ; NOP ;0346 -NOP ** Why? LEAX -1,X ;0347 -DEC HL 2069 ; ST#$01 ,X ;0348 -LD (HL),$01 Enable alien fire LDA #$01 STA ,X L034A LDA playerXr ;034A -LD A,(playerXr) Current player coordinates TFR A,B ;034D -LD B,A Hold it LDA gameMode ;034E -LD A,(gameMode) Are we in ... TSTA ;0351 -AND A ... game mode? LBNE L0363 ;0352 -JP NZ,$0363 Yes ... use switches as player controls ; LDA nextDemoCmd ;0355 -LD A,(nextDemoCmd) Get demo command RORA ;0358 -RRCA Is it right? LBCS MovePlayerRight ;0359 -JP C,MovePlayerRight Yes ... do right RORA ;035C -RRCA Is it left? LBCS L038E ;035D -JP C,$038E Yes ... do left LBRA L036F ;0360 -JP $036F Skip over movement (draw player and out) ; Player is in control L0363 JSR ReadInputs ;0363 -CALL ReadInputs Read active player controls $10=fire, $20=Left,$40=Right ROLA ;0366 -RLCA Test for ... ROLA ;0367 -RLCA ... right button LBCS MovePlayerRight ;0368 -JP C,MovePlayerRight Yes ... handle move right ROLA ;036B -RLCA Test for left button LBCS L038E ;036C -JP C,$038E Yes ... handle move left ; Draw player sprite L036F LDX #plyrSprPicL ;036F -LD HL,$2018 Active player descriptor JSR ReadDesc ;0372 -CALL ReadDesc Load 5 byte sprite descriptor in order: EDLHB JSR ConvToScr ;0375 -CALL ConvToScr Convert HL to screen coordinates JSR DrawSimpSprite ;0378 -CALL DrawSimpSprite Draw player LDA #$00 ;037B -LD A,$00 Clear the task timer. Nobody changes this but it could have ... STA obj0TimerExtra ;037D -LD (obj0TimerExtra),A ... been speed set for the player with a value other than 0 (not XORA) RTS ;0380 -RET Out MovePlayerRight: ; Handle player moving right TFR B,A ;0381 -LD A,B Player coordinate CMPA #$D9 ;0382 -CP $D9 At right edge? LBEQ L036F ;0384 -JP Z,$036F Yes ... ignore this INCA ;0387 -INC A Bump X coordinate STA playerXr ;0388 -LD (playerXr),A New X coordinate LBRA L036F ;038B -JP $036F Draw player and out MovePlayerLeft: ; = ; Handle player moving left L038E TFR B,A ;038E -LD A,B Player coordinate CMPA #$30 ;038F -CP $30 At left edge LBEQ L036F ;0391 -JP Z,$036F Yes ... ignore this DECA ;0394 -DEC A Bump X coordinate STA playerXr ;0395 -LD (playerXr),A New X coordinate LBRA L036F ;0398 -JP $036F Draw player and out DrawPlayerDie: ; Toggle the player's blowing-up sprite between two pictures and draw it INCA ;039B -INC A Toggle blowing-up ... ANDA #$01 ;039C -AND $01 ... player sprite (0,1,0,1) STA playerAlive ;039E -LD (playerAlive),A Hold current state ; ROLA ;03A1 -RLCA *2 ; ROLA ;03A2 -RLCA *4 ; ROLA ;03A3 -RLCA *8 ; ROLA ;03A4 -RLCA *16 LDB #16 MUL LDX #PlrBlowupSprites ;03A5 -LD HL,$1C70 Base blow-up sprite location ; TFR A,B ; B=A ABX ; Add B to X to point to the new blow-up sprite picture ; ADD A=A+L ;03A8 -ADD A,L Offset sprite ... ; TFR A,L ;03A9 -LD L,A ... pointer TFR X,D EXG A,B ; swap the MSB & LSB, to match intel weirdness STD plyrSprPicL ;03AA -LD (plyrSprPicL),HL New blow-up sprite picture LBRA L036F ;03AD -JP $036F Draw new blow-up sprite and out L03B0 LBNE L034A ;03B0 -JP NZ,$034A Alien shots enabled ... move player's ship, draw it, and out LEAX 1,X ;03B3 -INC HL To 206A DEC ,X ;03B4 -DEC (HL) Time until aliens can fire LBNE L034A ;03B5 -JP NZ,$034A Not time to enable ... move player's ship, draw it, and out LBRA L0346 ;03B8 -JP $0346 Enable alien fire ... move player's ship, draw it, and out GameObj1: ; Game object 1: Move/draw the player shot ; ; This task executes at either mid-screen ISR (if it is on the top half of the non-rotated screen) or ; at the end-screen ISR (if it is on the bottom half of the screen). ; ; PULS X ; RTS LDY #obj1CoorXr ;03BB -LD DE,$202A Object's Yn coordiante JSR CompYToBeam ;03BE -CALL CompYToBeam Compare to screen-update location PULS X ;03C1 -POP HL Pointer to task data BCS > ;03C2 -RET NC Make sure we are in the right ISR RTS ! * Jump here LEAX 1,X ;03C3 -INC HL Point to 2025 ... the shot status LDA ,X ;03C4 -LD A,(HL) Get shot status TSTA ;03C5 -AND A Return if ... BNE > ;03C6 -RET Z ... no shot is active RTS ! * Jump here ; CMPA #$01 ;03C7 -CP $01 Shot just starting (requested elsewhere)? LBEQ InitPlyShot ;03C9 -JP Z,InitPlyShot Yes ... go initiate shot ; CMPA #$02 ;03CC -CP $02 Progressing normally? LBEQ MovePlyShot ;03CE -JP Z,MovePlyShot Yes ... go move it ; LEAX 1,X ;03D1 -INC HL 2026 CMPA #$03 ;03D2 -CP $03 Shot blowing up (not because of alien)? LBNE L042A ;03D4 -JP NZ,$042A No ... try other options ; ; Shot blowing up because it left the playfield, hit a shield, or hit another bullet DEC ,X ;03D7 -DEC (HL) Decrement the timer LBEQ EndOfBlowup ;03D8 -JP Z,EndOfBlowup If done then LDA ,X ;03DB -LD A,(HL) Get timer value CMPA #$0F ;03DC -CP $0F Starts at 10 ... first decrement brings us here BEQ > ;03DE -RET NZ Not the first time ... explosion has been drawn RTS ! * Jump here ; Draw explosion first pass through timer loop PSHS X ;03DF -PUSH HL Hold pointer to data JSR ReadPlyShot ;03E0 -CALL ReadPlyShot Read shot descriptor JSR EraseShifted ;03E3 -CALL EraseShifted Erase the sprite PULS X ;03E6 -POP HL 2026 (timer flag) LEAX 1,X ;03E7 -INC HL 2027 point to sprite LSB INC ,X ;03E8 -INC (HL) Change 1C90 to 1C91 LEAX 1,X ;03E9 -INC HL 2028 LEAX 1,X ;03EA -INC HL 2029 DEC ,X ;03EB -DEC (HL) Drop X coordinate ... DEC ,X ;03EC -DEC (HL) ... by 2 LEAX 1,X ;03ED -INC HL 202A DEC ,X ;03EE -DEC (HL) Drop Y ... DEC ,X ;03EF -DEC (HL) ... coordinate ... DEC ,X ;03F0 -DEC (HL) ... by ... LEAX 1,X ;03F1 -INC HL ... 3 ; ST#$08 ,X ;03F2 -LD (HL),$08 202B 8 bytes in size of sprite LDA #$08 ; STA ,X F ; JSR ReadPlyShot ;03F4 -CALL ReadPlyShot Read player shot structure LBRA DrawShiftedSprite ;03F7 -JP DrawShiftedSprite Draw sprite and out ; InitPlyShot: INCA ;03FA -INC A Type is now ... STA ,X ;03FB -LD (HL),A ... 2 (in progress) LDA playerXr ;03FC -LD A,(playerXr) Players Y coordinate ADDA #$08 ;03FF -ADD A,$08 To center of player STA obj1CoorXr ;0401 -LD (obj1CoorXr),A Shot's Y coordinate JSR ReadPlyShot ;0404 -CALL ReadPlyShot Read 5 byte structure LBRA DrawShiftedSprite ;0407 -JP DrawShiftedSprite Draw sprite and out ; MovePlyShot: JSR ReadPlyShot ;040A -CALL ReadPlyShot Read the shot structure PSHS Y ;040D -PUSH DE Hold pointer to sprite image PSHS X ;040E -PUSH HL Hold sprite coordinates PSHS B ;040F -PUSH BC Hold sprite size (in B) JSR EraseShifted ;0410 -CALL EraseShifted Erase the sprite from the screen PULS B ;0413 -POP BC Restore size PULS X ;0414 -POP HL Restore coords PULS Y ;0415 -POP DE Restore pointer to sprite image LDA shotDeltaX ;0416 -LD A,(shotDeltaX) DeltaX for shot HL=6A28, A=peek(202c)=04 STX H_Temp ADDA L_Temp ;0419 -ADD A,L Move the shot ... A=A+L = 04 + 28 = 2C STA L_Temp ;041A -LD L,A ... up the screen L=A = 2C LDX H_Temp STA obj1CoorYr ;041B -LD (obj1CoorYr),A Store shot's new X coordinate poke 2029,2C JSR DrawSprCollision ;041E -CALL DrawSprCollision Draw sprite with collision detection LDA collision ;0421 -LD A,(collision) Test for ... TSTA ;0424 -AND A ... collision BNE > ;0425 -RET Z No collision ... out RTS ! * Jump here ; ; Collision with alien detected STA alienIsExploding ;0426 -LD (alienIsExploding),A Set to not-0 indicating ... RTS ;0429 -RET ... an alien is blowing up ; ; Other shot-status options L042A CMPA #$05 ;042A -CP $05 Alien explosion in progress? BNE > ;042C -RET Z Yes ... nothing to do RTS ! * Jump here LBRA EndOfBlowup ;042D -JP EndOfBlowup Anything else erases the shot and removes it from duty ReadPlyShot: LDX #plyShotStruct ;0430 -LD HL,$2027 Read 5 byte sprite structure for ... LBRA ReadDesc ;0433 -JP ReadDesc ... player shot EndOfBlowup: JSR ReadPlyShot ;0436 -CALL ReadPlyShot Read the shot structure JSR EraseShifted ;0439 -CALL EraseShifted Erase the player's shot LDX #plyrShotStatus ;043C -LD HL,$2025 Reinit ... LDY #RAMInit+$25 ;043F -LD DE,$1B25 ... shot structure ... LDB #$07 ;0442 -LD B,$07 ... from ... JSR BlockCopy ;0444 -CALL BlockCopy ... ROM mirror LDA sauScoreLSB ;0447 -LD HL,(sauScoreLSB) Get pointer to saucer-score table ; EXG A,B INCA ;044A -INC L Every shot explosion advances it one ; TFR X,D ;044B -LD A,L Have we passed ... CMPA #$63 ;044C -CP $63 ... the end at 1D63 (bug! this should be $64 to cover all 16 values) LBCS L0453 ;044E -JP C,$0453 No .... keep it LDA #$54 ;0451 -LD L,$54 Wrap back around to 1D54 L0453 STA sauScoreLSB ; STX sauScoreLSB ;0453 -LD (sauScoreLSB),HL New score pointer LDD shotCountLSB ;0456 -LD HL,(shotCountLSB) Increments with every shot ... INCA ;0459 -INC L ... but only LSB ** ... STD shotCountLSB ;045A -LD (shotCountLSB),HL ... used for saucer direction ; LDA saucerActive ;045D -LD A,(saucerActive) Is saucer ... TSTA ;0460 -AND A ... on screen? BEQ > ;0461 -RET NZ Yes ... don't reset it RTS ! * Jump here ; ; Setup saucer direction for next trip LDA ,X ;0462 -LD A,(HL) Shot counter ; ANDA #$01 ;0463 -AND $01 Lowest bit set? LDU #$0229 ;0465 -LD BC,$0229 Xr delta of 2 starting at Xr=29 ANDA #$01 ; Moved so BNE will be taken from the ANDA and not the LDU command LBNE L046E ;0468 -JP NZ,$046E Yes ... use 2/29 LDU #$FEE0 ;046B -LD BC,$FEE0 No ... Xr delta of -2 starting at Xr=E0 L046E LDX #saucerPriPicMSB ;046E -LD HL,$208A Saucer descriptor TFR U,D STB ,X ;0471 -LD (HL),C Store Xr coordinate LEAX 2,X ;0472 -INC HL Point to ... ; LEAX 1,X ;0473 -INC HL ... delta Xr STA ,X ;0474 -LD (HL),B Store delta Xr RTS ;0475 -RET Done GameObj2: ; Game object 2: Allien rolling-shot (targets player specifically) ; ; The 2-byte value at 2038 is where the firing-column-table-pointer would be (see other ; shots ... next game objects). This shot doesn't use that table. It targets the player ; specifically. Instead the value is used as a flag to have the shot skip its first ; attempt at firing every time it is reinitialized (when it blows up). ; ; The task-timer at 2032 is copied to 2080 in the game loop. The flag is used as a ; synchronization flag to keep all the shots processed on separate interrupt ticks. This ; has the main effect of slowing the shots down. ; ; When the timer is 2 the squiggly-shot/saucer (object 4 ) runs. ; When the timer is 1 the plunger-shot (object 3) runs. ; Whne the timer is 0 this object, the rolling-shot, runs. ; ; PULS X ; RTS PULS X ;0476 -POP HL Game object data LDA RAMInit+$32 ;0477 -LD A,($1B32) Restore delay from ... STA obj2TimerExtra ;047A -LD (obj2TimerExtra),A ... ROM mirror (value 2) LDX rolShotCFirLSB ;047D -LD HL,(rolShotCFirLSB) Get pointer to ... ; TFR L,A ;0480 -LD A,L ... column-firing table. A=L , TFR X,D - which makes B=L ; TFR X,D ; ORA Register - need tweaking H ;0481 -OR H All zeros? ORB with H=rolShotCFirLSB ; ORB rolShotCFirLSB LBNE L048A ;0482 -JP NZ,$048A No ... must be a valid column. Go fire. LEAX -1,X ;0485 -DEC HL Decrement the counter STX rolShotCFirLSB ;0486 -LD (rolShotCFirLSB),HL Store new counter value (run the shot next time) RTS ;0489 -RET And out L048A LDY #rolShotdatstct ;048A -LD DE,$2035 Rolling-shot data structure LDA #$F9 ;048D -LD A,$F9 Last picture of "rolling" alien shot JSR ToShotStruct ;048F -CALL ToShotStruct Set code to handle rolling-shot LDA pluShotStepCnt ;0492 -LD A,(pluShotStepCnt) Get the plunger-shot step count STA otherShot1 ;0495 -LD (otherShot1),A Hold it LDA squShotStepCnt ;0498 -LD A,(squShotStepCnt) Get the squiggly-shot step count STA otherShot2 ;049B -LD (otherShot2),A Hold it JSR HandleAlienShot ;049E -CALL HandleAlienShot Handle active shot structure LDX #rolShotdatstct ;Moved up from a few lines down so the TSTA value is what the BNE is using to decide (not this LDX instruction) LDA aShotBlowCnt ;04A1 -LD A,(aShotBlowCnt) Blow up counter TSTA ;04A4 -AND A Test if shot has cycled through blowing up ; LDX #rolShotdatstct ;04A5 -LD HL,$2035 Rolling-shot data structure LBNE FromShotStruct ;04A8 -JP NZ,FromShotStruct If shot is still running, copy the updated data and out ResetShot: ; The rolling-shot has blown up. Reset the data structure. LDY #RAMInit+$30 ;04AB -LD DE,$1B30 Reload ... LDX #objStruct2 ;04AE -LD HL,$2030 ... object ... LDB #$10 ;04B1 -LD B,$10 ... structure ... LBRA BlockCopy ;04B3 -JP BlockCopy ... from ROM mirror and out GameObj3: ; Game object 3: Alien plunger-shot ; This is skipped if there is only one alien left on the screen. ; ; PULS X ; RTS PULS X ;04B6 -POP HL Game object data LDA skipPlunger ;04B7 -LD A,(skipPlunger) One alien left? Skip plunger shot? TSTA ;04BA -AND A Check BEQ > ;04BB -RET NZ Yes. Only one alien. Skip this shot. RTS ! * Jump here LDA shotSync ;04BC -LD A,(shotSync) Sync flag (copied from GO-2's timer value) CMPA #$01 ;04BF -CP $01 GO-2 and GO-4 are idle? BEQ > ;04C1 -RET NZ No ... only one shot at a time RTS ! * Jump here LDY #pluShotdata ;04C2 -LD DE,$2045 Plunger alien shot data structure LDA #$ED ;04C5 -LD A,$ED Last picture of "plunger" alien shot JSR ToShotStruct ;04C7 -CALL ToShotStruct Copy the plunger alien to the active structure LDA rolShotStepCnt ;04CA -LD A,(rolShotStepCnt) Step count from rolling-shot STA otherShot1 ;04CD -LD (otherShot1),A Hold it LDA squShotStepCnt ;04D0 -LD A,(squShotStepCnt) Step count from squiggly shot STA otherShot2 ;04D3 -LD (otherShot2),A Hold it JSR HandleAlienShot ;04D6 -CALL HandleAlienShot Handle active shot structure LDA aShotCFirLSB ;04D9 -LD A,(aShotCFirLSB) LSB of column-firing table CMPA #$10 ;04DC -CP $10 Been through all entries in the table? LBCS L04E7 ;04DE -JP C,$04E7 Not yet ... table is OK LDA RAMInit+$48 ;04E1 -LD A,($1B48) Been through all .. STA aShotCFirLSB ;04E4 -LD (aShotCFirLSB),A ... so reset pointer into firing-column table L04E7 ; Rearrange LDX first before the TSTA (LD on 8080 don't change the condition codes but 6809 does) LDX #pluShotdata LDA aShotBlowCnt ;04E7 -LD A,(aShotBlowCnt) Get the blow up timer TSTA ;04EA -AND A Zero means shot is done ; LDX #pluShotdata ;04EB -LD HL,$2045 Plunger shot data LBNE FromShotStruct ;04EE -JP NZ,FromShotStruct If shot is still running, go copy the updated data and out ; LDY #RAMInit+$40 ;04F1 -LD DE,$1B40 Reload ... LDX #objStruct3 ;04F4 -LD HL,$2040 ... object ... LDB #$10 ;04F7 -LD B,$10 ... structure ... JSR BlockCopy ;04F9 -CALL BlockCopy ... from mirror ; LDA numAliens ;04FC -LD A,(numAliens) Number of aliens on screen DECA ;04FF -DEC A Is there only one left? LBNE L0508 ;0500 -JP NZ,$0508 No ... move on LDA #$01 ;0503 -LD A,$01 Disable plunger shot ... STA skipPlunger ;0505 -LD (skipPlunger),A ... when only one alien remains L0508 LDD aShotCFirLSB ;0508 -LD HL,(aShotCFirLSB) Set the plunger shot's ... LBRA L067E ;050B -JP $067E ... column-firing pointer data ; Game task 4 when splash screen alien is shooting extra "C" with a squiggly shot AlienShootC: PULS X ;050E -POP HL Ignore the task data pointer passed on stack ; ; GameObject 4 comes here if processing a squiggly shot L050F LDY #squShotStatus ;050F -LD DE,$2055 Squiggly shot data structure LDA #$DB ;0512 -LD A,$DB LSB of last byte of picture JSR ToShotStruct ;0514 -CALL ToShotStruct Copy squiggly shot to LDA pluShotStepCnt ;0517 -LD A,(pluShotStepCnt) Get plunger ... STA otherShot1 ;051A -LD (otherShot1),A ... step count LDA rolShotStepCnt ;051D -LD A,(rolShotStepCnt) Get rolling ... STA otherShot2 ;0520 -LD (otherShot2),A ... step count JSR HandleAlienShot ;0523 -CALL HandleAlienShot Handle active shot structure LDA aShotCFirLSB ;0526 -LD A,(aShotCFirLSB) LSB of column-firing table pointer CMPA #$15 ;0529 -CP $15 Have we processed all entries? LBCS L0534 ;052B -JP C,$0534 No ... don't reset it LDA RAMInit+$58 ;052E -LD A,($1B58) Reset the pointer ... STA aShotCFirLSB ;0531 -LD (aShotCFirLSB),A ... back to the start of the table L0534 ; Rearrange LDX first before the TSTA (LD on 8080 don't change the condition codes but 6809 does) LDX #squShotStatus LDA aShotBlowCnt ;0534 -LD A,(aShotBlowCnt) Check to see if squiggly shot is done TSTA ;0537 -AND A 0 means blow-up timer expired ; LDX #squShotStatus ;0538 -LD HL,$2055 Squiggly shot data structure LBNE FromShotStruct ;053B -JP NZ,FromShotStruct If shot is still running, go copy the updated data and out ; Shot explosion is over. Remove the shot. LDY #RAMInit+$50 ;053E -LD DE,$1B50 Reload LDX #objStruct4 ;0541 -LD HL,$2050 ... object ... LDB #$10 ;0544 -LD B,$10 ... structure ... JSR BlockCopy ;0546 -CALL BlockCopy ... from mirror LDD aShotCFirLSB ;0549 -LD HL,(aShotCFirLSB) Copy pointer to column-firing table ... ; EXG A,B STD squShotCFirLSB ;054C -LD (squShotCFirLSB),HL ... back to data structure (for next shot) RTS ;054F -RET Done ToShotStruct: STA shotPicEnd ;0550 -LD (shotPicEnd),A LSB of last byte of last picture in sprite LDX #aShotStatus ;0553 -LD HL,$2073 Destination is the shot-structure LDB #$0B ;0556 -LD B,$0B 11 bytes LBRA BlockCopy ;0558 -JP BlockCopy Block copy and out FromShotStruct: LDY #aShotStatus ;055B -LD DE,$2073 Source is the shot-structure LDB #$0B ;055E -LD B,$0B 11 bytes LBRA BlockCopy ;0560 -JP BlockCopy Block copy and out HandleAlienShot: ; Each of the 3 shots copy their data to the 2073 structure (0B bytes) and call this. ; Then they copy back if the shot is still active. Otherwise they copy from the mirror. ; ; The alien "fire rate" is based on the number of steps the other two shots on the screen ; have made. The smallest number-of-steps is compared to the reload-rate. If it is too ; soon then no shot is made. The reload-rate is based on the player's score. The MSB ; is looked up in a table to get the reload-rate. The smaller the rate the faster the ; aliens fire. Setting rate this way keeps shots from walking on each other. ; LDX #aShotStatus ;0563 -LD HL,$2073 Start of active shot structure LDA ,X ;0566 -LD A,(HL) Get the shot status ANDA #$80 ;0567 -AND $80 Is the shot active? LBNE L05C1 ;0569 -JP NZ,$05C1 Yes ... go move it LDA isrSplashTask ;056C -LD A,(isrSplashTask) ISR splash task CMPA #$04 ;056F -CP $04 Shooting the "C" ? PSHS CC ;Store the CC so BEQ below uses the correct conditions, 8080 does use CC from a LD instruction LDA enableAlienFire ;0571 -LD A,(enableAlienFire) Alien fire enabled flag PULS CC ;Restore the CC LBEQ L05B7 ;0574 -JP Z,$05B7 We are shooting the extra "C" ... just flag it active and out TSTA ;0577 -AND A Is alien fire enabled? BNE > ;0578 -RET Z No ... don't start a new shot RTS ! * Jump here LEAX 1,X ;0579 -INC HL 2074 step count of current shot CLR ,X ;057A -LD (HL),$00 clear the step count ; Make sure it isn't too soon to fire another shot LDA otherShot1 ;057C -LD A,(otherShot1) Get the step count of the 1st "other shot" TSTA ;057F -AND A Any steps made? LBEQ L0589 ;0580 -JP Z,$0589 No ... ignore this count ; TFR A,B ;0583 -LD B,A Shuffle off step count STA B_Temp ;Store A so we can compare to mem location LDA aShotReloadRate ;0584 -LD A,(aShotReloadRate) Get the reload rate (based on MSB of score) CMPA B_Temp ;0587 -CP B Too soon to fire again? BCS > ;0588 -RET NC Yes ... don't fire RTS ! * Jump here L0589 LDA otherShot2 ;0589 -LD A,(otherShot2) Get the step count of the 2nd "other shot" TSTA ;058C -AND A Any steps made? LBEQ L0596 ;058D -JP Z,$0596 No steps on any shot ... we are clear to fire ; TFR A,B ;0590 -LD B,A Shuffle off step count STA B_Temp ;Store A so we can compare to mem location LDA aShotReloadRate ;0591 -LD A,(aShotReloadRate) Get the reload rate (based on MSB of score) CMPA B_Temp ;0594 -CP B Too soon to fire again? BCS > ;0595 -RET NC Yes ... don't fire RTS ! * Jump here L0596 LEAX 1,X ;0596 -INC HL 2075 LDA ,X ;0597 -LD A,(HL) Get tracking flag TSTA ;0598 -AND A Does this shot track the player? LBEQ L061B ;0599 -JP Z,$061B LDD aShotCFirLSB ;059C -LD HL,(aShotCFirLSB) Column-firing table EXG A,B TFR D,X LDB ,X+ ;059F -LD C,(HL) Get next column to fire from STB C_Temp ; Store B in C_Temp (we don't have a c register) ; LEAX 1,X ;05A0 -INC HL Bump the ... Made increment above sith LDB ,X+ ; NOP ;05A1 -NOP % WHY? TFR X,D EXG A,B STD aShotCFirLSB ;05A2 -LD (aShotCFirLSB),HL ... pointer into column table L05A5 JSR FindInColumn ;05A5 -CALL FindInColumn Find alien in target column BCS > ;05A8 -RET NC No alien is alive in target column ... out RTS ! * Jump here JSR GetAlienCoords ;05A9 -CALL GetAlienCoords Get coordinates of alien (lowest alien in firing column) LDA C_Temp ; TFR C,A ;05AC -LD A,C Offset ... ADDA #$07 ;05AD -ADD A,$07 ... Y by 7 ; TFR A,H ;05AF -LD H,A To H STA H_Temp ; TFR L,A ;05B0 -LD A,L Offset ... LDA L_Temp SUBA #$0A ;05B1 -SUI $0A ... X down 10 ; TFR A,L ;05B3 -LD L,A To L STA L_Temp LDB H_Temp STD alienShotYr ;05B4 -LD (alienShotYr),HL Set shot coordinates below alien ; L05B7 LDX #aShotStatus ;05B7 -LD HL,$2073 Alien shot status LDA ,X ;05BA -LD A,(HL) Get the status ORA #$80 ;05BB -OR $80 Mark this shot ... STA ,X+ ;05BD -LD (HL),A ... as actively running ; LEAX 1,X ;05BE -INC HL 2074 step count Increment X above INC ,X ;05BF -INC (HL) Give this shot 1 step (it just started) RTS ;05C0 -RET Out ; ; Move the alien shot L05C1 LDY #alienShotYc ;05C1 -LD DE,$207C Alien-shot Y coordinate JSR CompYToBeam ;05C4 -CALL CompYToBeam Compare to beam position BCS > ;05C7 -RET NC Not the right ISR for this shot RTS ! * Jump here ; LEAX 1,X ;05C8 -INC HL 2073 status LDA ,X ;05C9 -LD A,(HL) Get shot status ANDA #$01 ;05CA -AND $01 Bit 0 is 1 if blowing up LBNE ShotBlowingUp ;05CC -JP NZ,ShotBlowingUp Go do shot-is-blowing-up sequence LEAX 1,X ;05CF -INC HL 2074 step count INC ,X ;05D0 -INC (HL) Count the steps (used for fire rate) JSR L0675 ;05D1 -CALL $0675 Erase shot LDA aShotImageLSB ;05D4 -LD A,(aShotImageLSB) Get LSB of the image pointer ADDA #$03 ;05D7 -ADD A,$03 Next set of images LDX #shotPicEnd ;05D9 -LD HL,$207F End of image CMPA ,X ;05DC -CP (HL) Have we reached the end of the set? LBCS L05E2 ;05DD -JP C,$05E2 No ... keep it SUBA #$0C ;05E0 -SUI $0C Back up to the 1st image in the set L05E2 STA aShotImageLSB ;05E2 -LD (aShotImageLSB),A New LSB image pointer LDA alienShotYr ;05E5 -LD A,(alienShotYr) Get shot's Y coordinate ; TFR A,B ;05E8 -LD B,A Hold it ; LDA alienShotDelta ;05E9 -LD A,(alienShotDelta) Get alien shot delta ADDA alienShotDelta ;05EC -ADD A,B Add to shots coordinate STA alienShotYr ;05ED -LD (alienShotYr),A New shot Y coordinate JSR L066C ;05F0 -CALL $066C Draw the alien shot LDA alienShotYr ;05F3 -LD A,(alienShotYr) Shot's Y coordinate CMPA #$15 ;05F6 -CP $15 Still in the active playfield? LBCS L0612 ;05F8 -JP C,$0612 No ... end it LDA collision ;05FB -LD A,(collision) Did shot collide ... TSTA ;05FE -AND A ... with something? BNE > ;05FF -RET Z No ... we are done here RTS ! * Jump here LDA alienShotYr ;0600 -LD A,(alienShotYr) Shot's Y coordinate CMPA #$1E ;0603 -CP $1E Is it below player's area? LBCS L0612 ;0605 -JP C,$0612 Yes ... end it CMPA #$27 ;0608 -CP $27 Is it above player's area? ; NOP ;060A -NOP ** WHY? LBCC L0612 ;060B -JP NC,$0612 Yes ... end it CLRA ;060E -SUB A Flag that player ... STA playerAlive ;060F -LD (playerAlive),A ... has been struck ; L0612 LDA aShotStatus ;0612 -LD A,(aShotStatus) Flag to ... ORA #$01 ;0615 -OR $01 ... start shot ... STA aShotStatus ;0617 -LD (aShotStatus),A ... blowing up RTS ;061A -RET Out ; ; Start a shot right over the player L061B LDA playerXr ;061B -LD A,(playerXr) player's X coordinate ADDA #$08 ;061E -ADD A,$08 Center of player STA H_Temp ;0620 -LD H,A To H for routine JSR FindColumn ;0621 -CALL FindColumn Find the column LDA C_Temp ;0624 -LD A,C Get the column right over player CMPA #$0C ;0625 -CP $0C Is it a valid column? LBCS L05A5 ;0627 -JP C,$05A5 Yes ... use what we found LDA #$0B ;062A -LD C,$0B Else use ... STA C_Temp LBRA L05A5 ;062C -JP $05A5 ... as far over as we can FindInColumn: ; C contains the target column. Look for a live alien in the column starting with ; the lowest position. Return C=1 if found ... HL points to found slot. DEC C_Temp ;062F -DEC C Column that is firing LDA playerDataMSB ;0630 -LD A,(playerDataMSB) player's MSB (21xx or 22xx) ; TFR A,H ;0633 -LD H,A To MSB of HL STA H_Temp ; TFR C,L ;0634 -LD L,C Column to L LDA C_Temp STA L_Temp LDA #$05 ;0635 -LD D,$05 5 rows of aliens STA D_Temp L0637 LDA [H_Temp] ;0637 -LD A,(HL) Get alien's status TSTA ;0638 -AND A 0 means dead ; moved below ORCC #$01 ;0639 -SCF In case not 0 BEQ > ;063A -RET NZ Alien is alive? Yes ... return LDX H_Temp ORCC #$01 ;0639 -SCF In case not 0 RTS ! * Jump here LDA L_Temp ;063B -LD A,L Get the flag pointer LSB ADDA #$0B ;063C -ADD A,$0B Jump to same column on next row of rack (+11 aliens per row) STA L_Temp ;063E -LD L,A New alien index DEC D_Temp ;063F -DEC D Tested all rows? LBNE L0637 ;0640 -JP NZ,$0637 No ... keep looking for a live alien up the rack LDX H_Temp ANDCC #$FE RTS ;0643 -RET Didn't find a live alien. Return with C=0. ShotBlowingUp: ; Alien shot is blowing up LDX #aShotBlowCnt ;0644 -LD HL,$2078 Blow up timer DEC ,X ;0647 -DEC (HL) Decrement the value LDA ,X ;0648 -LD A,(HL) Get the value CMPA #$03 ;0649 -CP $03 First tick, 4, we draw the explosion LBNE L0667 ;064B -JP NZ,$0667 After that just wait JSR L0675 ;064E -CALL $0675 Erase the shot LDD #AShotExplo ;0651 -LD HL,$1CDC Alien shot ... EXG A,B ;Swap MSB and LSB to match backward intel order STD aShotImageLSB ;0654 -LD (aShotImageLSB),HL ... explosion sprite LDX #alienShotYc ;0657 -LD HL,$207C Alien shot Y (Probably need to fix this for Intel backwards byte ordering) DEC ,X ;065A -DEC (HL) Left two for ... DEC ,X ;065B -DEC (HL) ... explosion LEAX 1,X ;065C -DEC HL Point slien shot X (Probably need to fix this for Intel backwards byte ordering) DEC ,X ;065D -DEC (HL) Up two for ... DEC ,X ;065E -DEC (HL) ... explosion LDA #$06 ;065F -LD A,$06 Alien shot descriptor ... STA alienShotSize ;0661 -LD (alienShotSize),A ... size 6 LBRA L066C ;0664 -JP $066C Draw alien shot explosion L0667 TSTA ;0667 -AND A Have we reached 0? BEQ > ;0668 -RET NZ No ... keep waiting RTS ! * Jump here LBRA L0675 ;0669 -JP $0675 Erase the explosion and out ; L066C LDX #aShotImageLSB ;066C -LD HL,$2079 Alien shot descriptor JSR ReadDesc ;066F -CALL ReadDesc Read 5 byte structure LBRA DrawSprCollision ;0672 -JP DrawSprCollision Draw shot and out ; L0675 LDX #aShotImageLSB ;0675 -LD HL,$2079 Alien shot descriptor JSR ReadDesc ;0678 -CALL ReadDesc Read 5 byte structure LBRA EraseShifted ;067B -JP EraseShifted Erase the shot and out L067E ; EXG A,B STD pluShotCFirLSB ;067E -LD (pluShotCFirLSB),HL From 50B, update ... RTS ;0681 -RET ... column-firing table pointer and out GameObj4: ; Game object 4: Flying Saucer OR squiggly shot ; ; This task is shared by the squiggly-shot and the flying saucer. The saucer waits until the ; squiggly-shot is over before it begins. ; ; PULS X ; RTS PULS X ;0682 -POP HL Pull data pointer from the stack (not going to use it) LDA shotSync ;0683 -LD A,(shotSync) Sync flag (copied from GO-2's timer value) CMPA #$02 ;0686 -CP $02 Are GO-2 and GO-3 idle? BEQ > ;0688 -RET NZ No ... only one at a time RTS ! * Jump here LDX #saucerStart ;0689 -LD HL,$2083 Time-till-saucer flag LDA ,X ;068C -LD A,(HL) Is it time ... TSTA ;068D -AND A ... for a saucer? LBEQ L050F ;068E -JP Z,$050F No ... go process squiggly shot LDA squShotStepCnt ;0691 -LD A,(squShotStepCnt) Is there a ... TSTA ;0694 -AND A ... squiggly shot going? LBNE L050F ;0695 -JP NZ,$050F Yes ... go handle squiggly shot LEAX 1,X ;0698 -INC HL Saucer on screen flag LDA ,X ;0699 -LD A,(HL) (2084) Is the saucer ... TSTA ;069A -AND A ... already on the screen? LBNE L06AB ;069B -JP NZ,$06AB Yes ... go handle it LDA numAliens ;069E -LD A,(numAliens) Number of aliens remaining CMPA #$08 ;06A1 -CP $08 Less than ... LBCS L050F ;06A3 -JP C,$050F ... 8 ... no saucer LDA #$01 ; store 1 where X points STA ,X ;06A6 -LD (HL),$01 (2084) The saucer is on the screen JSR L073C ;06A8 -CALL $073C Draw the flying saucer L06AB LDY #saucerPriPicMSB ;06AB -LD DE,$208A Saucer's Y coordinate (8080 byte ordering?) JSR CompYToBeam ;06AE -CALL CompYToBeam Compare to beam position BCS > ;06B1 -RET NC Not the right ISR for moving saucer RTS ! * Jump here LDX #saucerHit ;06B2 -LD HL,$2085 Saucer hit flag LDA ,X ;06B5 -LD A,(HL) Has saucer ... TSTA ;06B6 -AND A ... been hit? LBNE L06D6 ;06B7 -JP NZ,$06D6 Yes ... don't move it LDX #saucerPriPicMSB ;06BA -LD HL,$208A Saucer's structure (8080 byte ordering?) LDA ,X ;06BD -LD A,(HL) Get saucer's Y coordinate LEAX 2,X ;06BE -INC HL Bump to ... ; LEAX 1,X ;06BF -INC HL ... delta Y ADDA ,X ;06C0 -ADD A,(HL) Move saucer STA saucerPriPicMSB ;06C1 -LD (saucerPriPicMSB),A New coordinate JSR L073C ;06C4 -CALL $073C Draw the flying saucer LDX #saucerPriPicMSB ;06C7 -LD HL,$208A Saucer's structure (8080 byte ordering?) LDA ,X ;06CA -LD A,(HL) Y coordinate CMPA #$28 ;06CB -CP $28 Too low? End of screen? LBCS L06F9 ;06CD -JP C,$06F9 Yes ... remove from play CMPA #$E1 ;06D0 -CP $E1 Too high? End of screen? LBCC L06F9 ;06D2 -JP NC,$06F9 Yes ... remove from play RTS ;06D5 -RET Done L06D6 LDB #$FE ;06D6 -LD B,$FE Turn off ... JSR SoundBits3Off ;06D8 -CALL SoundBits3Off ... flying saucer sound LEAX 1,X ;06DB -INC HL (2086) show-hit timer DEC ,X ;06DC -DEC (HL) Count down show-hit timer LDA ,X ;06DD -LD A,(HL) Get current value CMPA #$1F ;06DE -CP $1F Starts at 20 ... is this the first tick of show-hit timer? LBEQ L074B ;06E0 -JP Z,$074B Yes ... go show the explosion CMPA #$18 ;06E3 -CP $18 A little later ... LBEQ L070C ;06E5 -JP Z,$070C ... show the score besides the saucer and add it TSTA ;06E8 -AND A Has timer expired? BEQ > ;06E9 -RET NZ No ... let it run RTS ! * Jump here LDB #$EF ;06EA -LD B,$EF 1110_1111 (mask off saucer hit sound) STB B_Temp LDX #soundPort5 ;06EC -LD HL,$2098 Get current ... LDA ,X ;06EF -LD A,(HL) ... value of port 5 sound ANDA B_Temp ;06F0 -AND B Mask off the saucer-hit sound STA ,X ;06F1 -LD (HL),A Set the new value ANDA #$20 ;06F2 -AND $20 All sound off but ... STA IO_05_SOUND2 ;06F4 -OUT (SOUND2),A ... cocktail cabinet bit ; NOP ;06F6 -NOP ** Why ; NOP ;06F7 -NOP ** ; NOP ;06F8 -NOP ** ; L06F9 JSR L0742 ;06F9 -CALL $0742 Covert pixel pos from descriptor to HL screen and shift JSR ClearSmallSprite ;06FC -CALL ClearSmallSprite Clear a one byte sprite at HL LDX #saucerStart ;06FF -LD HL,$2083 Saucer structure LDB #$0A ;0702 -LD B,$0A 10 bytes in saucer structure JSR L075F ;0704 -CALL $075F Re-initialize saucer structure L0707 LDB #$FE ;0707 -LD B,$FE Turn off UFO ... LBRA SoundBits3Off ;0709 -JP SoundBits3Off ... sound and out L070C LDA #$01 ;070C -LD A,$01 Flag the score ... STA adjustScore ;070E -LD (adjustScore),A ... needs updating LDD sauScoreLSB ;0711 -LD HL,(sauScoreLSB) Saucer score table EXG A,B TFR D,X LDB ,X ;0714 -LD B,(HL) Get score for this saucer STB B_Temp LDA #$04 ;0715 -LD C,$04 There are only 4 possibilities STA C_Temp LDX #TabCorStrPrnt ;0717 -LD HL,$1D50 Possible scores table LDY #TabPosSaScor ;071A -LD DE,$1D4C Print strings for each score L071D LDA ,Y ;071D -LD A,(DE) Find ... CMPA B_Temp ;071E -CP B ... the ... LBEQ L0728 ;071F -JP Z,$0728 ... print ... LEAX 1,X ;0722 -INC HL ... string ... LEAY 1,Y ;0723 -INC DE ... for ... DEC C_Temp ;0724 -DEC C ... the ... LBNE L071D ;0725 -JP NZ,$071D ... score L0728 LDA ,X ;0728 -LD A,(HL) Get LSB of message (MSB is 2088 which is 1D) STA saucerPriLocLSB ;0729 -LD (saucerPriLocLSB),A Message's LSB (_50=1D94 100=1D97 150=1D9A 300=1D9D) ; LDH #$00 ;072C -LD H,$00 MSB = 0 ... H=0 ; TFR B,L ;072E -LD L,B HL = B HL=B = B_Temp , Multiply B*16, so we can use MUL instruction LDB B_Temp LDA #$10 ;A = 16 ; ADD X=X+X ;072F -ADD HL,HL *2 ; ADD X=X+X ;0730 -ADD HL,HL *4 ; ADD X=X+X ;0731 -ADD HL,HL *8 ; ADD X=X+X ;0732 -ADD HL,HL *16 MUL ; Muliply B * 16 EXG A,B STD scoreDeltaLSB ;0733 -LD (scoreDeltaLSB),HL Add score for hitting saucer (015 becomes 150 in BCD). Store D instead of X at scoreDeltaLSB JSR L0742 ;0736 -CALL $0742 Get the flying saucer score descriptor LBRA L08F1 ;0739 -JP $08F1 Print the three-byte score and out L073C JSR L0742 ;073C -CALL $0742 Draw the ... LBRA DrawSimpSprite ;073F -JP DrawSimpSprite ... flying saucer L0742 LDX #saucerPriLocLSB ;0742 -LD HL,$2087 Read flying saucer ... JSR ReadDesc ;0745 -CALL ReadDesc ... structure LBRA ConvToScr ;0748 -JP ConvToScr Convert pixel number to screen and shift and out ; L074B LDB #$10 ;074B -LD B,$10 Saucer hit sound bit LDX #soundPort5 ;074D -LD HL,$2098 Current state of sounds LDA ,X ;0750 -LD A,(HL) OR ... ORA #$10 ;0751 -OR B ... in ... STA ,X ;0752 -LD (HL),A ... saucer-hit sound JSR L1770 ;0753 -CALL $1770 Turn off fleet sound and start saucer-hit LDD #SpriteSaucerExp ;0756 -LD HL,$1D7C Sprite for saucer blowing up EXG A,B ;0759 -LD (saucerPriLocLSB),HL Store it in structure STD saucerPriLocLSB LBRA L073C ;075C -JP $073C Draw the flying saucer ; L075F LDY #DataSaucer ;075F -LD DE,$1B83 Data for saucer (702 sets count to 0A) LBRA BlockCopy ;0762 -JP BlockCopy Reset saucer object data WaitForStart: ; Wait for player 1 start button press LDA #$01 ;0765 -LD A,$01 Tell ISR that we ... STA waitStartLoop ;0767 -LD (waitStartLoop),A ... have started to wait LDS #Stack_Pointer ;076A -LD SP,$2400 Reset stack ANDCC #$EF ;076D -EI Enable interrupts JSR L1979 ;076E -CALL $1979 Suspend game tasks JSR ClearPlayField ;0771 -CALL ClearPlayField Clear center window LDX #Screen_Start+$C13 ;0774 -LD HL,$3013 Screen coordinates LDY #MessagePush ;0777 -LD DE,$1FF3 "PRESS" LDB #$04 ;077A -LD C,$04 Message length STB C_Temp JSR PrintMessage ;077C -CALL PrintMessage Print it L077F LDA numCoins ;077F -LD A,(numCoins) Number of credits ; DECA ;0782 -DEC A Set flags LDX #Screen_Start+$410 ;0783 -LD HL,$2810 Screen coordinates LDB #$14 ;0786 -LD C,$14 Message length STB C_Temp DECA ; Moved Down from a few lines up so the BNE will be effected by the DECA and not the LDB or STB as the 8080 code LBNE L0857 ;0788 -JP NZ,$0857 Take 1 or 2 player start LDY #Message1Only ;078B -LD DE,$1ACF "ONLY 1PLAYER BUTTON " JSR PrintMessage ;078E -CALL PrintMessage Print message ; LDA IO_01_INP1 ;0791 -IN A,(INP1) Read player controls (Added code for 1 player button below) ; ANDA #$04 ;0793 -AND $04 1Player start button? ; LBEQ L077F ;0795 -JP Z,$077F No ... wait for button or credit LDX #PIA0_Byte_0_KeyRow_Joy * PIA0 LDA #%11111101 * bit 1 STA $02,X * LDA ,X * BITA #%00010000 * and bit 4 = 1 = 1 Player Start Button BNE L077F ;============================================================= NewGame: ; START NEW GAME ; ; 1 Player start LDB #$99 ;0798 -LD B,$99 Essentially a -1 for DAA STB B_Temp CLRA ;079A -XOR A Clear two player flag ; ; 2 player start sequence enters here with a=1 and B=98 (-2) L079B STA twoPlayers ;079B -LD (twoPlayers),A Set flag for 1 or 2 players LDA numCoins ;079E -LD A,(numCoins) Number of credits ADDA B_Temp ;07A1 -ADD A,B Take away credits DAA ;07A2 -DAA Convert back to DAA STA numCoins ;07A3 -LD (numCoins),A New credit count JSR DrawNumCredits ;07A6 -CALL DrawNumCredits Display number of credits LDX #$0000 ;07A9 -LD HL,$0000 Score of 0000 STX P1ScorL ;07AC -LD (P1ScorL),HL Clear player-1 score STX P2ScorL ;07AF -LD (P2ScorL),HL CLear player-2 score JSR L1925 ;07B2 -CALL $1925 Print player-1 score JSR L192B ;07B5 -CALL $192B Print player-2 score JSR DsableGameTasks ;07B8 -CALL DsableGameTasks Disable game tasks LDD #$0101 ;07BB -LD HL,$0101 Two bytes 1, 1 ; TFR H,A ;07BE -LD A,H 1 to A STA gameMode ;07BF -LD (gameMode),A 20EF=1 ... game mode STD player1Alive ;07C2 -LD (player1Alive),HL 20E7 and 20E8 both one ... players 1 and 2 are alive STD player1Ex ;07C5 -LD (player1Ex),HL Extra-ship is available for player-1 and player-2 JSR DrawStatus ;07C8 -CALL DrawStatus Print scores and credits JSR DrawShieldPl1 ;07CB -CALL DrawShieldPl1 Draw shields for player-1 JSR DrawShieldPl2 ;07CE -CALL DrawShieldPl2 Draw shields for player-2 JSR GetShipsPerCred ;07D1 -CALL GetShipsPerCred Get number of ships from DIP settings STA p1ShipsRem ;07D4 -LD (p1ShipsRem),A Player-1 ships STA p2ShipsRem ;07D7 -LD (p2ShipsRem),A Player-2 ships JSR L00D7 ;07DA -CALL $00D7 Set player-1 and player-2 alien racks going right CLRA ;07DD -XOR A Make a 0 STA p1RackCnt ;07DE -LD (p1RackCnt),A Player 1 is on first rack of aliens STA p2RackCnt ;07E1 -LD (p2RackCnt),A Player 2 is on first rack of aliens JSR InitAliens ;07E4 -CALL InitAliens Initialize 55 aliens for player 1 JSR InitAliensP2 ;07E7 -CALL InitAliensP2 Initialize 55 aliens for player 2 LDX #Screen_Start+$1478 ;07EA -LD HL,$3878 Screen coordinates for lower-left alien STX p1RefAlienY ;07ED -LD (p1RefAlienY),HL Initialize reference alien for player 1 STX p2RefAlienYr ;07F0 -LD (p2RefAlienYr),HL Initialize reference alien for player 2 JSR CopyRAMMirror ;07F3 -CALL CopyRAMMirror Copy ROM mirror to RAM (2000 - 20C0) JSR RemoveShip ;07F6 -CALL RemoveShip Initialize ship hold indicator ; L07F9 JSR PromptPlayer ;07F9 -CALL PromptPlayer Prompt with "PLAY PLAYER " JSR ClearPlayField ;07FC -CALL ClearPlayField Clear the playfield ; NOP ;07FF -NOP % Why? CLRA ;0800 -XOR A Make a 0 STA isrSplashTask ;0801 -LD (isrSplashTask),A Disable isr splash-task animation L0804 JSR DrawBottomLine ;0804 -CALL DrawBottomLine Draw line across screen under player LDA playerDataMSB ;0807 -LD A,(playerDataMSB) Current player RORA ;080A -RRCA Right bit tells all LBCS L0872 ;080B -JP C,$0872 Go do player 1 ; JSR RestoreShields2 ;080E -CALL RestoreShields2 Restore shields for player 2 JSR DrawBottomLine ;0811 -CALL DrawBottomLine Draw line across screen under player L0814 JSR InitRack ;0814 -CALL InitRack Initialize alien rack for current player L0817 JSR EnableGameTasks ;0817 -CALL EnableGameTasks Enable game tasks in ISR LDB #$20 ;081A -LD B,$20 Enable ... JSR SoundBits3On ;081C -CALL SoundBits3On ... sound amplifier ; ; GAME LOOP ; L081F JSR PlrFireOrDemo ;081F -CALL PlrFireOrDemo Initiate player shot if button pressed JSR PlyrShotAndBump ;0822 -CALL PlyrShotAndBump Collision detect player's shot and rack-bump JSR CountAliens ;0825 -CALL CountAliens Count aliens (count to 2082) JSR AdjustScore ;0828 -CALL AdjustScore Adjust score (and print) if there is an adjustment LDA numAliens ;082B -LD A,(numAliens) Number of live aliens TSTA ;082E -AND A All aliens gone? LBEQ L09EF ;082F -JP Z,$09EF Yes ... end of turn JSR AShotReloadRate ;0832 -CALL AShotReloadRate Update alien-shot-rate based on player's score JSR L0935 ;0835 -CALL $0935 Check (and handle) extra ship award JSR SpeedShots ;0838 -CALL SpeedShots Adjust alien shot speed JSR ShotSound ;083B -CALL ShotSound Shot sound on or off with 2025 JSR L0A59 ;083E -CALL $0A59 Check if player is hit LBEQ L0849 ;0841 -JP Z,$0849 No hit ... jump handler LDB #$04 ;0844 -LD B,$04 Player hit sound JSR SoundBits3On ;0846 -CALL SoundBits3On Make explosion sound L0849 JSR FleetDelayExShip ;0849 -CALL FleetDelayExShip Extra-ship sound timer, set fleet-delay, play fleet movement sound STA IO_06_WATCHDOG ;084C -OUT (WATCHDOG),A Feed the watchdog JSR CtrlSaucerSound ;084E -CALL CtrlSaucerSound Control saucer sound LBRA L081F ;0851 -JP $081F Continue game loop ; FCB $00 ;0854 - ** Why? ; Test for 1 or 2 player start button press L0857 LDY #MessageB1or2 ;0857 -LD DE,$1ABA "1 OR 2PLAYERS BUTTON" JSR PrintMessage ;085A -CALL PrintMessage Print message LDB #$98 ;085D -LD B,$98 -2 (take away 2 credits) STB B_Temp ; LDA IO_01_INP1 ;085F -IN A,(INP1) Read player controls (Added Code below to handle keyboard inputs) ; RORA ;0861 -RRCA Test ... ; RORA ;0862 -RRCA ... bit 2 ; LBCS L086D ;0863 -JP C,$086D 2 player button pressed ... do it ; RORA ;0866 -RRCA Test bit 3 ; LBCS NewGame ;0867 -JP C,NewGame One player start ... do it LDX #PIA0_Byte_0_KeyRow_Joy * PIA0 LDA #%11111101 * bit 1 STA $02,X * LDA ,X * BITA #%00010000 * and bit 4 = 1 = 1 Player Start Button LBEQ NewGame LDA #%11111011 * bit 2 STA $02,X * LDA ,X * BITA #%00010000 * and bit 4 = 2 = 2 Player Start Button BEQ L086D LBRA L077F ;086A -JP $077F Keep waiting on credit or button ; 2 PLAYER START L086D LDA #$01 ;086D -LD A,$01 Flag 2 player game LBRA L079B ;086F -JP $079B Continue normal startup L0872 JSR RestoreShields1 ;0872 -CALL RestoreShields1 Restore shields for player 1 LBRA L0814 ;0875 -JP $0814 Continue in game loop L0878 LDA refAlienDXr ;0878 -LD A,(refAlienDXr) Alien deltaY TFR A,B ;087B -LD B,A Hold it PSHS D LDD refAlienYr ;087C -LD HL,(refAlienYr) Alien coordinates EXG A,B EXG Y,D ;087F -EX DE,HL Coordinates to DE PULS D LBRA GetAlRefPtr ;0880 -JP GetAlRefPtr HL is 21FC or 22FC and out ; FCB $00 ;0883 - ** Why? GetAlRefPtr: ; Get pointer to player's alien ref coordiantes LDA playerDataMSB ;0886 -LD A,(playerDataMSB) Player data MSB (21 or 22) PSHS B ; TFR A,H ;0889 -LD H,A To H ; LDL #$FC ;088A -LD L,$FC 21FC or 22FC ... alien coordinates LDB #$FC TFR D,X PULS B RTS ;088C -RET Done PromptPlayer: ; Print "PLAY PLAYER " and blink score for 2 seconds. ; LDX #Screen_Start+$711 ;088D -LD HL,$2B11 Screen coordinates LDY #MesssageP1 ;0890 -LD DE,$1B70 Message "PLAY PLAYER<1>" LDB #$0E ;0893 -LD C,$0E 14 bytes in message STB C_Temp JSR PrintMessage ;0895 -CALL PrintMessage Print the message LDA playerDataMSB ;0898 -LD A,(playerDataMSB) Get the player number RORA ;089B -RRCA C will be set for player 1 LDA #$1C ;089C -LD A,$1C The "2" character LDX #Screen_Start+$1311 ;089E -LD HL,$3711 Replace the "<1>" with "<2"> BCS L08A4 ;08A1 - If player 2 ... change the message JSR DrawChar ;**** -CALL NC,DrawChar Added for 6809 worakaround L08A4 LDA #$B0 ;08A4 -LD A,$B0 Delay of 176 (roughly 2 seconds) STA isrDelay ;08A6 -LD (isrDelay),A Set the ISR delay value ; L08A9 LDA isrDelay ;08A9 -LD A,(isrDelay) Get the ISR delay value TSTA ;08AC -AND A Has the 2 second delay expired? BNE > ;08AD -RET Z Yes ... done RTS ! * Jump here ANDA #$04 ;08AE -AND $04 Every 4 ISRs ... LBNE L08BC ;08B0 -JP NZ,$08BC ... flash the player's score JSR L09CA ;08B3 -CALL $09CA Get the score descriptor for the active player JSR DrawScore ;08B6 -CALL DrawScore Draw the score LBRA L08A9 ;08B9 -JP $08A9 Back to the top of the wait loop ; L08BC LDB #$20 ;08BC -LD B,$20 32 rows (4 characters * 8 bytes each) LDX #Screen_Start+$031C ;08BE -LD HL,$271C Player-1 score on the screen LDA playerDataMSB ;08C1 -LD A,(playerDataMSB) Get the player number RORA ;08C4 -RRCA C will be set for player 1 LBCS L08CB ;08C5 -JP C,$08CB We have the right score coordinates LDX #Screen_Start+$151C ;08C8 -LD HL,$391C Use coordinates for player-2's score L08CB JSR ClearSmallSprite ;08CB -CALL ClearSmallSprite Clear a one byte sprite at HL LBRA L08A9 ;08CE -JP $08A9 Back to the top of the wait loop GetShipsPerCred: ; Get number of ships from DIP settings LDA IO_02_INP2 ;08D1 -IN A,(INP2) DIP settings ANDA #$03 ;08D3 -AND $03 Get number of ships ADDA #$03 ;08D5 -ADD A,$03 From 3-6 RTS ;08D7 -RET Out SpeedShots: ; With less than 9 aliens on the screen the alien shots get a tad bit faster. Probably ; because the advancing rack can catch them. ; LDA numAliens ;08D8 -LD A,(numAliens) Number of aliens on screen CMPA #$09 ;08DB -CP $09 More than 8? BCS > ;08DD -RET NC Yes ... leave shot speed alone RTS ! * Jump here LDA #$FB ;08DE -LD A,$FB Normally FF (-1) ... now FB (-4) STA alienShotDelta ;08E0 -LD (alienShotDelta),A Speed up alien shots RTS ;08E3 -RET Done L08E4 LDA twoPlayers ;08E4 -LD A,(twoPlayers) Number of players TSTA ;08E7 -AND A Skip if ... BEQ > ;08E8 -RET NZ ... two player RTS ! * Jump here LDX #Screen_Start+$151C ;08E9 -LD HL,$391C Player 2's score LDB #$20 ;08EC -LD B,$20 32 rows is 4 digits * 8 rows each LBRA ClearSmallSprite ;08EE -JP ClearSmallSprite Clear a one byte sprite (32 rows long) at HL L08F1 LDB #$03 ;08F1 -LD C,$03 Length of saucer-score message ... fall into print STB C_Temp PrintMessage: ;Pr1nt a message on the screen ; HL = coordinates ; DE = message buffer ; C = length ; LDA ,Y ;08F3 -LD A,(DE) Get character ; PSHS Y ;08F4 -PUSH DE Preserve ; JSR DrawChar ;08F5 -CALL DrawChar Print character ; PULS Y ;08F8 -POP DE Restore ; LEAY 1,Y ;08F9 -INC DE Next character ; DEC C_Temp ;08FA -DEC C All done? ; LBNE PrintMessage ;08FB -JP NZ,PrintMessage Print all of message ; RTS ;08FE -RET Out LDA ,Y ;08F3 -LD A,(DE) Get character PSHS Y ;08F4 -PUSH DE Preserve JSR DrawChar ;08F5 -CALL DrawChar Print character PULS Y ;08F8 -POP DE Restore LEAY 1,Y ;08F9 -INC DE Next character DEC C_Temp ;08FA -DEC C All done? LBNE PrintMessage ;08FB -JP NZ,PrintMessage Print all of message RTS ;08FE -RET Out ;============================================================= DrawChar: ; Get pointer to 8 byte sprite number in A and ; draw sprite on screen at HL ; LDY #Characters ;08FF -LD DE,$1E00 Character set ; PSHS X ;0902 -PUSH HL Preserve ; LDH #$00 ;0903 -LD H,$00 MSB=0 (A=0, not needed because it is loaded with 8 before MUL) ; TFR A,L ;0905 -LD L,A Character number to L ; ADD X=X+X ;0906 -ADD HL,HL HL = HL *2 ; ADD X=X+X ;0907 -ADD HL,HL *4 ; ADD X=X+X ;0908 -ADD HL,HL *8 (8 bytes each) ; ADD X=X+Y ;0909 -ADD HL,DE Get pointer to sprite ; EXG Y,D ;090A -EX DE,HL Now into DE ; PULS X ;090B -POP HL Restore HL ; LDB #$08 ;090C -LD B,$08 8 bytes each ; STA IO_06_WATCHDOG ;090E -OUT (WATCHDOG),A Feed watchdog ; LBRA DrawSimpSprite ;0910 -JP DrawSimpSprite To screen LDY #Characters ;08FF -LD DE,$1E00 Character set PSHS X ;0902 -PUSH HL Preserve TFR A,B LDA #$08 MUL LEAY D,Y PULS X ;090B -POP HL Restore HL LDB #$08 ;090C -LD B,$08 8 bytes each STA IO_06_WATCHDOG ;090E -OUT (WATCHDOG),A Feed watchdog LBRA DrawSimpSprite ;0910 -JP DrawSimpSprite To screen TimeToSaucer: LDA refAlienYr ;0913 -LD A,(refAlienYr) Reference alien's X coordinate CMPA #$78 ;0916 -CP $78 Don't process saucer timer ... ($78 is 1st rack Yr) BCS > ;0918 -RET NC ... unless aliens are closer to bottom RTS ! * Jump here LDD tillSaucerLSB ;0919 -LD HL,(tillSaucerLSB) Time to saucer ; TFR L,A ;091C -LD A,L Is it time ... ; ORA tillSaucerLSB ;091D -OR H ... for a saucer LBNE L0929 ;091E -JP NZ,$0929 No ... skip flagging LDD #$0600 ;0921 -LD HL,$0600 Reset timer to 600 game loops EXG A,B ; Deal with Intel byte swapping STD tillSaucerLSB ; LDA #$01 ;0924 -LD A,$01 Flag a ... ; STA saucerStart ;0926 -LD (saucerStart),A ... saucer sequence CLR saucerStart INC saucerStart L0929 EXG A,B ; Deal with Intel byte swapping SUBD #$0001 ;0929 -DEC HL Decrement the ... EXG A,B ; Deal with Intel byte swapping STD tillSaucerLSB ;092A -LD (tillSaucerLSB),HL ... time-to-saucer RTS ;092D -RET Done ;============================================================= ; Get number of ships for acive player L092E JSR GetPlayerDataPtr ;092E -CALL GetPlayerDataPtr HL points to player data TFR X,D LDB #$FF ;0931 -LD L,$FF Last byte = numbe of ships TFR D,X LDA ,X ;0933 -LD A,(HL) Get number of ships RTS ;0934 -RET Done ;============================================================= ; Award extra ship if score has reached ceiling L0935 JSR CurPlyAlive ;0935 -CALL CurPlyAlive Get descriptor of sorts LEAX -1,X ;0938 -DEC HL Back up ... LEAX -1,X ;0939 -DEC HL ... two bytes LDA ,X ;093A -LD A,(HL) Has extra ship ... TSTA ;093B -AND A already been awarded? BNE > ;093C -RET Z Yes ... ignore RTS ! * Jump here LDB #$15 ;093D -LD B,$15 Default 1500 LDA IO_02_INP2 ;093F -IN A,(INP2) Read DIP settings ANDA #$08 ;0941 -AND $08 Extra ship at 1000 or 1500 LBEQ L0948 ;0943 -JP Z,$0948 0=1500 LDB #$10 ;0946 -LD B,$10 Awarded at 1000 L0948 JSR L09CA ;0948 -CALL $09CA Get score descriptor for active player LEAX 1,X ;094B -INC HL MSB of score ... LDA ,X ;094C -LD A,(HL) ... to accumulator STB B_Temp CMPA B_Temp ;094D -CP B Time for an extra ship? BCC > ;094E -RET C No ... out RTS ! * Jump here JSR L092E ;094F -CALL $092E Get pointer to number of ships INC ,X ;0952 -INC (HL) Bump number of ships LDA ,X ;0953 -LD A,(HL) Get the new total PSHS A,CC ;0954 -PUSH AF Hang onto it for a bit LDX #Screen_Start+$0101 ;0955 -LD HL,$2501 Screen coords for ship hold L0958 ;INC H ;0958 -INC H Bump to ... ; INC H ;0959 -INC H ... next LEAX $200,X DECA ;095A -DEC A ... spot LBNE L0958 ;095B -JP NZ,$0958 Find spot for new ship LDB #$10 ;095E -LD B,$10 16 byte sprite LDY #PlayerSprite ;0960 -LD DE,$1C60 Player sprite JSR DrawSimpSprite ;0963 -CALL DrawSimpSprite Draw the sprite PULS CC,A ;0966 -POP AF Restore the count INCA ;0967 -INC A +1 JSR L1A8B ;0968 -CALL $1A8B Print the number of ships JSR CurPlyAlive ;096B -CALL CurPlyAlive Get descriptor for active player of some sort LEAX -1,X ;096E -DEC HL Back up ... LEAX -1,X ;096F -DEC HL ... two bytes CLR ,X ;0970 -LD (HL),$00 Flag extra ship has been awarded LDA #$FF ;0972 -LD A,$FF Set timer ... STA extraHold ;0974 -LD (extraHold),A ... for extra-ship sound LDB #$10 ;0977 -LD B,$10 Make sound ... LBRA SoundBits3On ;0979 -JP SoundBits3On ... for extra man AlienScoreValue: LDX #AlienScores ;097C -LD HL,$1DA0 Table for scores for hitting alien CMPA #$02 ;097F -CP $02 0 or 1 (lower two rows) ... BCC > ;0981 -RET C ... return HL points to value 10 RTS ! * Jump here LEAX 1,X ;0982 -INC HL next value CMPA #$04 ;0983 -CP $04 2 or 3 (middle two rows) ... BCC > ;0985 -RET C ... return HL points to value 20 RTS ! * Jump here LEAX 1,X ;0986 -INC HL Top row ... RTS ;0987 -RET ... return HL points to value 30 AdjustScore: ; Adjust the score for the active player. 20F1 is 1 if there is a new value to add. ; The adjustment is in 20F2,20F3. Then print the score. JSR L09CA ;0988 -CALL $09CA Get score structure for active player LDA adjustScore ;098B -LD A,(adjustScore) Does the score ... TSTA ;098E -AND A ... need increasing? BNE > ;098F -RET Z No ... done RTS ! * Jump here CLRA ;0990 -XOR A Mark score ... STA adjustScore ;0991 -LD (adjustScore),A ... as adjusted PSHS X ;0994 -PUSH HL Hold the pointer to the structure LDD scoreDeltaLSB ;0995 -LD HL,(scoreDeltaLSB) Get requested adjustment EXG A,B STD D_Temp ;0998 -EX DE,HL Adjustment to DE PULS X ;0999 -POP HL Get back pointer to structure LDA ,X ;099A -LD A,(HL) Add adjustment ... ADDA E_Temp ;099B -ADD A,E ... first byte DAA ;099C -DAA Adjust it for BCD STA ,X ;099D -LD (HL),A Store new LSB STA E_Temp ;099E -LD E,A Add adjustment ... LEAX 1,X ;099F -INC HL ... to ... LDA ,X ;09A0 -LD A,(HL) ... second ... ADCA D_Temp ;09A1 -ADC A,D ... byte DAA ;09A2 -DAA Adjust for BCD (cary gets dropped) STA ,X ;09A3 -LD (HL),A Store second byte STA D_Temp ;09A4 -LD D,A Second byte to D (first byte still in E) LEAX 1,X ;09A5 -INC HL Load ... LDA ,X ;09A6 -LD A,(HL) ... the ... LEAX 1,X ;09A7 -INC HL ... screen ... LDB ,X ;09A8 -LD H,(HL) ... coordinates ... STB H_Temp STA L_Temp ;09A9 -LD L,A ... to HL LDX H_Temp LDY D_Temp LBRA Print4Digits ;09AA -JP Print4Digits ** Usually a good idea, but wasted here Print4Digits: ; Print 4 digits in DE STY D_Temp LDA D_Temp ;09AD -LD A,D Get first 2 digits of BCD or hex JSR DrawHexByte ;09AE -CALL DrawHexByte Print them LDA E_Temp ;09B1 -LD A,E Get second 2 digits of BCD or hex (fall into print) DrawHexByte: ; Display 2 digits in A to screen at HL PSHS Y ;09B2 -PUSH DE Preserve PSHS A,CC ;09B3 -PUSH AF Save for later RORA ;09B4 -RRCA Get ... RORA ;09B5 -RRCA ... RORA ;09B6 -RRCA ... RORA ;09B7 -RRCA ... left digit ANDA #$0F ;09B8 -AND $0F Mask out lower digit's bits JSR L09C5 ;09BA -CALL $09C5 To screen at HL PULS CC,A ;09BD -POP AF Restore digit ANDA #$0F ;09BE -AND $0F Mask out upper digit JSR L09C5 ;09C0 -CALL $09C5 To screen PULS Y ;09C3 -POP DE Restore RTS ;09C4 -RET Done ; L09C5 ADDA #$1A ;09C5 -ADD A,$1A Bump to number characters LBRA DrawChar ;09C7 -JP DrawChar Continue ... ; Get score descriptor for active player L09CA LDA playerDataMSB ;09CA -LD A,(playerDataMSB) Get active player RORA ;09CD -RRCA Test for player LDX #P1ScorL ;09CE -LD HL,$20F8 Player 1 score descriptor BCC > ;09D1 -RET C Keep it if player 1 is active RTS ! * Jump here LDX #P2ScorL ;09D2 -LD HL,$20FC Else get player 2 descriptor RTS ;09D5 -RET Out ClearPlayField: ; Clear center window of screen ; LDX #Screen_Start+2 ;09D6 -LD HL,$2402 Thrid from left, top of screen ;L09D9 CLR ,X ;09D9 -LD (HL),$00 Clear screen byte ; LEAX 1,X ;09DB -INC HL Next in row ; TFR L,A ;09DC -LD A,L Get X ... ; TFR X,D ; ANDB #$1F ;09DD -AND $1F ... coordinate ; CMPB #$1C ;09DF -CP $1C Edge minus a buffer? ; LBCS L09E8 ;09E1 -JP C,$09E8 No ... keep going ; LDY #$0006 ;09E4 -LD DE,$0006 Else ... bump to ; ADD X=X+Y ;09E7 -ADD HL,DE ... next edge + buffer ; LEAX $6,X ;L09E8 TFR H,A ;09E8 -LD A,H Get Y coordinate ; CMPA #$40 ;09E9 -CP $40 Reached bottom? ; LBCS L09D9 ;09EB -JP C,$09D9 No ... keep going ; RTS ;09EE -RET Done LDX #Screen_Start+2 ;09D6 -LD HL,$2402 Thrid from left, top of screen L09D9 CLR ,X+ ;09D9 -LD (HL),$00 Clear screen byte TFR X,D ANDB #$1F ;09DD -AND $1F ... coordinate CMPB #$1C ;09DF -CP $1C Edge minus a buffer? LBCS L09E8 ;09E1 -JP C,$09E8 No ... keep going LEAX $6,X L09E8 CMPA #$40 ;09E9 -CP $40 Reached bottom? LBCS L09D9 ;09EB -JP C,$09D9 No ... keep going RTS ;09EE -RET Done L09EF JSR L0A3C ;09EF -CALL $0A3C CLRA ;09F2 -XOR A Suspend ... STA suspendPlay ;09F3 -LD (suspendPlay),A ... ISR game tasks JSR ClearPlayField ;09F6 -CALL ClearPlayField Clear playfield LDA playerDataMSB ;09F9 -LD A,(playerDataMSB) Hold current player number ... PSHS A,CC ;09FC -PUSH AF ... on stack JSR CopyRAMMirror ;09FD -CALL CopyRAMMirror Block copy RAM mirror from ROM PULS A,CC ;0A00 -POP AF Restore ... STA playerDataMSB ;0A01 -LD (playerDataMSB),A ... current player number ; LDA playerDataMSB ;0A04 -LD A,(playerDataMSB) ** Why load this again? Nobody ever jumps to 0A04? ; TFR A,H ;0A07 -LD H,A To H STX H_Temp STA H_Temp LDX H_Temp PSHS X ;0A08 -PUSH HL Hold player-data pointer ; LDL #$FE ;0A09 -LD L,$FE 2xFE ... rack count LDA #$FE STA L_Temp LDX H_Temp LDA ,X ;0A0B -LD A,(HL) Get the number of racks the player has beaten ANDA #$07 ;0A0C -AND $07 0-7 INCA ;0A0E -INC A Now 1-8 STA ,X ;0A0F -LD (HL),A Update count since player just beat a rack LDX #AlienStartTable-$1 ;0A10 -LD HL,$1DA2 Starting coordinate of alien table L0A13 LEAX 1,X ;0A13 -INC HL Find the ... DECA ;0A14 -DEC A ... right entry ... LBNE L0A13 ;0A15 -JP NZ,$0A13 ... in the table LDA ,X ;0A18 -LD A,(HL) Get the starting Y coordiante PULS X ;0A19 -POP HL Restore player's pointer STX H_Temp ; LDL #$FC ;0A1A -LD L,$FC 2xFC ... LDB #$FC STB L_Temp LDX H_Temp ; STA ,X+ ;0A1C -LD (HL),A Set rack's starting Y coordinate ; LEAX 1,X ;0A1D -INC HL Point to X LDB #$38 ;0A1E -LD (HL),$38 Set rack's starting X coordinate to 38 STB ,X+ STA ,X ; TFR H,A ;0A20 -LD A,H Player ... TFR X,D RORA ;0A21 -RRCA ... number to carry LBCS L0A33 ;0A22 -JP C,$0A33 2nd player stuff LDA #$21 ;0A25 -LD A,$21 Start fleet with ... STA soundPort5 ;0A27 -LD (soundPort5),A ... first sound JSR DrawShieldPl2 ;0A2A -CALL DrawShieldPl2 Draw shields for player 2 JSR InitAliensP2 ;0A2D -CALL InitAliensP2 Initalize aliens for player 2 LBRA L0804 ;0A30 -JP $0804 Continue at top of game loop ; L0A33 JSR DrawShieldPl1 ;0A33 -CALL DrawShieldPl1 Draw shields for player 1 JSR InitAliens ;0A36 -CALL InitAliens Initialize aliens for player 1 LBRA L0804 ;0A39 -JP $0804 Continue at top of game loop ; L0A3C JSR L0A59 ;0A3C -CALL $0A59 Check player collision LBNE L0A52 ;0A3F -JP NZ,$0A52 Player is not alive ... skip delay LDA #$30 ;0A42 -LD A,$30 Half second delay STA isrDelay ;0A44 -LD (isrDelay),A Set ISR timer L0A47 LDA isrDelay ;0A47 -LD A,(isrDelay) Has timer expired? TSTA ;0A4A -AND A Check exipre BNE > ;0A4B -RET Z Out if done RTS ! * Jump here JSR L0A59 ;0A4C -CALL $0A59 Check player collision LBEQ L0A47 ;0A4F -JP Z,$0A47 No collision ... wait on timer L0A52 JSR L0A59 ;0A52 -CALL $0A59 Wait for ... LBNE L0A52 ;0A55 -JP NZ,$0A52 ... collision to end RTS ;0A58 -RET Done ; Check to see if player is hit L0A59 LDA playerAlive ;0A59 -LD A,(playerAlive) Active player hit flag CMPA #$FF ;0A5C -CP $FF All FFs means player is OK RTS ;0A5E -RET Out ScoreForAlien: ; Start the hit-alien sound and flag the adjustment for the score. ; B contains the row, which determines the score value. LDA gameMode ;0A5F -LD A,(gameMode) Are we in ... TSTA ;0A62 -AND A ... game mode? LBEQ L0A7C ;0A63 -JP Z,$0A7C No ... skip scoring in demo ; TFR B,C ;0A66 -LD C,B Hold row number STB C_Temp LDB #$08 ;0A67 -LD B,$08 Alien hit sound JSR SoundBits3On ;0A69 -CALL SoundBits3On Enable sound ; TFR C,B ;0A6C -LD B,C Restore row number LDB C_Temp TFR B,A ;0A6D -LD A,B Into A JSR AlienScoreValue ;0A6E -CALL AlienScoreValue Look up the score for the alien LDA ,X ;0A71 -LD A,(HL) Get the score value LDX #pointscrdelta ;0A72 -LD HL,$20F3 Pointer to score delta CLR ,X ;0A75 -LD (HL),$00 Upper byte of score delta is "00" LEAX -1,X ;0A77 -DEC HL Point to score delta LSB STA ,X ;0A78 -LD (HL),A Set score for hitting alien LEAX -1,X ;0A79 -DEC HL Point to adjust-score-flag CLR ,X ;0A7A -LD (HL),$01 The score will get changed elsewhere INC ,X L0A7C LDX #exploAliendes ;0A7C -LD HL,$2062 Return exploding-alien descriptor RTS ;0A7F -RET Out Animate: ; Start the ISR moving the sprite. Return when done. LDA #$02 ;0A80 -LD A,$02 Start simple linear ... STA isrSplashTask ;0A82 -LD (isrSplashTask),A ... sprite animation (splash) L0A85 STA IO_06_WATCHDOG ;0A85 -OUT (WATCHDOG),A Feed watchdog LDA splashReached ;0A87 -LD A,(splashReached) Has the ... TSTA ;0A8A -AND A ... sprite reached target? LBEQ L0A85 ;0A8B -JP Z,$0A85 No ... wait CLRA ;0A8E -XOR A Stop ... STA isrSplashTask ;0A8F -LD (isrSplashTask),A ... ISR animation RTS ;0A92 -RET Done PrintMessageDel: ; Print message from DE to screen at HL (length in C) with a ; delay between letters. PSHS Y ;0A93 -PUSH DE Preserve LDA ,Y ;0A94 -LD A,(DE) Get character JSR DrawChar ;0A95 -CALL DrawChar Draw character on screen PULS Y ;0A98 -POP DE Preserve LDA #$07 ;0A99 -LD A,$07 Delay between letters STA isrDelay ;0A9B -LD (isrDelay),A Set counter L0A9E LDA isrDelay ;0A9E -LD A,(isrDelay) Get counter DECA ;0AA1 -DEC A Is it 1? LBNE L0A9E ;0AA2 -JP NZ,$0A9E No ... wait on it LEAY 1,Y ;0AA5 -INC DE Next in message DEC C_Temp ;0AA6 -DEC C All done? LBNE PrintMessageDel ;0AA7 -JP NZ,PrintMessageDel No ... do all RTS ;0AAA -RET Out SplashSquiggly: LDX #objStruct4 ;0AAB -LD HL,$2050 Pointer to game-object 4 timer LBRA L024B ;0AAE -JP $024B Process squiggly-shot in demo mode OneSecDelay: ; Delay 64 interrupts LDA #$40 ;0AB1 -LD A,$40 Delay of 64 (tad over 1 sec) LBRA WaitOnDelay ;0AB3 -JP $0AD7 Do delay TwoSecDelay: ; Delay 128 interrupts LDA #$80 ;0AB6 -LD A,$80 Delay of 80 (tad over 2 sec) LBRA WaitOnDelay ;0AB8 -JP $0AD7 Do delay SplashDemo: PULS X ;0ABB -POP HL Drop the call to ABF and ... LBRA L0072 ;0ABC -JP $0072 ... do a demo game loop without sound ISRSplTasks: ; Different types of splash tasks managed by ISR in splash screens. The ISR ; calls this if in splash-mode. These may have been bit flags to allow all 3 ; at the same time. Maybe it is just easier to do a switch with a rotate-to-carry. ; LDA isrSplashTask ;0ABF -LD A,(isrSplashTask) Get the ISR task number RORA ;0AC2 -RRCA In demo play mode? LBCS SplashDemo ;0AC3 -JP C,SplashDemo 1: Yes ... go do game play (without sound) RORA ;0AC6 -RRCA Moving little alien from point A to B? LBCS SplashSprite ;0AC7 -JP C,SplashSprite 2: Yes ... go move little alien from point A to B RORA ;0ACA -RRCA Shooting extra "C" with squiggly shot? LBCS SplashSquiggly ;0ACB -JP C,SplashSquiggly 4: Yes ... go shoot extra "C" in splash RTS ;0ACE -RET No task to do ; Message to center of screen. ; Only used in one place for "SPACE INVADERS" L0ACF LDX #Screen_Start+$714 ;0ACF -LD HL,$2B14 Near center of screen LDB #$0F ;0AD2 -LD C,$0F 15 bytes in message STB C_Temp LBRA PrintMessageDel ;0AD4 -JP PrintMessageDel Print and out WaitOnDelay: ; = ; Wait on ISR counter to reach 0 L0AD7 STA isrDelay ;0AD7 -LD (isrDelay),A Delay counter L0ADA LDA isrDelay ;0ADA -LD A,(isrDelay) Get current delay TSTA ;0ADD -AND A Zero yet? LBNE L0ADA ;0ADE -JP NZ,$0ADA No ... wait on it RTS ;0AE1 -RET Out IniSplashAni: ; Init the splash-animation block LDX #splashAnForm ;0AE2 -LD HL,$20C2 The splash-animation descriptor LDB #$0C ;0AE5 -LD B,$0C C bytes LBRA BlockCopy ;0AE7 -JP BlockCopy Block copy DE to descriptor ;============================================================= ; After initialization ... splash screens L0AEA CLRA ;0AEA -XOR A Make a 0 STA IO_03_SOUND1 ;0AEB -OUT (SOUND1),A Turn off sound STA IO_05_SOUND2 ;0AED -OUT (SOUND2),A Turn off sound JSR L1982 ;0AEF -CALL $1982 Turn off ISR splash-task ANDCC #$EF ;0AF2 -EI Enable interrupts (using them for delays) JSR OneSecDelay ;0AF3 -CALL OneSecDelay One second delay ; LDA splashAnimate ;0AF6 -LD A,(splashAnimate) Splash screen type ; TST splashAnimate down below and do BNE based on that check as the 8080 doesn't modify the CC while doing an LD ; TSTA ;0AF9 -AND A Set flags based on type LDX #Screen_Start+$C17 ;0AFA -LD HL,$3017 Screen coordinates (middle near top) ; LDC #$04 ;0AFD -LD C,$04 4 characters in "PLAY" LDA #$04 STA C_Temp TST splashAnimate LBNE L0BE8 ;0AFF -JP NZ,$0BE8 Not 0 ... do "normal" PLAY LDY #MessagePlayUY ;0B02 -LD DE,$1CFA The "PLAy" with an upside down 'Y' JSR PrintMessageDel ;0B05 -CALL PrintMessageDel Print the "PLAy" LDY #MessageInvaders ;0B08 -LD DE,$1DAF "SPACE INVADERS" message L0B0B JSR L0ACF ;0B0B -CALL $0ACF Print to middle-ish of screen JSR OneSecDelay ;0B0E -CALL OneSecDelay One second delay JSR DrawAdvTable ;0B11 -CALL DrawAdvTable Draw "SCORE ADVANCE TABLE" with print delay JSR TwoSecDelay ;0B14 -CALL TwoSecDelay Two second delay LDA splashAnimate ;0B17 -LD A,(splashAnimate) Do splash ... TSTA ;0B1A -AND A ... animations? LBNE L0B4A ;0B1B -JP NZ,$0B4A Not 0 ... no animations ; ; Animate small alien replacing upside-down Y with correct Y LDY #SplashScrnAnm ;0B1E -LD DE,$1A95 Animate sprite from Y=FE to Y=9E step -1 JSR IniSplashAni ;0B21 -CALL IniSplashAni Copy to splash-animate structure JSR Animate ;0B24 -CALL Animate Wait for ISR to move sprite (small alien) LDY #AlienSprCYA1 ;0B27 -LD DE,$1BB0 Animate sprite from Y=98 to Y=FF step 1 JSR IniSplashAni ;0B2A -CALL IniSplashAni Copy to splash-animate structure JSR Animate ;0B2D -CALL Animate Wait for ISR to move sprite (alien pulling upside down Y) JSR OneSecDelay ;0B30 -CALL OneSecDelay One second delay LDY #SplshScrnAnm3 ;0B33 -LD DE,$1FC9 Animate sprite from Y=FF to Y=97 step 1 JSR IniSplashAni ;0B36 -CALL IniSplashAni Copy to splash-animate structure JSR Animate ;0B39 -CALL Animate Wait for ISR to move sprite (alien pushing Y) JSR OneSecDelay ;0B3C -CALL OneSecDelay One second delay LDX #Screen_Start+$FB7 ;0B3F -LD HL,$33B7 Where the splash alien ends up LDB #$0A ;0B42 -LD B,$0A 10 rows JSR ClearSmallSprite ;0B44 -CALL ClearSmallSprite Clear a one byte sprite at HL JSR TwoSecDelay ;0B47 -CALL TwoSecDelay Two second delay ; ; Play demo L0B4A JSR ClearPlayField ;0B4A -CALL ClearPlayField Clear playfield LDA p1ShipsRem ;0B4D -LD A,(p1ShipsRem) Number of ships for player-1 TSTA ;0B50 -AND A If non zero ... LBNE L0B5D ;0B51 -JP NZ,$0B5D ... keep it (counts down between demos) JSR GetShipsPerCred ;0B54 -CALL GetShipsPerCred Get number of ships from DIP settings STA p1ShipsRem ;0B57 -LD (p1ShipsRem),A Reset number of ships for player-1 JSR RemoveShip ;0B5A -CALL RemoveShip Remove a ship from stash and update indicators ; L0B5D JSR CopyRAMMirror ;0B5D -CALL CopyRAMMirror Block copy ROM mirror to initialize RAM JSR InitAliens ;0B60 -CALL InitAliens Initialize all player 1 aliens JSR DrawShieldPl1 ;0B63 -CALL DrawShieldPl1 Draw shields for player 1 (to buffer) JSR RestoreShields1 ;0B66 -CALL RestoreShields1 Restore shields for player 1 (to screen) LDA #$01 ;0B69 -LD A,$01 ISR splash-task ... STA isrSplashTask ;0B6B -LD (isrSplashTask),A ... playing demo JSR DrawBottomLine ;0B6E -CALL DrawBottomLine Draw playfield line L0B71 JSR PlrFireOrDemo ;0B71 -CALL PlrFireOrDemo In demo ... process demo movement and always fire JSR L0BF1 ;0B74 -CALL $0BF1 Check player shot and aliens bumping edges of screen and hidden message STA IO_06_WATCHDOG ;0B77 -OUT (WATCHDOG),A Feed watchdog JSR L0A59 ;0B79 -CALL $0A59 Has demo player been hit? LBEQ L0B71 ;0B7C -JP Z,$0B71 No ... continue game CLRA ;0B7F -XOR A Remove player shot ... STA plyrShotStatus ;0B80 -LD (plyrShotStatus),A ... from activity L0B83 JSR L0A59 ;0B83 -CALL $0A59 Wait for demo player ... LBNE L0B83 ;0B86 -JP NZ,$0B83 ... to stop exploding ; ; Credit information L0B89 CLRA ;0B89 -XOR A Turn off ... STA isrSplashTask ;0B8A -LD (isrSplashTask),A ... splash animation JSR OneSecDelay ;0B8D -CALL OneSecDelay One second delay JSR L1988 ;0B90 -CALL $1988 ** Something else at one time? Jump straight to clear-play-field LDB #$0C ;0B93 -LD C,$0C Message size STB C_Temp LDX #Screen_Start+$811 ;0B95 -LD HL,$2C11 Screen coordinates LDY #MessageCoin ;0B98 -LD DE,$1F90 "INSERT COIN" JSR PrintMessage ;0B9B -CALL PrintMessage Print message LDA splashAnimate ;0B9E -LD A,(splashAnimate) Do splash ... CMPA #$00 ;0BA1 -CP $00 ... animations? LBNE L0BAE ;0BA3 -JP NZ,$0BAE Not 0 ... not on this screen LDX #Screen_Start+$F11 ;0BA6 -LD HL,$3311 Screen coordinates LDA #$02 ;0BA9 -LD A,$02 Character "C" JSR DrawChar ;0BAB -CALL DrawChar Put an extra "C" for "CCOIN" on the screen L0BAE LDU #CreditTable ;0BAE -LD BC,$1F9C "<1 OR 2 PLAYERS> " JSR ReadPriStruct ;0BB1 -CALL ReadPriStruct Load the screen,pointer JSR L184C ;0BB4 -CALL $184C Print the message LDA IO_02_INP2 ;0BB7 -IN A,(INP2) Display coin info (bit 7) ... ROLA ;0BB9 -RLCA ... on demo screen? LBCS L0BC3 ;0BBA -JP C,$0BC3 1 means no ... skip it LDU #CreditTable1 ;0BBD -LD BC,$1FA0 "*1 PLAYER 1 COIN " JSR L183A ;0BC0 -CALL $183A Load the descriptor L0BC3 JSR TwoSecDelay ;0BC3 -CALL TwoSecDelay Print TWO descriptors worth LDA splashAnimate ;0BC6 -LD A,(splashAnimate) Doing splash ... CMPA #$00 ;0BC9 -CP $00 ... animation? LBNE L0BDA ;0BCB -JP NZ,$0BDA Not 0 ... not on this screen LDY #SplshScrnAnm4 ;0BCE -LD DE,$1FD5 Animation for small alien to line up with extra "C" JSR IniSplashAni ;0BD1 -CALL IniSplashAni Copy the animation block JSR Animate ;0BD4 -CALL Animate Wait for the animation to complete JSR L189E ;0BD7 -CALL $189E Animate alien shot to extra "C" L0BDA LDX #splashAnimate ;0BDA -LD HL,$20EC Toggle ... LDA ,X ;0BDD -LD A,(HL) ... the ... INCA ;0BDE -INC A ... splash screen ... ANDA #$01 ;0BDF -AND $01 ... animation for ... STA ,X ;0BE1 -LD (HL),A ... next time JSR ClearPlayField ;0BE2 -CALL ClearPlayField Clear play field LBRA L18DF ;0BE5 -JP $18DF Keep splashing L0BE8 LDY #MessagePlayY ;0BE8 -LD DE,$1DAB "PLAY" with normal 'Y' JSR PrintMessageDel ;0BEB -CALL PrintMessageDel Print it LBRA L0B0B ;0BEE -JP $0B0B Continue with splash (HL will be pointing to next message) L0BF1 JSR PlyrShotAndBump ;0BF1 -CALL PlyrShotAndBump Check if player is shot and aliens bumping the edge of screen LBRA CheckHiddenMes ;0BF4 -JP CheckHiddenMes Check for hidden-message display sequence MessageCorp: ; "TAITO COP" ;0BF7: FCB $13,$00,$08,$13,$0E,$26,$02,$0E,$0F DrawShiftedSprite: ; These next Sprite routines will proably need to be optimized the most to speed up the game on the CoCo ; The only differences between this and EraseSimpleSprite is two CPL instructions in the latter and ; the use of AND instead of OR. NOP takes the same amount of time/space as CPL. So the two NOPs ; here make these two parallel routines the same size and speed. ; NOP ;1400 -NOP Time/size pad to match CPL in EraseShiftedSprite JSR CnvtPixNumber ;1401 -CALL CnvtPixNumber Convert pixel number to coord and shift NOP ;1404 -NOP Time/size pad to match CPL in EraseShiftedSprite L1405 PSHS B ;1405 -PUSH BC Hold count PSHS X ;1406 -PUSH HL Hold start coordinate LDA ,Y ;1407 -LD A,(DE) Get the picture bits JSR IO_04_SHFT_DATA ;1408 -OUT (SHFT_DATA),A Store in shift register JSR IO_03_SHFT_IN ;140A -IN A,(SHFT_IN) Read the shifted pixels PSHS X LDX #ReverseLookUp LDA A,X PULS X ORA ,X ;140C -OR (HL) OR them onto the screen STA ,X ;140D -LD (HL),A Store them back to screen LEAX 1,X ;140E -INC HL Next colummn on screen LEAY 1,Y ;140F -INC DE Next in picture CLRA ;1410 -XOR A Shift over ... JSR IO_04_SHFT_DATA ;1411 -OUT (SHFT_DATA),A ... to next byte in register (shift in 0) JSR IO_03_SHFT_IN ;1413 -IN A,(SHFT_IN) Read the shifted pixels PSHS X LDX #ReverseLookUp LDA A,X PULS X ORA ,X ;1415 -OR (HL) OR them onto the screen STA ,X ;1416 -LD (HL),A Store them back to screen PULS X ;1417 -POP HL Restore starting coordinate ; LDU #$0020 ;1418 -LD BC,$0020 Add 32 ... ; ADD X=X+U ;141B -ADD HL,BC ... to coordinate (move to next row) LEAX $20,X PULS B ;141C -POP BC Restore count DECB ;141D -DEC B All done? LBNE L1405 ;141E -JP NZ,$1405 No ... go do all rows RTS ;1421 -RET Done ; FCB $00 ;1422 - ** Why? EraseSimpleSprite: ; Clear a sprite from the screen (standard pixel number descriptor). ; ** We clear 2 bytes even though the draw-simple only draws one. JSR CnvtPixNumber ;1424 -CALL CnvtPixNumber Convert pixel number in HL L1427 PSHS B ;1427 -PUSH BC Hold PSHS X ;1428 -PUSH HL Hold CLRA ;1429 -XOR A 0 STA ,X ;142A -LD (HL),A Clear screen byte LEAX 1,X ;142B -INC HL Next byte STA ,X ;142C -LD (HL),A Clear byte LEAX 1,X ;142D -INC HL ** Is this to mimic timing? We increment then pop PULS X ;142E -POP HL Restore screen coordinate ; LDU #$0020 ;142F -LD BC,$0020 Add 1 row ... ; ADD X=X+U ;1432 -ADD HL,BC ... to screen coordinate LEAX $20,X PULS B ;1433 -POP BC Restore counter DECB ;1434 -DEC B All rows done? LBNE L1427 ;1435 -JP NZ,$1427 Do all rows RTS ;1438 -RET out DrawSimpSprite: ; Display character to screen ; HL = screen coordinates ; DE = character data ; B = number of rows ; PSHS B ;1439 -PUSH BC Preserve counter ; LDA ,Y ;143A -LD A,(DE) From character set ... ; STA ,X ;143B -LD (HL),A ... to screen ; LEAY 1,Y ;143C -INC DE Next in character set ; LDU #$0020 ;143D -LD BC,$0020 Next row ... ; ADD X=X+U ;1440 -ADD HL,BC ... on screen ; PULS B ;1441 -POP BC Restore counter ; DECB ;1442 -DEC B Decrement counter ; LBNE DrawSimpSprite ;1443 -JP NZ,DrawSimpSprite Do all ; RTS ;1446 -RET Out ;We have to reverse the order of the A value loaded takes 84 cylces using RORA & ROL Memory, 52 cycles using RORA and ROLB ;Using Reverse Look Up table takes 21 cycles ! PSHS X LDA ,Y ;143A -LD A,(DE) From character set ... LDX #ReverseLookUp LDA A,X PULS X STA ,X ;143B -LD (HL),A ... to screen LEAY 1,Y ;143C -INC DE Next in character set LEAX $20,X DECB ;1442 -DEC B Decrement counter LBNE < ;1443 -JP NZ,DrawSimpSprite Do all RTS ;1446 -RET Out ; FCB $00 ;1447 -0 00 00 00 00 00 ; ** Why? ** Why? EraseShifted: ; Erases a shifted sprite from screen (like for player's explosion) JSR CnvtPixNumber ;1452 -CALL CnvtPixNumber Convert pixel number in HL to coorinates with shift L1455 PSHS B ;1455 -PUSH BC Hold BC PSHS X ;1456 -PUSH HL Hold coordinate LDA ,Y ;1457 -LD A,(DE) Get picture value JSR IO_04_SHFT_DATA ;1458 -OUT (SHFT_DATA),A Value into shift register JSR IO_03_SHFT_IN ;145A -IN A,(SHFT_IN) Read shifted sprite picture PSHS X LDX #ReverseLookUp LDA A,X PULS X COMA ;145C -CPL Reverse it (erasing bits) ANDA ,X ;145D -AND (HL) Erase the bits from the screen STA ,X ;145E -LD (HL),A Store the erased pattern back LEAX 1,X ;145F -INC HL Next column on screen LEAY 1,Y ;1460 -INC DE Next in image CLRA ;1461 -XOR A Shift register over ... JSR IO_04_SHFT_DATA ;1462 -OUT (SHFT_DATA),A ... 8 bits (shift in 0) JSR IO_03_SHFT_IN ;1464 -IN A,(SHFT_IN) Read 2nd byte of image PSHS X LDX #ReverseLookUp LDA A,X PULS X COMA ;1466 -CPL Reverse it (erasing bits) ANDA ,X ;1467 -AND (HL) Erase the bits from the screen STA ,X ;1468 -LD (HL),A Store the erased pattern back PULS X ;1469 -POP HL Restore starting coordinate ; LDU #$0020 ;146A -LD BC,$0020 Add 32 ... ; ADD X=X+U ;146D -ADD HL,BC ... to next row LEAX $20,X PULS B ;146E -POP BC Restore BC (count) DECB ;146F -DEC B All rows done? LBNE L1455 ;1470 -JP NZ,$1455 No ... erase all RTS ;1473 -RET Done CnvtPixNumber: ; Convert pixel number in HL to screen coordinate and shift amount. ; HL gets screen coordinate. ; Hardware shift-register gets amount. ; TFR L,A ;1474 -LD A,L Get X coordinate STX H_Temp LDA L_Temp ANDA #$07 ;1475 -AND $07 Shift by pixel position JSR IO_02_SHFTAMNT ;1477 -OUT (SHFTAMNT),A Write shift amount to hardware LBRA ConvToScr ;1479 -JP ConvToScr HL = HL/8 + 2000 (screen coordinate) RememberShields: ; In a multi-player game the player's shields are block-copied to and from RAM between turns. ; HL = screen pointer ; DE = memory buffer ; B = number of rows ; C = number of columns PSHS U ;147C -PUSH BC Hold counter STU B_Temp ; save B_Temp & C_Temp PSHS X ;147D -PUSH HL Hold start L147E LDA ,X ;147E -LD A,(HL) From sprite ... (should be DE) STA ,Y ;147F -LD (DE),A ... to screen ... (should be HL) LEAY 1,Y ;1480 -INC DE Next in sprite LEAX 1,X ;1481 -INC HL Next on screen DEC C_Temp ;1482 -DEC C All columns done? LBNE L147E ;1483 -JP NZ,$147E No ... do multi columns PULS X ;1486 -POP HL Restore screen start ; LDU #$0020 ;1487 -LD BC,$0020 Add 32 ... ; ADD X=X+U ;148A -ADD HL,BC ... to get to next row LEAX $20,X PULS U ;148B -POP BC Pop the counters STU B_Temp ; save B_Temp & C_Temp DEC B_Temp ;148C -DEC B All rows done? ; LBNE RememberShields ;148D -JP NZ,RememberShields No ... do multi rows ; RTS ;1490 -RET Done BEQ > LDU B_Temp BRA RememberShields ! RTS DrawSprCollision: JSR CnvtPixNumber ;1491 -CALL CnvtPixNumber Convert pixel number to coord and shift CLRA ;1494 -XOR A Clear the ... STA collision ;1495 -LD (collision),A ... collision-detection flag L1498 PSHS B ;1498 -PUSH BC Hold count PSHS X ;1499 -PUSH HL Hold screen LDA ,Y ;149A -LD A,(DE) Get byte JSR IO_04_SHFT_DATA ;149B -OUT (SHFT_DATA),A Write first byte to shift register JSR IO_03_SHFT_IN ;149D -IN A,(SHFT_IN) Read shifted pattern PSHS X LDX #ReverseLookUp LDA A,X PULS X PSHS A,CC ;149F -PUSH AF Hold the pattern ANDA ,X ;14A0 -AND (HL) Any bits from pixel collide with bits on screen? LBEQ L14A9 ;14A1 -JP Z,$14A9 No ... leave flag alone LDA #$01 ;14A4 -LD A,$01 Yes ... set ... STA collision ;14A6 -LD (collision),A ... collision flag L14A9 PULS A,CC ;14A9 -POP AF Restore the pixel pattern ORA ,X ;14AA -OR (HL) OR it onto the screen STA ,X ;14AB -LD (HL),A Store new screen value LEAX 1,X ;14AC -INC HL Next byte on screen LEAY 1,Y ;14AD -INC DE Next in pixel pattern CLRA ;14AE -XOR A Write zero ... JSR IO_04_SHFT_DATA ;14AF -OUT (SHFT_DATA),A ... to shift register JSR IO_03_SHFT_IN ;14B1 -IN A,(SHFT_IN) Read 2nd half of shifted sprite PSHS X LDX #ReverseLookUp LDA A,X PULS X PSHS A,CC ;14B3 -PUSH AF Hold pattern ANDA ,X ;14B4 -AND (HL) Any bits from pixel collide with bits on screen? LBEQ L14BD ;14B5 -JP Z,$14BD No ... leave flag alone LDA #$01 ;14B8 -LD A,$01 Yes ... set ... STA collision ;14BA -LD (collision),A ... collision flag L14BD PULS A,CC ;14BD -POP AF Restore the pixel pattern ORA ,X ;14BE -OR (HL) OR it onto the screen STA ,X ;14BF -LD (HL),A Store new screen pattern PULS X ;14C0 -POP HL Starting screen coordinate ; LDU #$0020 ;14C1 -LD BC,$0020 Add 32 ... ; ADD X=X+U ;14C4 -ADD HL,BC ... to get to next row LEAX $20,X PULS B ;14C5 -POP BC Restore count DECB ;14C6 -DEC B All done? LBNE L1498 ;14C7 -JP NZ,$1498 No ... do all rows RTS ;14CA -RET Done ClearSmallSprite: ; Clear a one byte sprite at HL. B=number of rows. CLRA ;14CB -XOR A 0 L14CC ;PSHS U ;14CC -PUSH BC Preserve BC STA ,X ;14CD -LD (HL),A Clear screen byte ; LDU #$0020 ;14CE -LD BC,$0020 Bump HL ... ; ADD X=X+U ;14D1 -ADD HL,BC ... one screen row LEAX $20,X ;PULS U ;14D2 -POP BC Restore DECB ;14D3 -DEC B All done? LBNE L14CC ;14D4 -JP NZ,$14CC No ... clear all RTS ;14D7 -RET Done PlayerShotHit: ; The player's shot hit something (or is being removed from play) ; LDA plyrShotStatus ;14D8 -LD A,(plyrShotStatus) Player shot flag CMPA #$05 ;14DB -CP $05 Alien explosion in progress? BNE > ;14DD -RET Z Yes ... ignore this function RTS ! * Jump here CMPA #$02 ;14DE -CP $02 Normal movement? BEQ > ;14E0 -RET NZ No ... out RTS ! * Jump here ; LDA obj1CoorYr ;14E1 -LD A,(obj1CoorYr) Get Yr coordinate of player shot CMPA #$D8 ;14E4 -CP $D8 Compare to 216 (40 from Top-rotated) TFR A,B ;14E6 -LD B,A Hold value for later LBCC L1530 ;14E7 -JP NC,$1530 Yr is within 40 from top initiate miss-explosion (shot flag 3) LDA alienIsExploding ;14EA -LD A,(alienIsExploding) Is an alien ... TSTA ;14ED -AND A ... blowing up? BNE > ;14EE -RET Z No ... out RTS ! * Jump here ; TFR B,A ;14EF -LD A,B Get original Yr coordinate back to A CMPA #$CE ;14F0 -CP $CE Compare to 206 (50 from rotated top) LBCC L1579 ;14F2 -JP NC,$1579 Yr is within 50 from top? Yes ... saucer must be hit ADDA #$06 ;14F5 -ADD A,$06 Offset to coordinate for wider "explosion" picture TFR A,B ;14F7 -LD B,A Hold that LDA refAlienYr ;14F8 -LD A,(refAlienYr) Ref alien Y coordianate ; If the lower 4 rows are all empty then the reference alien's Y coordinate will wrap around from 0 to F8. ; At this point the top row of aliens is in the shields and we will assume that everything is within ; the rack. CMPA #$90 ;14FB -CP $90 This is true if ... LBCC CodeBug1 ;14FD -JP NC,CodeBug1 ... aliens are down in the shields STB B_Temp CMPA B_Temp ;1500 -CP B Compare to shot's coordinate LBCC L1530 ;1501 -JP NC,$1530 Outside the rack-square ... do miss explosion CodeBug1: ; ; We get here if the player's shot hit something within the rack area (a shot or an alien). ; Find the alien that is (or would be) where the shot hit. If there is no alien alive at the row/column ; thn the player hit an alien missile. If there is an alien then explode the alien. ; ; There is a code bug here, but it is extremely subtle. The algorithm for finding the row/column in the ; rack works by adding 16 to the reference coordinates (X for column, Y for row) until it passes or equals ; the target coordinates. This works great as long as the target point is within the alien's rack area. ; If the reference point is far to the right, the column number will be greater than 11, which messes ; up the column/row-to-pointer math. ; ; The entire rack of aliens is based on the lower left alien. Imagine all aliens are dead except the ; upper left. It wiggles down the screen and enters the players shields on the lower left where it begins ; to eat them. Imagine the player is under his own shields on the right side of the screen and fires a ; shot into his own shield. ; ; The alien is in the rack on row 4 (rows are numbered from bottom up starting with 0). The shot hits ; the shields below the alien's Y coordinate and gets correctly assigned to row 3. The alien is in the rack ; at column 0 (columns are numbered from left to right starting with 0). The shot hits the shields far to ; the right of the alien's X coordinate. The algorithm says it is in column 11. But 0-10 are the only ; correct values. ; ; The column/row-to-pointer math works by multiplying the row by 11 and adding the column. For the alien ; that is 11*4 + 0 = 44. For the shot that is 11*3 +11 = 44. The game thinks the shot hit the alien. ; STX H_Temp ; TFR B,L ;1504 -LD L,B L now holds the shot coordinate (adjusted) STB L_Temp LDX H_Temp JSR FindRow ;1505 -CALL FindRow Look up row number to B (Converts L - Y Coord to Row Coord L & Row # in C) LDA obj1CoorXr ;1508 -LD A,(obj1CoorXr) player's shot's Xr coordinate ... ; TFR A,H ;150B -LD H,A ... to H STA H_Temp LDX H_Temp JSR FindColumn ;150C -CALL FindColumn Get alien's coordinate (Converts H - X Coord to Column Coord H & Column # in C) PSHS B LDB H_Temp LDA L_Temp ; Fix Intel backwards byte order when storing HL STD expAlienYr ;150F -LD (expAlienYr),HL Put it in the exploding-alien descriptor PULS B LDA #$05 ;1512 -LD A,$05 Flag alien explosion ... STA plyrShotStatus ;1514 -LD (plyrShotStatus),A ... in progress JSR GetAlienStatPtr ;1517 -CALL GetAlienStatPtr Get descriptor for alien LDA ,X ;151A -LD A,(HL) Is alien ... TSTA ;151B -AND A ... alive LBEQ L1530 ;151C -JP Z,$1530 No ... must have been an alien shot ; CLR ,X ;151F -LD (HL),$00 Make alien invader dead JSR ScoreForAlien ;1521 -CALL ScoreForAlien Makes alien explosion sound and adjust score JSR ReadDesc ;1524 -CALL ReadDesc Load 5 byte sprite descriptor JSR DrawSprite ;1527 -CALL DrawSprite Draw explosion sprite on screen LDA #$10 ;152A -LD A,$10 Initiate alien-explosion STA expAlienTimer ;152C -LD (expAlienTimer),A ... timer to 16 RTS ;152F -RET Out ; ; Player shot leaving playfield, hitting shield, or hitting an alien shot L1530 LDA #$03 ;1530 -LD A,$03 Mark ... STA plyrShotStatus ;1532 -LD (plyrShotStatus),A ... player shot hit something other than alien LBRA L154A ;1535 -JP $154A Finish up ; AExplodeTime: ; Time down the alien explosion. Remove when done. LDX #expAlienTimer ;1538 -LD HL,$2003 Decrement alien explosion ... DEC ,X ;153B -DEC (HL) ... timer BEQ > ;153C -RET NZ Not done ... out RTS ! * Jump here LDB expAlienYr LDA expAlienYr+1 ; Fix Intel backwards byte order when storing HL TFR D,X ;153D -LD HL,(expAlienYr) Pixel pointer for exploding alien LDB #$10 ;1540 -LD B,$10 16 row pixel JSR EraseSimpleSprite ;1542 -CALL EraseSimpleSprite Clear the explosion sprite from the screen L1545 LDA #$04 ;1545 -LD A,$04 4 means that ... STA plyrShotStatus ;1547 -LD (plyrShotStatus),A ... alien has exploded (remove from active duty) ; L154A CLRA ;154A -XOR A Turn off ... STA alienIsExploding ;154B -LD (alienIsExploding),A ... alien-is-blowing-up flag LDB #$F7 ;154E -LD B,$F7 Turn off ... LBRA SoundBits3Off ;1550 -JP SoundBits3Off ... alien exploding sound ; FCB $00 ;1553 Cnt16s: ; Count number of 16s needed to bring reference (in A) up to target (in H). ; If the reference starts out beyond the target then we add 16s as long as ; the reference has a signed bit. But these aren't signed quantities. This ; doesn't make any sense. This counting algorithm produces questionable ; results if the reference is beyond the target. ; ; LDC #$00 ;1554 -LD C,$00 Count of 16s CLR C_Temp CMPA H_Temp ;1556 -CP H Compare reference coordinate to target BCS L155A ;1557 - If reference is greater or equal then do something questionable ... see below JSR WrapRef ;**** -CALL NC,WrapRef Added for 6809 worakaround L155A CMPA H_Temp ;155A -CP H Compare reference coordinate to target BCS > ;155B -RET NC If reference is greater or equal then done RTS ! * Jump here ADDA #$10 ;155C -ADD A,$10 Add 16 to reference INC C_Temp ;155E -INC C Bump 16s count LBRA L155A ;155F -JP $155A Keep testing FindRow: ; L contains a Yr coordinate. Find the row number within the rack that corresponds ; to the Yr coordinate. Return the row coordinate in L and the row number in C. ; LDA L_Temp STA H_Temp LDA refAlienYr ;1562 -LD A,(refAlienYr) Reference alien Yr coordinate ; TFR L,H ;1565 -LD H,L Target Yr coordinate to H JSR Cnt16s ;1566 -CALL Cnt16s Count 16s needed to bring ref alien to target ; TFR C,B ;1569 -LD B,C Count to B LDB C_Temp DECB ;156A -DEC B Base 0 SBCA #$10 ;156B -SBI A,$10 The counting also adds 16 no matter what ; TFR A,L ;156D -LD L,A To coordinate STA L_Temp RTS ;156E -RET Done FindColumn: ; H contains a Xr coordinate. Find the column number within the rack that corresponds ; to the Xr coordinate. Return the column coordinate in H and the column number in C. ; LDA refAlienXr ;156F -LD A,(refAlienXr) Reference alien Yn coordinate JSR Cnt16s ;1572 -CALL Cnt16s Count 16s to bring Y to target Y SBCA #$10 ;1575 -SBI A,$10 Subtract off extra 16 ; TFR A,H ;1577 -LD H,A To H STA H_Temp RTS ;1578 -RET Done L1579 LDA #$01 ;1579 -LD A,$01 Mark flying ... STA saucerHit ;157B -LD (saucerHit),A ... saucer has been hit LBRA L1545 ;157E -JP $1545 Remove player shot GetAlienStatPtr: ; B is row number. C is column number (starts at 1). ; Return pointer to alien-status flag for current player. TFR B,A ;1581 -LD A,B Hold original STB B_Temp ROLA ;1582 -RLCA *2 ROLA ;1583 -RLCA *4 ROLA ;1584 -RLCA *8 ADDA B_Temp ;1585 -ADD A,B *9 ADDA B_Temp ;1586 -ADD A,B *10 ADDA B_Temp ;1587 -ADD A,B *11 ADDA C_Temp ;1588 -ADD A,C Add row offset to column offset DECA ;1589 -DEC A -1 ; TFR A,L ;158A -LD L,A Set LSB of HL STA L_Temp LDA playerDataMSB ;158B -LD A,(playerDataMSB) Set ... ; TFR A,H ;158E -LD H,A ... MSB of HL with active player indicator STA H_Temp LDX H_Temp RTS ;158F -RET Done WrapRef: ; This is called if the reference point is greater than the target point. I believe the goal is to ; wrap the reference back around until it is lower than the target point. But the algorithm simply adds ; until the sign bit of the the reference is 0. If the target is 2 and the reference is 238 then this ; algorithm moves the reference 238+16=244 then 244+16=4. Then the algorithm stops. But the reference is ; STILL greater than the target. ; ; Also imagine that the target is 20 and the reference is 40. The algorithm adds 40+16=56, which is not ; negative, so it stops there. ; ; I think the intended code is "JP NC" instead of "JP M", but even that doesn't make sense. ; INC C_Temp ;1590 -INC C Increase 16s count ADDA #$10 ;1591 -ADD A,$10 Add 16 to ref LBMI WrapRef ;1593 -JP M,WrapRef Keep going till result is positive RTS ;1596 -RET Out RackBump: ; When rack bumps the edge of the screen then the direction flips and the rack ; drops 8 pixels. The deltaX and deltaY values are changed here. Interestingly ; if there is only one alien left then the right value is 3 instead of the ; usual 2. The left direction is always -2. LDA rackDirection ;1597 -LD A,(rackDirection) Get rack direction TSTA ;159A -AND A Moving right? LBNE L15B7 ;159B -JP NZ,$15B7 No ... handle moving left ; LDX #Screen_Start+$1AA4 ;159E -LD HL,$3EA4 Line down the right edge of playfield JSR L15C5 ;15A1 -CALL $15C5 Check line down the edge BCS > ;15A4 -RET NC Nothing is there ... return RTS ! * Jump here LDB #$FE ;15A5 -LD B,$FE Delta X of -2 LDA #$01 ;15A7 -LD A,$01 Rack now moving right ; L15A9 STA rackDirection ;15A9 -LD (rackDirection),A Set new rack direction TFR B,A ;15AC -LD A,B B has delta X STA refAlienDXr ;15AD -LD (refAlienDXr),A Set new delta X LDA rackDownDelta ;15B0 -LD A,(rackDownDelta) Set delta Y ... STA refAlienDYr ;15B3 -LD (refAlienDYr),A ... to drop rack by 8 RTS ;15B6 -RET Done ; L15B7 LDX #Screen_Start+$124 ;15B7 -LD HL,$2524 Line down the left edge of playfield JSR L15C5 ;15BA -CALL $15C5 Check line down the edge BCS > ;15BD -RET NC Nothing is there ... return RTS ! * Jump here JSR L18F1 ;15BE -CALL $18F1 Get moving-right delta X value of 2 (3 if just one alien left) CLRA ;15C1 -XOR A Rack now moving left LBRA L15A9 ;15C2 -JP $15A9 Set rack direction ; L15C5 LDB #$17 ;15C5 -LD B,$17 Checking 23 bytes in a line up the screen from near the bottom L15C7 LDA ,X ;15C7 -LD A,(HL) Get screen memory TSTA ;15C8 -AND A Is screen memory empty? LBNE L166B ;15C9 -JP NZ,$166B No ... set carry flag and out LEAX 1,X ;15CC -INC HL Next byte on screen DECB ;15CD -DEC B All column done? LBNE L15C7 ;15CE -JP NZ,$15C7 No ... keep looking ANDCC #$FE ;Clear the Carry Flag if we got here no aliens hitting the Edge RTS ;15D1 -RET Return with carry flag clear ; NOP ;15D2 -NOP ** Why? Something optimized? DrawSprite: ; Draw sprite at [DE] to screen at pixel position in HL ; The hardware shift register is used in converting pixel positions ; to screen coordinates. JSR CnvtPixNumber ;15D3 -CALL CnvtPixNumber Convert pixel number to screen/shift PSHS X ;15D6 -PUSH HL Preserve screen coordinate L15D7 PSHS B ;15D7 -PUSH BC Hold for a second PSHS X ;15D8 -PUSH HL Hold for a second LDA ,Y ;15D9 -LD A,(DE) From sprite data JSR IO_04_SHFT_DATA ;15DA -OUT (SHFT_DATA),A Write data to shift register JSR IO_03_SHFT_IN ;15DC -IN A,(SHFT_IN) Read back shifted amount LDU #ReverseLookUp LDA A,U STA ,X ;15DE -LD (HL),A Shifted sprite to screen LEAX 1,X ;15DF -INC HL Adjacent cell LEAY 1,Y ;15E0 -INC DE Next in sprite data CLRA ;15E1 -XOR A 0 JSR IO_04_SHFT_DATA ;15E2 -OUT (SHFT_DATA),A Write 0 to shift register JSR IO_03_SHFT_IN ;15E4 -IN A,(SHFT_IN) Read back remainder of previous LDU #ReverseLookUp LDA A,U STA ,X ;15E6 -LD (HL),A Write remainder to adjacent PULS X ;15E7 -POP HL Old screen coordinate ; LDU #$0020 ;15E8 -LD BC,$0020 Offset screen ... ; ADD X=X+U ;15EB -ADD HL,BC ... to next row LEAX $20,X PULS B ;15EC -POP BC Restore count DECB ;15ED -DEC B All done? LBNE L15D7 ;15EE -JP NZ,$15D7 No ... do all PULS X ;15F1 -POP HL Restore HL RTS ;15F2 -RET Done CountAliens: ; Count number of aliens remaining in active game and return count 2082 holds the current count. ; If only 1, 206B gets a flag of 1 ** but ever nobody checks this ; JSR GetPlayerDataPtr ;15F3 -CALL GetPlayerDataPtr Get active player descriptor ;; LDU #$3700 ;15F6 -LD BC,$3700 B=55 aliens to check? ; LDB #$37 ;; CLR C_Temp ;L15F9 LDA ,X ;15F9 -LD A,(HL) Get byte ; TSTA ;15FA -AND A Is it a zero? ; LBEQ L15FF ;15FB -JP Z,$15FF Yes ... don't count it ; INC C_Temp ;15FE -INC C Count the live aliens ;L15FF LEAX 1,X ;15FF -INC HL Next alien ; DECB ;1600 -DEC B Count ... ; LBNE L15F9 ;1601 -JP NZ,$15F9 ... all alien indicators ;; TFR C,A ;1604 -LD A,C Get the count ; LDA C_Temp ; STA numAliens ;1605 -LD (numAliens),A Hold it ; CMPA #$01 ;1608 -CP $01 Just one? ; BEQ > ;160A -RET NZ No keep going ; RTS ;! * Jump here ; LDX #flag1Alien ;160B -LD HL,$206B Set flag if ... ;; ST#$01 ,X ;160E -LD (HL),$01 ... only one alien left ; LDA #$01 ; STA ,X ; RTS ;1610 -RET Out ; C_Temp above was being over written during the IRQ routines, so I re-wrote the code above to not use C_Temp - Probably affecting other sections of code :( JSR GetPlayerDataPtr CLR AlienCount LDB #$37 L15F9 LDA ,X+ BEQ L15FF INC AlienCount L15FF DECB BNE L15F9 LDA AlienCount CMPA #$01 BEQ > LDA AlienCount STA numAliens RTS ! LDX #flag1Alien LDA #$01 STA ,X LDA AlienCount STA numAliens RTS GetPlayerDataPtr: ; Set HL with 2100 if player 1 is active or 2200 if player 2 is active ; ; LDL #$00 ;1611 -LD L,$00 Byte boundary LDA playerDataMSB ;1613 -LD A,(playerDataMSB) Active player number CLRB TFR D,X ;1616 -LD H,A Set HL to data RTS ;1617 -RET Done PlrFireOrDemo: ; Initiate player fire if button is pressed. ; Demo commands are parsed here if in demo mode LDA playerAlive ;1618 -LD A,(playerAlive) Is there an active player? CMPA #$FF ;161B -CP $FF FF = alive BEQ > ;161D -RET NZ Player has been shot - no firing RTS ! * Jump here LDX #firstgameobj ;161E -LD HL,$2010 Get player ... LDA ,X ;1621 -LD A,(HL) ... task ... LEAX 1,X ;1622 -INC HL ... timer ... ; LDB ,X ;1623 -LD B,(HL) ... value ; ORA Register - need tweaking B ;1624 -OR B Is the timer 0 (object active)? ORA ,X BEQ > ;1625 -RET NZ No ... no firing till player object starts RTS ! * Jump here LDA plyrShotStatus ;1626 -LD A,(plyrShotStatus) Does the player have ... TSTA ;1629 -AND A ... a shot on the screen? BEQ > ;162A -RET NZ Yes ... ignore RTS ! * Jump here LDA gameMode ;162B -LD A,(gameMode) Are we in ... TSTA ;162E -AND A ... game mode? LBEQ L1652 ;162F -JP Z,$1652 No ... in demo mode ... constant firing in demo LDA fireBounce ;1632 -LD A,(fireBounce) Is fire button ... TSTA ;1635 -AND A ... being held down? LBNE L1648 ;1636 -JP NZ,$1648 Yes ... wait for bounce JSR ReadInputs ;1639 -CALL ReadInputs Read active player controls ANDA #$10 ;163C -AND $10 Fire-button pressed? BNE > ;163E -RET Z No ... out RTS ! * Jump here LDA #$01 ;163F -LD A,$01 Flag STA plyrShotStatus ;1641 -LD (plyrShotStatus),A Flag shot active STA fireBounce ;1644 -LD (fireBounce),A Flag that fire button is down RTS ;1647 -RET Out L1648 JSR ReadInputs ;1648 -CALL ReadInputs Read active player controls ANDA #$10 ;164B -AND $10 Fire-button pressed? BEQ > ;164D -RET NZ Yes ... ignore RTS ! * Jump here STA fireBounce ;164E -LD (fireBounce),A Else ... clear flag RTS ;1651 -RET Out ; Handle demo (constant fire, parse demo commands) L1652 LDX #plyrShotStatus ;1652 -LD HL,$2025 Demo fires ... LDA #$01 STA ,X ;1655 -LD (HL),$01 ... constantly LDD demoCmdPtrLSB ;1657 -LD HL,(demoCmdPtrLSB) Demo command bufer EXG A,B ADDD #1 ;165A -INC HL Next position ; TFR L,A ;165B -LD A,L Command buffer ... ; STX H_Temp ; LDA L_Temp CMPB #$7E ;165C -CP $7E ... wraps around LBCS L1663 ;165E -JP C,$1663 ... Buffer from 1F74 to 1F7E LDB #$74 ;1661 -LD L,$74 ... overflow ; LDA H_Temp L1663 TFR D,X EXG A,B STD demoCmdPtrLSB ;1663 -LD (demoCmdPtrLSB),HL Next demo command LDA ,X ;1666 -LD A,(HL) Get next command STA nextDemoCmd ;1667 -LD (nextDemoCmd),A Set command for movement RTS ;166A -RET Done L166B ORCC #$01 ;166B -SCF Set carry flag RTS ;166C -RET Done L166D CLRA ;166D -XOR A 0 JSR L1A8B ;166E -CALL $1A8B Print ZERO ships remain L1671 JSR CurPlyAlive ;1671 -CALL CurPlyAlive Get active-flag ptr for current player CLR ,X ;1674 -LD (HL),$00 Flag player is dead JSR L09CA ;1676 -CALL $09CA Get score descriptor for current player LEAX 1,X ;1679 -INC HL Point to high two digits LDY #curhiscorU2dig ;167A -LD DE,$20F5 Current high score upper two digits LDA ,Y ;167D -LD A,(DE) Is player score greater ... CMPA ,X ;167E -CP (HL) ... than high score? LEAY -1,Y ;167F -DEC DE Point to LSB LEAX -1,X ;1680 -DEC HL Point to LSB LDA ,Y ;1681 -LD A,(DE) Go ahead and fetch high score lower two digits LBEQ L168B ;1682 -JP Z,$168B Upper two are the same ... have to check lower two LBCC L1698 ;1685 -JP NC,$1698 Player score is lower than high ... nothing to do LBRA L168F ;1688 -JP $168F Player socre is higher ... go copy the new high score ; L168B CMPA ,X ;168B -CP (HL) Is lower digit higher? (upper was the same) LBCC L1698 ;168C -JP NC,$1698 No ... high score is still greater than player's score L168F LDA ,X ;168F -LD A,(HL) Copy the new ... STA ,Y ;1690 -LD (DE),A ... high score lower two digits LEAY 1,Y ;1691 -INC DE Point to MSB LEAX 1,X ;1692 -INC HL Point to MSB LDA ,X ;1693 -LD A,(HL) Copy the new ... STA ,Y ;1694 -LD (DE),A ... high score upper two digits JSR PrintHiScore ;1695 -CALL PrintHiScore Draw the new high score L1698 LDA twoPlayers ;1698 -LD A,(twoPlayers) Number of players TSTA ;169B -AND A Is this a single player game? LBEQ L16C9 ;169C -JP Z,$16C9 Yes ... short message LDX #Screen_Start+$403 ;169F -LD HL,$2803 Screen coordinates LDY #MessageGOver ;16A2 -LD DE,$1AA6 "GAME OVER PLAYER< >" ; LDC #$14 ;16A5 -LD C,$14 20 characters LDB #$14 STB C_Temp JSR PrintMessageDel ;16A7 -CALL PrintMessageDel Print message STX H_Temp DEC H_Temp ;16AA -DEC H Back up ... DEC H_Temp ;16AB -DEC H ... to player indicator LDB #$1B ;16AC -LD B,$1B "1" LDA playerDataMSB ;16AE -LD A,(playerDataMSB) Player number RORA ;16B1 -RRCA Is this player 1? LBCS L16B7 ;16B2 -JP C,$16B7 Yes ... keep the digit LDB #$1C ;16B5 -LD B,$1C Else ... set digit 2 L16B7 TFR B,A ;16B7 -LD A,B To A LDX H_Temp JSR DrawChar ;16B8 -CALL DrawChar Print player number JSR OneSecDelay ;16BB -CALL OneSecDelay Short delay JSR L18E7 ;16BE -CALL $18E7 Get current player "alive" flag LDA ,X ;16C1 -LD A,(HL) Is player ... TSTA ;16C2 -AND A ... alive? LBEQ L16C9 ;16C3 -JP Z,$16C9 No ... skip to "GAME OVER" sequence LBRA L02ED ;16C6 -JP $02ED Switch players and game loop ; L16C9 LDX #Screen_Start+$918 ;16C9 -LD HL,$2D18 Screen coordinates LDY #MessageGOver ;16CC -LD DE,$1AA6 "GAME OVER PLAYER< >" ; LDC #$0A ;16CF -LD C,$0A Just the "GAME OVER" part LDB #$0A STB C_Temp JSR PrintMessageDel ;16D1 -CALL PrintMessageDel Print message JSR TwoSecDelay ;16D4 -CALL TwoSecDelay Long delay JSR ClearPlayField ;16D7 -CALL ClearPlayField Clear center window CLRA ;16DA -XOR A Now in ... STA gameMode ;16DB -LD (gameMode),A ... demo mode STA IO_05_SOUND2 ;16DE -OUT (SOUND2),A All sound off JSR EnableGameTasks ;16E0 -CALL EnableGameTasks Enable ISR game tasks LBRA L0B89 ;16E3 -JP $0B89 Print credit information and do splash L16E6 LDS #Stack_Pointer ;16E6 -LD SP,$2400 Reset stack ANDCC #$EF ;16E9 -EI Enable interrupts CLRA ;16EA -XOR A Flag ... STA playerAlive ;16EB -LD (playerAlive),A ... player is shot L16EE JSR PlayerShotHit ;16EE -CALL PlayerShotHit player's shot collision detection LDB #$04 ;16F1 -LD B,$04 Player has been hit ... JSR SoundBits3On ;16F3 -CALL SoundBits3On ... sound JSR L0A59 ;16F6 -CALL $0A59 Has flag been set? LBNE L16EE ;16F9 -JP NZ,$16EE No ... wait for the flag JSR DsableGameTasks ;16FC -CALL DsableGameTasks Disable ISR game tasks LDX #Screen_Start+$301 ;16FF -LD HL,$2701 player's stash of ships JSR L19FA ;1702 -CALL $19FA Erase the stash of shps CLRA ;1705 -XOR A Print ... JSR L1A8B ;1706 -CALL $1A8B ... a zero (number of ships) LDB #$FB ;1709 -LD B,$FB Turn off ... LBRA L196B ;170B -JP $196B ... player shot sound AShotReloadRate: ; Use the player's MSB to determine how fast the aliens reload their ; shots for another fire. JSR L09CA ;170E -CALL $09CA Get score descriptor for active player LEAX 1,X ;1711 -INC HL MSB value LDA ,X ;1712 -LD A,(HL) Get the MSB value LDY #AReloadScoreTab ;1713 -LD DE,$1CB8 Score MSB table LDX #ShotReloadRate ;1716 -LD HL,$1AA1 Corresponding fire reload rate table ; LDC #$04 ;1719 -LD C,$04 Only 4 entries (a 5th value of 7 is used after that) LDB #$04 STB C_Temp ; TFR A,B ;171B -LD B,A Hold the score value STA B_Temp L171C LDA ,Y ;171C -LD A,(DE) Get lookup from table CMPA B_Temp ;171D -CP B Compare them LBCC L1727 ;171E -JP NC,$1727 Equal or below ... use this table entry LEAX 1,X ;1721 -INC HL Next ... LEAY 1,Y ;1722 -INC DE ... entry in table DEC C_Temp ;1723 -DEC C Do all ... LBNE L171C ;1724 -JP NZ,$171C ... 4 entries in the tables L1727 LDA ,X ;1727 -LD A,(HL) Load the shot reload value STA aShotReloadRate ;1728 -LD (aShotReloadRate),A Save the value for use in shot routine RTS ;172B -RET Done ; Shot sound on or off depending on 2025 ShotSound: LDA plyrShotStatus ;172C -LD A,(plyrShotStatus) Player shot flag CMPA #$00 ;172F -CP $00 Active shot? LBNE L1739 ;1731 -JP NZ,$1739 Yes ... go LDB #$FD ;1734 -LD B,$FD Sound mask LBRA SoundBits3Off ;1736 -JP SoundBits3Off Mask off sound ; L1739 LDB #$02 ;1739 -LD B,$02 Sound bit LBRA SoundBits3On ;173B -JP SoundBits3On OR on sound ; FCB $00 ;173E - ** Why? TimeFleetSound: ; This called from the ISR times down the fleet and sets the flag at 2095 if ; the fleet needs a change in sound handling (new delay, new sound) LDX #fleetSndHold ;1740 -LD HL,$209B Pointer to hold time for fleet DEC ,X ;1743 -DEC (HL) Decrement hold time BNE L1747 ;1744 - If 0 turn fleet movement sound off JSR L176D ;**** -CALL Z,$176D Added for 6809 worakaround L1747 LDA playerOK ;1747 -LD A,(playerOK) Is player OK? TSTA ;174A -AND A 1 means OK LBEQ L176D ;174B -JP Z,$176D Player not OK ... fleet movement sound off and out LDX #curtimesltsnd ;174E -LD HL,$2096 Current time on fleet sound DEC ,X ;1751 -DEC (HL) Count down BEQ > ;1752 -RET NZ Not time to change sound ... out RTS ! * Jump here LDX #soundPort5 ;1753 -LD HL,$2098 Current sound port 3 value LDA ,X ;1756 -LD A,(HL) Get value STA IO_05_SOUND2 ;1757 -OUT (SOUND2),A Set sounds LDA numAliens ;1759 -LD A,(numAliens) Number of aliens on active screen TSTA ;175C -AND A Is it zero? LBEQ L176D ;175D -JP Z,$176D Yes ... turn off fleet movement sound and out LEAX -1,X ;1760 -DEC HL (2097) Point to fleet timer reload LDA ,X ;1761 -LD A,(HL) Get fleet delay value LEAX -1,X ;1762 -DEC HL (2096) Point to fleet timer STA ,X ;1763 -LD (HL),A Reload the timer LEAX -1,X ;1764 -DEC HL Point to change-sound ; ST#$01 ,X ;1765 -LD (HL),$01 (2095) time to change sound LDA #$01 STA ,X LDA #$04 ;1767 -LD A,$04 Set hold ... STA fleetSndHold ;1769 -LD (fleetSndHold),A ... time for fleet sound RTS ;176C -RET Done L176D LDA soundPort5 ;176D -LD A,(soundPort5) Current sound port 3 value L1770 ANDA #$30 ;1770 -AND $30 Mask off fleet movement sounds STA IO_05_SOUND2 ;1772 -OUT (SOUND2),A Set sounds RTS ;1774 -RET Out FleetDelayExShip: ; This game-loop routine handles two sound functions. The routine does: ; 1) Time out the extra-ship awarded sound and turn it off when done ; 2) Load the fleet sound delay based on number of remaining aliens ; 3) Make the changing fleet sound ; ; The 2095 flag is set by the ISR and cleared here. The ISR does the timing and sets 2095 when it ; is time to make a new fleet sound. ; LDA changeFleetSnd ;1775 -LD A,(changeFleetSnd) Time for new ... TSTA ;1778 -AND A ... fleet movement sound? LBEQ L17AA ;1779 -JP Z,$17AA No ... skip to extra-man timing LDX #AlienDelLst1 ;177C -LD HL,$1A11 Number of aliens list coupled ... LDY #AlienDelLst2 ;177F -LD DE,$1A21 ... with delay list LDA numAliens ;1782 -LD A,(numAliens) Get the number of aliens on the screen L1785 CMPA ,X ;1785 -CP (HL) Compare it to the first list value LBCC L178E ;1786 -JP NC,$178E Number of live aliens is higher than value ... use the delay LEAX 1,X ;1789 -INC HL Move to ... LEAY 1,Y ;178A -INC DE ... next list value LBRA L1785 ;178B -JP $1785 Find the right delay L178E LDA ,Y ;178E -LD A,(DE) Get the delay from the second list STA fleetSndReload ;178F -LD (fleetSndReload),A Store the new alien sound delay LDX #soundPort5 ;1792 -LD HL,$2098 Get current state ... LDA ,X ;1795 -LD A,(HL) ... of sound port ANDA #$30 ;1796 -AND $30 Mask off all fleet movement sounds ; TFR A,B ;1798 -LD B,A Hold the value STA B_Temp LDA ,X ;1799 -LD A,(HL) Get current state ANDA #$0F ;179A -AND $0F This time ONLY the fleet movement sounds ROLA ;179C -RLCA Shift next to next sound CMPA #$10 ;179D -CP $10 Overflow? LBNE L17A4 ;179F -JP NZ,$17A4 No ... keep it LDA #$01 ;17A2 -LD A,$01 Reset back to first sound L17A4 ORA B_Temp ;17A4 -OR B Add fleet sounds to current sound value STA ,X ;17A5 -LD (HL),A Store new sound value CLRA ;17A6 -XOR A Restart ... STA changeFleetSnd ;17A7 -LD (changeFleetSnd),A ... waiting on fleet time ; L17AA LDX #extraHold ;17AA -LD HL,$2099 Sound timer for award extra ship DEC ,X ;17AD -DEC (HL) Time expired? BEQ > ;17AE -RET NZ No ... leave sound playing RTS ! * Jump here LDB #$EF ;17AF -LD B,$EF Turn off bit set with #$10 (award extra ship) LBRA SoundBits3Off ;17B1 -JP SoundBits3Off Stop sound and out SndOffExtPly: ; LDB #$EF ;17B4 -LD B,$EF Mask off sound bit 4 (Extended play) LDX #soundPort5 ;17B6 -LD HL,$2098 Current sound content LDA ,X ;17B9 -LD A,(HL) Get current sound bits ANDA #$EF ;17BA -AND B Turn off extended play STA ,X ;17BB -LD (HL),A Remember settings STA IO_05_SOUND2 ;17BC -OUT (SOUND2),A Turn off extended play RTS ;17BE -RET Out FCB $00 ;17BF -RET ** Why? ReadInputs: ; Read control inputs for active player ; LDA playerDataMSB ;17C0 -LD A,(playerDataMSB) Get active player ; RORA ;17C3 -RRCA Test player ; LBCC L17CA ;17C4 -JP NC,$17CA Player 2 ... read port 2 ; LDA IO_01_INP1 ;17C7 -IN A,(INP1) Player 1 ... read port 1 ; RTS ;17C9 -RET Out ;L17CA LDA IO_02_INP2 ;17CA -IN A,(INP2) Get controls for player 2 ; RTS ;17CC -RET Out PSHS B CLRB * Clear IO_01_INP1 LDA playerDataMSB ;17C0 -LD A,(playerDataMSB) Get active player RORA ;17C3 -RRCA Test player BCC L17CA ;17C4 -JP NC,$17CA Player 2 ... read port 2 L17C7 LDU #PIA0_Byte_0_KeyRow_Joy * Point to PIA0 ;Fire Button - Space Bar LDA #%01111111 * Bit 7 STA $02,U LDA ,U BITA #%00001000 * and bit 3 = Space Bar BNE > ORB #%00010000 * ORB Bit 4 = Fire Button ; Left Button - Left Arrow ! LDA #%11011111 * Bit 5 STA $02,U LDA ,U BITA #%00001000 * and bit 3 = Left Arrow Key BNE > ORB #%00100000 * ORB Bit 5 Left Arrow Button ; Right Button - Right Arrow ! LDA #%10111111 * Bit 6 STA $02,U LDA ,U BITA #%00001000 * and bit 3 = Right Arrow Key BNE > ORB #%01000000 * ORB Bit 6 = Right Arrow Button ! TFR B,A PULS B RTS L17CA BRA L17C7 * Jump here if we have seperate controls for player 2 - For now use Player 1 buttons ; Check and handle TILT CheckHandleTilt: LDA IO_02_INP2 ;17CD -IN A,(INP2) Read input port ANDA #$04 ;17CF -AND $04 Tilt? BNE > ;17D1 -RET Z No tilt ... return RTS ! * Jump here LDA tilt ;17D2 -LD A,(tilt) ; Already in TILT handle? TSTA ;17D5 -AND A 1 = yes BEQ > ;17D6 -RET NZ Yes ... ignore it now RTS ! * Jump here LDS #Stack_Pointer ;17D7 -LD SP,$2400 Reset stack LDB #$04 ;17DA -LD B,$04 Do this 4 times L17DC JSR ClearPlayField ;17DC -CALL ClearPlayField Clear center window DECB ;17DF -DEC B All done? LBNE L17DC ;17E0 -JP NZ,$17DC No ... do again LDA #$01 ;17E3 -LD A,$01 Flag ... STA tilt ;17E5 -LD (tilt),A ; ... handling TILT JSR DsableGameTasks ;17E8 -CALL DsableGameTasks Disable game tasks ANDCC #$EF ;17EB -EI Re-enable interrupts LDY #MessageTilt ;17EC -LD DE,$1CBC Message "TILT" LDX #Screen_Start+$C16 ;17EF -LD HL,$3016 Center of screen LDB #$04 ;17F2 -LD C,$04 Four letters STB C_Temp JSR PrintMessageDel ;17F4 -CALL PrintMessageDel Print "TILT" JSR OneSecDelay ;17F7 -CALL OneSecDelay Short delay CLRA ;17FA -XOR A Zero STA tilt ;17FB -LD (tilt),A ; TILT handle over STA waitStartLoop ;17FE -LD (waitStartLoop),A Back into splash screens LBRA L16C9 ;1801 -JP $16C9 Handle game over for player CtrlSaucerSound: LDX #saucerActive ;1804 -LD HL,$2084 Saucer on screen flag LDA ,X ;1807 -LD A,(HL) Is the saucer ... TSTA ;1808 -AND A ... on the screen? LBEQ L0707 ;1809 -JP Z,$0707 No ... UFO sound off LEAX 1,X ;180C -INC HL Saucer hit flag LDA ,X ;180D -LD A,(HL) (2085) Get saucer hit flag TSTA ;180E -AND A Is saucer in "hit" sequence? BEQ > ;180F -RET NZ Yes ... out RTS ! * Jump here LDB #$01 ;1810 -LD B,$01 Retrigger saucer ... LBRA SoundBits3On ;1812 -JP SoundBits3On ... sound (retrigger makes it warble?) DrawAdvTable: ; Draw "SCORE ADVANCE TABLE" LDX #Screen_Start+$410 ;1815 -LD HL,$2810 0x410 is 1040 rotCol=32, rotRow=16 LDY #MessageAdv ;1818 -LD DE,$1CA3 "*SCORE ADVANCE TABLE*" LDB #$15 ;181B -LD C,$15 21 bytes in message STB C_Temp JSR PrintMessage ;181D -CALL PrintMessage Print message LDA #$0A ;1820 -LD A,$0A 10 bytes in every "=xx POINTS" string STA temp206C ;1822 -LD (temp206C),A Hold the count LDU #ScoreAdvTable ;1825 -LD BC,$1DBE Coordinate/sprite for drawing table L1828 JSR ReadPriStruct ;1828 -CALL ReadPriStruct Get HL=coordinate, DE=image LBCS L1837 ;182B -JP C,$1837 Move on if done JSR L1844 ;182E -CALL $1844 Draw 16-byte sprite LBRA L1828 ;1831 -JP $1828 Do all in table ; JSR OneSecDelay ;1834 -CALL OneSecDelay One second delay L1837 LDU #AlienScoreTable ;1837 -LD BC,$1DCF Coordinate/message for drawing table L183A JSR ReadPriStruct ;183A -CALL ReadPriStruct Get HL=coordinate, DE=message BCC > ;183D -RET C Out if done RTS ! * Jump here JSR L184C ;183E -CALL $184C Print message LBRA L183A ;1841 -JP $183A Do all in table ; L1844 PSHS U ;1844 -PUSH BC Hold BC LDB #$10 ;1845 -LD B,$10 16 bytes JSR DrawSimpSprite ;1847 -CALL DrawSimpSprite Draw simple PULS U ;184A -POP BC Restore BC RTS ;184B -RET Out ; L184C PSHS U ;184C -PUSH BC Hold BC LDA temp206C ;184D -LD A,(temp206C) Count of 10 ... STA C_Temp ;1850 -LD C,A ... to C JSR PrintMessageDel ;1851 -CALL PrintMessageDel Print the message with delay between letters PULS U ;1854 -POP BC Restore BC RTS ;1855 -RET Out ReadPriStruct: ; Read a 4-byte print-structure pointed to by BC ; HL=Screen coordiante, DE=pointer to message ; If the first byte is FF then return with Carry Flag set = 1. LDA ,U ;1856 -LD A,(BC) Get the screen LSB CMPA #$FF ;1857 -CP $FF Valid? BNE > ;185A -RET Z Return if 255 ORCC #$01 ;1859 -SCF If not Carry will be 1 RTS ! * Jump here ; TFR A,L ;185B -LD L,A Screen LSB to L ; STA L_Temp ; INC U ;185C -INC BC Next ; LDA ,U+ ;185D -LD A,(BC) Read screen MSB ; TFR A,H ;185E -LD H,A Screen MSB to H ; STA H_Temp ; INC U ;185F -INC BC Next ; LDA ,U+ ;1860 -LD A,(BC) Read message LSB ; TFR A,E ;1861 -LD E,A Message LSB to E ; STA E_Temp ; INC U ;1862 -INC BC Next ; LDA ,U+ ;1863 -LD A,(BC) Read message MSB ; TFR A,D ;1864 -LD D,A Message MSB to D ; STA D_Temp ; INC U ;1865 -INC BC Next (for next print) LDD ,U++ EXG A,B TFR D,X LDD ,U++ EXG A,B TFR D,Y ANDCC #$FE ;1866 -AND A Clear Carry RTS ;1867 -RET Done SplashSprite: ; Moves a sprite up or down in splash mode. Interrupt moves the sprite. When it reaches ; Y value in 20CA the flag at 20CB is raised. The image flips between two pictures every ; 4 movements. LDX #splashAnForm ;1868 -LD HL,$20C2 Descriptor INC ,X ;186B -INC (HL) Change image LEAX 1,X ;186C -INC HL Point to delta-x LDB ,X ;186D -LD C,(HL) Get delta-x STB C_Temp JSR AddDelta ;186E -CALL AddDelta Add delta-X and delta-Y to X and Y ; TFR A,B ;1871 -LD B,A Current y coordinate ; LDA splashTargetY ;1872 -LD A,(splashTargetY) Has sprite reached ... CMPA splashTargetY ;1875 -CP B ... target coordinate? LBEQ L1898 ;1876 -JP Z,$1898 Yes ... flag and out LDD splashImRestLSB EXG A,B TFR D,X LDA splashAnForm ;1879 -LD A,(splashAnForm) Image number ANDA #$04 ;187C -AND $04 Watching bit 3 for flip delay ; TFR splashImRestLSB,X ;187E -LD HL,(splashImRestLSB) Image LBNE L1888 ;1881 -JP NZ,$1888 Did bit 3 go to 0? No ... keep current image ; LDY #$0030 ;1884 -LD DE,$0030 16*3 ... ; ADD X=X+Y ;1887 -ADD HL,DE ... use other image form LEAX $30,X L1888 TFR X,D EXG A,B STD splashImageLSB ;1888 -LD (splashImageLSB),HL Image to descriptor structure LDX #xyimagedesc ;188B -LD HL,$20C5 X,Y,Image descriptor JSR ReadDesc ;188E -CALL ReadDesc Read sprite descriptor EXG Y,X ;1891 -EX DE,HL Image to DE, position to HL LBRA DrawSprite ;1892 -JP DrawSprite Draw the sprite ; FCB $00X ;1895 -JP DrawSprite L1898 LDA #$01 ;1898 -LD A,$01 Flag that sprite ... STA splashReached ;189A -LD (splashReached),A ... reached location RTS ;189D -RET Out ;Animate alien shot to extra "C" in splash L189E LDX #objStruct4 ;189E -LD HL,$2050 Task descriptor for game object 4 (squiggly shot) LDY #ShotExtraC ;18A1 -LD DE,$1BC0 Task info for animate-shot-to-extra-C LDB #$10 ;18A4 -LD B,$10 Block copy ... JSR BlockCopy ;18A6 -CALL BlockCopy ... 16 bytes LDA #$02 ;18A9 -LD A,$02 Set shot sync ... STA shotSync ;18AB -LD (shotSync),A ... to run the squiggly shot LDA #$FF ;18AE -LD A,$FF Shot direction (-1) STA alienShotDelta ;18B0 -LD (alienShotDelta),A Alien shot delta LDA #$04 ;18B3 -LD A,$04 Animate ... STA isrSplashTask ;18B5 -LD (isrSplashTask),A ... shot L18B8 LDA squShotStatus ;18B8 -LD A,(squShotStatus) Has shot ... ANDA #$01 ;18BB -AND $01 ... collided? LBEQ L18B8 ;18BD -JP Z,$18B8 No ... keep waiting L18C0 LDA squShotStatus ;18C0 -LD A,(squShotStatus) Wait ... ANDA #$01 ;18C3 -AND $01 ... for explosion ... LBNE L18C0 ;18C5 -JP NZ,$18C0 ... to finish LDX #Screen_Start+$F11 ;18C8 -LD HL,$3311 Here is where the extra C is LDA #$26 ;18CB -LD A,$26 Space character ; NOP ;18CD -NOP ** Why? JSR DrawChar ;18CE -CALL DrawChar Draw character LBRA TwoSecDelay ;18D1 -JP TwoSecDelay Two second delay and out ; Initializiation comes here ; L18D4 LDS #Stack_Pointer ;18D4 -LD SP,$2400 Set stack pointer just below screen LDB #$00 ;18D7 -LD B,$00 Count 256 bytes JSR L01E6 ;18D9 -CALL $01E6 Copy ROM to RAM JSR DrawStatus ;18DC -CALL DrawStatus Print scores and credits ; L18DF LDA #$08 ;18DF -LD A,$08 Set alien ... STA aShotReloadRate ;18E1 -LD (aShotReloadRate),A ... shot reload rate LBRA L0AEA ;18E4 -JP $0AEA Top of splash screen loop ; Get player-alive flag for OTHER player L18E7 LDA playerDataMSB ;18E7 -LD A,(playerDataMSB) Player data MSB LDX #player1Alive ;18EA -LD HL,$20E7 Alive flags (player 1 and 2) RORA ;18ED -RRCA Bit 1=1 for player 1 BCS > ;18EE -RET NC Player 2 ... we have it ... out RTS ! * Jump here LEAX 1,X ;18EF -INC HL Player 1's flag RTS ;18F0 -RET Done ; If there is one alien left then the right motion is 3 instead of 2. That's ; why the timing is hard to hit after the change. L18F1 LDB #$02 ;18F1 -LD B,$02 Rack moving right delta X LDA numAliens ;18F3 -LD A,(numAliens) Number of aliens on screen DECA ;18F6 -DEC A Just one left? BEQ > ;18F7 -RET NZ No ... use right delta X of 2 RTS ! * Jump here INCB ;18F8 -INC B Just one alien ... move right at 3 instead of 2 RTS ;18F9 -RET Done SoundBits3On: ; Add in bit for sound STB B_Temp LDA soundPort3 ;18FA -LD A,(soundPort3) Current value of sound port ORA B_Temp ;18FD -OR B Add in new sounds STA soundPort3 ;18FE -LD (soundPort3),A New value of sound port STA IO_03_SOUND1 ;1901 -OUT (SOUND1),A Write new value to sound hardware RTS ;1903 -RET Return InitAliensP2: LDX #Ply2DataArea ;1904 -LD HL,$2200 Player 2 data area Ply2DataArea LBRA L01C3 ;1907 -JP $01C3 Initialize player 2 aliens PlyrShotAndBump: JSR PlayerShotHit ;190A -CALL PlayerShotHit player's shot collision detection LBRA RackBump ;190D -JP RackBump Change alien deltaX and deltaY when rack bumps edges CurPlyAlive: ; Get the current player's alive status LDX #player1Alive ;1910 -LD HL,$20E7 Alive flags LDA playerDataMSB ;1913 -LD A,(playerDataMSB) Player 1 or 2 RORA ;1916 -RRCA Will be 1 if player 1 BCC > ;1917 -RET C Return if player 1 RTS ! * Jump here LEAX 1,X ;1918 -INC HL Bump to player 2 RTS ;1919 -RET Return DrawScoreHead: ; Print score header " SCORE<1> HI-SCORE SCORE<2> " LDB #$1C ;191A -LD C,$1C 28 bytes in message STB C_Temp LDX #Screen_Start+$1E ;191C -LD HL,$241E Screen coordinates LDY #MessageScore ;191F -LD DE,$1AE4 Score header message LBRA PrintMessage ;1922 -JP PrintMessage Print score header L1925 LDX #P1ScorL ;1925 -LD HL,$20F8 Player 1 score descriptor LBRA DrawScore ;1928 -JP DrawScore Print score L192B LDX #P2ScorL ;192B -LD HL,$20FC Player 2 score descriptor LBRA DrawScore ;192E -JP DrawScore Print score DrawScore: ; Print score. ; HL = descriptor LDB ,X+ LDA ,X+ TFR D,Y LDB ,X+ LDA ,X+ TFR D,X ; LDE ,X ;1931 -LD E,(HL) Get score LSB ; LEAX 1,X ;1932 -INC HL Next ; LDD ,X ;1933 -LD D,(HL) Get score MSB ; LEAX 1,X ;1934 -INC HL Next ; LDA ,X ;1935 -LD A,(HL) Get coordinate LSB ; LEAX 1,X ;1936 -INC HL Next ; LDH ,X ;1937 -LD H,(HL) Get coordiante MSB ; TFR A,L ;1938 -LD L,A Set LSB LBRA Print4Digits ;1939 -JP Print4Digits Print 4 digits in DE ; Print message "CREDIT " L193C LDB #$07 ;193C -LD C,$07 7 bytes in message STB C_Temp LDX #Screen_Start+$1101 ;193E -LD HL,$3501 Screen coordinates LDY #MessageCredit ;1941 -LD DE,$1FA9 Message = "CREDIT " LBRA PrintMessage ;1944 -JP PrintMessage Print message DrawNumCredits: ; Display number of credits on screen LDA numCoins ;1947 -LD A,(numCoins) Number of credits LDX #Screen_Start+$1801 ;194A -LD HL,$3C01 Screen coordinates LBRA DrawHexByte ;194D -JP DrawHexByte Character to screen (OK - needs fixing) PrintHiScore: LDX #hiscoredesc ;1950 -LD HL,$20F4 Hi Score descriptor LBRA DrawScore ;1953 -JP DrawScore Print Hi-Score DrawStatus: ; Print scores (with header) and credits (with label) STA isrDelay ;0AD7 -LD (isrDelay),A Delay counter ! LDA isrDelay ;0ADA -LD A,(isrDelay) Get current delay TSTA ;0ADD -AND A Zero yet? LBNE < ;0ADE -JP NZ,$0ADA No ... wait on it ;0AE1 -RET Out JSR ClearScreen ;1956 -CALL ClearScreen Clear the screen ; JSR TestScreen JSR DrawScoreHead ;1959 -CALL DrawScoreHead Print score header (OK - needs fixing) JSR L1925 ;195C -CALL $1925 Print player 1 score (OK - needs fixing) JSR L192B ;195F -CALL $192B Print player 2 score (OK - needs fixing) JSR PrintHiScore ;1962 -CALL PrintHiScore Print hi score (OK - needs fixing) JSR L193C ;1965 -CALL $193C Print credit lable (OK - needs fixing) LBRA DrawNumCredits ;1968 -JP DrawNumCredits Number of credits (OK - needs fixing) L196B JSR SoundBits3Off ;196B -CALL SoundBits3Off From 170B with B=FB. Turn off player shot sound LBRA L1671 ;196E -JP $1671 Update high-score if player's score is greater L1971 LDA #$01 ;1971 -LD A,$01 Set flag that ... STA invaded ;1973 -LD (invaded),A ... aliens reached bottom of screen LBRA L16E6 ;1976 -JP $16E6 End of round L1979 JSR DsableGameTasks ;1979 -CALL DsableGameTasks Disable ISR game tasks JSR DrawNumCredits ;197C -CALL DrawNumCredits Display number of credits on screen LBRA L193C ;197F -JP $193C Print message "CREDIT" L1982 STA isrSplashTask ;1982 -LD (isrSplashTask),A Set ISR splash task RTS ;1985 -RET Done ; The original code (from TAITO) printed this message on the screen. When Midway branched the code ; they changed the logic so it isn't printed. ;1986: ; FCB $8B,$19 ; Points to print TAITO CORPORATION message ... not sure why L1988 LBRA ClearPlayField ;1988 -JP ClearPlayField Clear playfield and out ;Print "*TAITO CORPORATION*" LDX #Screen_Start+$403 ;198B -LD HL,$2803 Screen coordinates LDY #MessageTaito ;198E -LD DE,$19BE Message "*TAITO CORPORATION*" LDB #$13 ;1991 -LD C,$13 Messgae length STB C_Temp LBRA PrintMessage ;1993 -JP PrintMessage Print message ; The original TAITO code: ;1985: C3 8B 19 JP $198B ; ;1988: CD D6 09 CALL $09D6 ; ;198B: 21 03 28 LD HL,$2803 ; ;198E: 11 BE 19 LD DE,$19BE ; ;1991: 0E 13 LD C,$13 ; ;1993: C3 F3 08 JP $08F3 ; ;1996: ; FCB $00,$00,$00,$00 ; ** Why? CheckHiddenMes: ; There is a hidden message "TAITO COP" (with no "R") in the game. It can only be ; displayed in the demonstration game during the splash screens. You must enter ; 2 seqences of buttons. Timing is not critical. As long as you eventually get all ; the buttons up/down in the correct pattern then the game will register the ; sequence. ; ; 1st: 2start(down) 1start(up) 1fire(down) 1left(down) 1right(down) ; 2nd: 2start(up) 1start(down) 1fire(down) 1left(down) 1right(up) ; ; Unfortunately MAME does not deliver the simultaneous button presses correctly. You can see the message in ; MAME by changing 19A6 to 02 and 19B1 to 02. Then the 2start(down) is the only sequence. ; LDA hidMessSeq ;199A -LD A,(hidMessSeq) Has the 1st "hidden-message" sequence ... TSTA ;199D -AND A ... been registered? LBNE L19AC ;199E -JP NZ,$19AC Yes ... go look for the 2nd sequence LDA IO_01_INP1 ;19A1 -IN A,(INP1) Get player inputs ANDA #$76 ;19A3 -AND $76 0111_0110 Keep 2Pstart, 1Pstart, 1Pshot, 1Pleft, 1Pright SUBA #$72 ;19A5 -SUI $72 0111_0010 1st sequence: 2Pstart, 1Pshot, 1Pleft, 1Pright BEQ > ;19A7 -RET NZ Not first sequence ... out RTS ! * Jump here INCA ;19A8 -INC A Flag that 1st sequence ... STA hidMessSeq ;19A9 -LD (hidMessSeq),A ... has been entered L19AC LDA IO_01_INP1 ;19AC -IN A,(INP1) Check inputs for 2nd sequence ANDA #$76 ;19AE -AND $76 0111_0110 Keep 2Pstart, 1Pstart, 1Pshot, 1Pleft, 1Pright CMPA #$34 ;19B0 -CP $34 0011_0100 2nd sequence: 1Pstart, 1Pshot, 1Pleft BEQ > ;19B2 -RET NZ If not second sequence ignore RTS ! * Jump here LDX #Screen_Start+$A1B ;19B3 -LD HL,$2E1B Screen coordinates LDY #MessageCorp ;19B6 -LD DE,$0BF7 Message = "TAITO COP" (no R) LDB #$09 ;19B9 -LD C,$09 Message length STB C_Temp LBRA PrintMessage ;19BB -JP PrintMessage Print message and out MessageTaito: ; "*TAITO CORPORATION*" ;19BE: FCB $28,$13,$00,$08,$13,$0E,$26,$02,$0E,$11,$0F,$0E,$11 ;19CB: FCB $00,$13,$08,$0E,$0D,$28 EnableGameTasks: ; Enable ISR game tasks LDA #$01 ;19D1 -LD A,$01 Set ISR ... L19D3 STA suspendPlay ;19D3 -LD (suspendPlay),A ... game tasks enabled RTS ;19D6 -RET Done DsableGameTasks: ; Disable ISR game tasks ; Clear 20E9 flag CLRA ;19D7 -XOR A Clear ISR game tasks flag LBRA L19D3 ;19D8 -JP $19D3 Save a byte (the RET) ; FCB $00 ;19DB - ** Here is the byte saved. I wonder if this was an optimizer pass. SoundBits3Off: ; Turn off bit in sound port STB B_Temp LDA soundPort3 ;19DC -LD A,(soundPort3) Current sound effects value ANDA B_Temp ;19DF -AND B Mask bits off STA soundPort3 ;19E0 -LD (soundPort3),A Store new hold value STA IO_03_SOUND1 ;19E3 -OUT (SOUND1),A Change sounds RTS ;19E5 -RET Done DrawNumShips: ; Show ships remaining in hold for the player LDX #Screen_Start+$301 ;19E6 -LD HL,$2701 Screen coordinates TSTA * Added TSTA because the 8080 is still comparing the DECA from the previous rotuine (8080 ignores CC on LD and ST instructions) LBEQ L19FA ;19E9 -JP Z,$19FA None in reserve ... skip display ; Draw line of ships L19EC STA A_Temp ! LDY #PlayerSprite ;19EC -LD DE,$1C60 Player sprite LDB #$10 ;19EF -LD B,$10 16 rows ; TFR A,C ;19F1 -LD C,A Hold count JSR DrawSimpSprite ;19F2 -CALL DrawSimpSprite Display 1byte sprite to screen ; TFR C,A ;19F5 -LD A,C Restore remaining DEC A_Temp ;19F6 -DEC A All done? LBNE < ;19F7 -JP NZ,$19EC No ... keep going ; Clear remainder of line L19FA LDB #$10 ;19FA -LD B,$10 16 rows JSR ClearSmallSprite ;19FC -CALL ClearSmallSprite Clear 1byte sprite at HL TFR X,D ;19FF -LD A,H Get Y coordinate CMPA #$35 ;1A00 -CP $35 At edge? LBNE L19FA ;1A02 -JP NZ,$19FA No ... do all RTS ;1A05 -RET Out CompYToBeam: ; ; The ISRs set the upper bit of 2072 based on where the beam is. This is compared to the ; upper bit of an object's Y coordinate to decide whic ISR should handle it. When the ; beam passes the halfway point (or near it ... at scanline 96), the upper bit is cleared. ; When the beam reaches the end of the screen the upper bit is set. ; ; The task then runs in the ISR if the Y coordiante bit matches the 2072 flag. Objects that ; are at the top of the screen (upper bit of Y clear) run in the mid-screen ISR when ; the beam has moved to the bottom of the screen. Objects that are at the bottom of the screen ; (upper bit of Y set) run in the end-screen ISR when the beam is moving back to the top. ; ; The pointer to the object's Y coordinate is passed in DE. CF is set if the upper bits are ; the same (the calling ISR should execute the task). ; LDX #vblankStatus ;1A06 -LD HL,$2072 Get the ... LDB ,X ;1A09 -LD B,(HL) ... beam position status LDA ,Y ;1A0A -LD A,(DE) Get the task structure flag ANDA #$80 ;1A0B -AND $80 Only upper bits count EORA ,X ;1A0D -XOR B XOR them together BEQ > ;1A0E -RET NZ Not the same (CF cleared) ANDCC #$FE ; Clear the Carry Flag RTS ; ! * Jump here ORCC #$01 ; Set the Carry Flag RTS ;1A10 -RET Done ; Alien delay lists. First list is the number of aliens. The second list is the corresponding delay. ; This delay is only for the rate of change in the fleet's sound. ; The check takes the first num-aliens-value that is lower or the same as the actual num-aliens on screen. ; ; The game starts with 55 aliens. The aliens are move/drawn one per interrupt which means it ; takes 55 interrupts. The first delay value is 52 ... which is almost in sync with the number ; of aliens. It is a tad faster and you can observe the sound and steps getting out of sync. ; AlienDelLst1: ;1A11: FCB $32,$2B,$24,$1C,$16,$11,$0D,$0A,$08,$07,$06,$05,$04,$03,$02,$01 AlienDelLst2: ;1A21: FCB $34,$2E,$27,$22,$1C,$18,$15,$13,$10,$0E,$0D,$0C,$0B,$09,$07,$05 ;1A31: FCB $FF ; ** Needless terminator. The list value "1" catches everything. BlockCopy: ; Copy from [DE] to [HL] (b bytes) LDA ,Y ;1A32 -LD A,(DE) Copy from [DE] to ... STA ,X ;1A33 -LD (HL),A ... [HL] LEAX 1,X ;1A34 -INC HL Next destination LEAY 1,Y ;1A35 -INC DE Next source DECB ;1A36 -DEC B Count in B LBNE BlockCopy ;1A37 -JP NZ,BlockCopy Do all RTS ;1A3A -RET Done ReadDesc: ; Load 5 bytes sprite descriptor from [HL] ; LDE ,X ;1A3B -LD E,(HL) Descriptor ... ; LEAX 1,X ;1A3C -INC HL ... sprite ... ; LDD ,X ;1A3D -LD D,(HL) ... ; LEAX 1,X ;1A3E -INC HL ... picture ; LDA ,X ;1A3F -LD A,(HL) Descriptor ... ; LEAX 1,X ;1A40 -INC HL ... screen ... ; LDC ,X ;1A41 -LD C,(HL) ... ; LEAX 1,X ;1A42 -INC HL ... location ; LDB ,X ;1A43 -LD B,(HL) Number of bytes in sprite ; TFR C,H ;1A44 -LD H,C From A,C to ... ; TFR A,L ;1A45 -LD L,A ... H,L LDB ,X+ LDA ,X+ TFR D,Y LDB ,X+ LDA ,X+ TFR D,U LDB ,X+ TFR U,X RTS ;1A46 -RET Done ConvToScr: ;HL = HL/8 + 2000 (screen coordinate) Proably need to fix pointer to screen location here... ; The screen is organized as one-bit-per-pixel. ; In: HL contains pixel number (bbbbbbbbbbbbbppp) ; Convert from pixel number to screen coordinates (without shift) ; Shift HL right 3 bits (clearing the top 2 bits) ; and set the third bit from the left. PSHS B ;1A47 -PUSH BC Hold B (will mangle) LDB #$03 ;1A48 -LD B,$03 3 shifts (divide by 8) STX H_Temp ; TFR X,D ; RORA ; RORB ; RORA ; RORB ; RORA ; RORB L1A4A LDA H_Temp ;1A4A -LD A,H H to A RORA ;1A4B -RRA Shift right (into carry, from doesn't matter) STA H_Temp ;1A4C -LD H,A Back to H LDA L_Temp ;1A4D -LD A,L L to A RORA ;1A4E -RRA Shift right (from/to carry) STA L_Temp ;1A4F -LD L,A Back to L DECB ;1A50 -DEC B Do all ... LBNE L1A4A ;1A51 -JP NZ,$1A4A ... 3 shifts LDA H_Temp ;1A54 -LD A,H H to A ANDA #$3F ;1A55 -AND $3F Mask off all but screen (less than or equal 3F) ORA #$20 ;1A57 -OR $20 Offset into RAM STA H_Temp ;1A59 -LD H,A Back to H LDX H_Temp PULS B ;1A5A -POP BC Restore B RTS ;1A5B -RET Done ClearScreen: ; Clear the screen ; Thanks to Mark Tankard for pointing out what this really does LDX #Screen_Start ;1A5C -LD HL,$2400 Screen coordinate L1A5F CLR ,X+ ;1A5F -LD (HL),$00 Clear it ; LEAX 1,X ;1A61 -INC HL Next byte ; TFR H,A ;1A62 -LD A,H Have we done ... CMPX #Screen_Start+$1C20 ;1A63 -CP $40 ... all the screen? LBNE L1A5F ;1A65 -JP NZ,$1A5F No ... keep going RTS ;1A68 -RET Out RestoreShields: ; Logically OR the player's shields back onto the playfield ; DE = sprite ; HL = screen ; C = bytes per row ; B = number of rows ;; PSHS U ;1A69 -PUSH BC Preserve BC ; PSHS X ;1A6A -PUSH HL Hold for a bit ; LDB C_Temp ;L1A6B LDA ,Y ;1A6B -LD A,(DE) From sprite ; ORA ,X ;1A6C -OR (HL) OR with screen ; STA ,X ;1A6D -LD (HL),A Back to screen ; LEAY 1,Y ;1A6E -INC DE Next sprite ; LEAX 1,X ;1A6F -INC HL Next on screen ; DECB ; C_Temp ;1A70 -DEC C Row done? ; LBNE L1A6B ;1A71 -JP NZ,$1A6B No ... do entire row ; PULS X ;1A74 -POP HL Original start0 ;; LDU #$0020 ;1A75 -LD BC,$0020 Bump HL by ... ;; ADD X=X+U ;1A78 -ADD HL,BC ... one screen row ; LEAX $20,X ;; PULS U ;1A79 -POP BC Restore ; DEC B_Temp ;1A7A -DEC B Row counter ; LBNE RestoreShields ;1A7B -JP NZ,RestoreShields Do all rows ; RTS ;1A7E -RET Return *** Removed the reversal below and decided to reverse the original shield data, as it was being revsersed here but after a restore from a two player *** Game it would get double reversed... which would mess it up. NOP Test NOP ; PSHS U ;1A69 -PUSH BC Preserve BC ; LDU #ReverseLookUp * Fix reversal of bit order point to table PSHS X ;1A6A -PUSH HL Hold for a bit LDB C_Temp L1A6B LDA ,Y ;1A6B -LD A,(DE) From sprite ORA ,X ;1A6C -OR (HL) OR with screen ; LDA A,U * Fix reversal of bit order load value from table STA ,X ;1A6D -LD (HL),A Back to screen LEAY 1,Y ;1A6E -INC DE Next sprite LEAX 1,X ;1A6F -INC HL Next on screen DECB ; C_Temp ;1A70 -DEC C Row done? LBNE L1A6B ;1A71 -JP NZ,$1A6B No ... do entire row PULS X ;1A74 -POP HL Original start0 ; LDU #$0020 ;1A75 -LD BC,$0020 Bump HL by ... ; ADD X=X+U ;1A78 -ADD HL,BC ... one screen row LEAX $20,X ; PULS U ;1A79 -POP BC Restore DEC B_Temp ;1A7A -DEC B Row counter LBNE Test ; Should be LBNE RestoreShields ;1A7B -JP NZ,RestoreShields Do all rows RTS ;1A7E -RET Return RemoveShip: ; Remove a ship from the players stash and update the ; hold indicators on the screen. JSR L092E ;1A7F -CALL $092E Get last byte from player data TSTA ;1A82 -AND A Is it 0? BNE > ;1A83 -RET Z Skip RTS ! * Jump here PSHS A ;1A84 -PUSH AF Preserve number remaining DECA ;1A85 -DEC A Remove a ship from the stash STA ,X ;1A86 -LD (HL),A New number of ships JSR DrawNumShips ;1A87 -CALL DrawNumShips Draw the line of ships PULS A ;1A8A -POP AF Restore number L1A8B LDX #Screen_Start+$101 ;1A8B -LD HL,$2501 Screen coordinates ANDA #$0F ;1A8E -AND $0F Make sure it is a digit LBRA L09C5 ;1A90 -JP $09C5 Print number remaining SI_End: ;================================================================================================================== ; DATA FROM HERE DOWN ORG $1A93 DataStart: ; ;1A93: FCB $00,$00 SplashScrnAnm: ; Splash screen animation structure 1 ; 00 Image form (increments each draw) ; 00 Delta X ; FF Delta Y is -1 ; B8 X coordinate ; FE Y starting coordiante ; 1C20 Base image (small alien) ; 10 Size of image (16 bytes) ; 9E Target Y coordiante ; 00 Reached Y flag ; 1C20 Base iamge (small alien) ;1A95: FCB $00,$00,$FF,$B8,$FE,$20,$1C,$10,$9E,$00,$20,$1C ShotReloadRate: ; The tables at 1CB8 and 1AA1 control how fast shots are created. The speed is based ; on the upper byte of the player's score. For a score of less than or equal 0200 then ; the fire speed is 30. For a score less than or equal 1000 the shot speed is 10. Less ; than or equal 2000 the speed is 0B. Less than or equal 3000 is 08. And anything ; above 3000 is 07. ; ; 1CB8: 02 10 20 30 ; ;1AA1: FCB $30,$10,$0B,$08 ;1AA5: FCB $07 ; Fastest shot firing speed MessageGOver: ; "GAME OVER PLAYER< >" ;1AA6: FCB $06,$00,$0C,$04,$26,$0E,$15,$04,$11,$26,$26,$0F ;1AB2: FCB $0B,$00,$18,$04,$11,$24,$26,$25 MessageB1or2: ; "1 OR 2PLAYERS BUTTON" ;1ABA: FCB $1B,$26,$0E,$11,$26,$1C,$0F,$0B,$00,$18,$04 ;1AC5: FCB $11,$12,$26,$01,$14,$13,$13,$0E,$0D,$26 Message1Only: ; "ONLY 1PLAYER BUTTON " ; Note the space on the end ... both alternatives are same length ;1ACF: FCB $0E,$0D,$0B,$18,$26,$1B,$0F,$0B,$00,$18,$04,$11,$26,$26 ;1ADD: FCB $01,$14,$13,$13,$0E,$0D,$26 MessageScore: ; " SCORE<1> HI-SCORE SCORE<2>" ;1AE4: FCB $26,$12,$02,$0E,$11,$04,$24,$1B,$25,$26,$07,$08 ;1AF0: FCB $3F,$12,$02,$0E,$11,$04,$26,$12,$02,$0E,$11,$04 ;1AFC: FCB $24,$1C,$25,$26 RAMInit: ;-------------------------- RAM initialization ----------------------------- ; Coppied to RAM (2000) C0 bytes as initialization. ; See the description of RAM at the top of this file for the details on this data. ;1B00: FCB $01 ;waitOnDraw FCB $00 FCB $00 ;alienIsExploding FCB $10 ;expAlienTimer FCB $00 ;alienRow FCB $00 ;alienanimframe FCB $00 ;alienCurIndex FCB $00 ;refAlienDYr FCB $02 ;refAlienDXr FCB $78 ;refAlienYr FCB $38 ;refAlienXr FCB $78 ;alienPosLSB FCB $38 ;alienPosMSB FCB $00 ;rackDirection FCB $F8 ;rackDownDelta FCB $00 ;1B10: - Game object 0: Move/draw the player FCB $00 ;firstgameobj FCB $80 ;obj0TimerLSB FCB $00 ;obj0TimerExtra ; FCB $8E ;GameObj0 address pointer LSB - Orignianl ; FCB $02 ;GameObj0 address pointer MSB - Orignianl FDB GameObj0 ; Original GameObj0 address pointer $8E,$02 address ($028E) FCB $FF ;playerAlive FCB $05 FCB $0C FCB $60 ;plyrSprPicL FCB $1C FCB $20 ;playerYr FCB $30 ;playerXr FCB $10 FCB $01 ;nextDemoCmd FCB $00 ;hidMessSeq FCB $00 ;1820: - Game object 1: Move/draw the player shot FCB $00 ;objStruct1 FCB $00 FCB $00 ; FCB $BB ;GameObj1 address pointer LSB - Orignianl ; FCB $03 ;GameObj1 address pointer MSB - Orignianl FDB GameObj1 ; Original GameObj1 address pointer $BB,$03 address ($03BB) FCB $00 ;plyrShotStatus FCB $10 FCB $90 ;plyShotStruct FCB $1C FCB $28 ;obj1CoorYr FCB $30 ;obj1CoorXr FCB $01 FCB $04 ;shotDeltaX FCB $00 ;fireBounce FCB $FF FCB $FF ;1B30 - Game object 2: Allien rolling-shot (targets player specifically) FCB $00 ;objStruct2 FCB $00 FCB $02 ;obj2TimerExtra ; FCB $76 ;GameObj2 address pointer LSB - Orignianl ; FCB $04 ;GameObj2 address pointer MSB - Orignianl FDB GameObj2 ; Original GameObj2 address pointer $76,$04 address ($0476) FCB $00 ;rolShotdatstct FCB $00 ;rolShotStepCnt FCB $00 FCB $00 ;rolShotCFirLSB FCB $00 FCB $04 FCB $EE FCB $1C FCB $00 FCB $00 FCB $03 ;1B40 - Game object 3: Alien plunger-shot FCB $00 ;objStruct3 FCB $00 FCB $00 ; FCB $B6 ;GameObj3 address pointer LSB - Orignianl ; FCB $04 ;GameObj3 address pointer MSB - Orignianl FDB GameObj3 ; Original GameObj3 address pointer $B6,$04 address ($04B6) FCB $00 ;pluShotdata FCB $00 ;pluShotStepCnt FCB $01 FCB $00 ;pluShotCFirLSB FCB $1D FCB $04 FCB $E2 FCB $1C FCB $00 FCB $00 FCB $03 ;1B50 - Game object 4: Flying Saucer OR squiggly shot FCB $00 ;objStruct4 FCB $00 FCB $00 ; FCB $82 ;GameObj4 address pointer LSB - Orignianl ; FCB $06 ;GameObj4 address pointer MSB - Orignianl FDB GameObj4 ; Original GameObj4 address pointer $82,$06 address ($0682) FCB $00 ;squShotStatus FCB $00 ;squShotStepCnt FCB $01 FCB $06 ;squShotCFirLSB FCB $1D FCB $04 FCB $D0 FCB $1C FCB $00 FCB $00 FCB $03 ;1B60 FCB $FF FCB $00 ;collision FCB $C0 ;exploAliendes FCB $1C FCB $00 ;expAlienYr FCB $00 FCB $10 FCB $21 ;playerDataMSB FCB $01 ;playerOK FCB $00 ;enableAlienFire FCB $30 ;alienFireDelay FCB $00 ;flag1Alien FCB $12 ;temp206C FCB $00 ;invaded FCB $00 ;skipPlunger FCB $00 ; These don't need to be copied over to RAM (see 1BA0 below). MesssageP1: ; "PLAY PLAYER<1>" ;1B70 FCB $0F ;otherShot1 FCB $0B ;otherShot2 FCB $00 ;vblankStatus FCB $18 ;aShotStatus FCB $26 FCB $0F FCB $0B ;aShotCFirLSB FCB $00 FCB $18 ;aShotBlowCnt FCB $04 ;aShotImageLSB FCB $11 FCB $24 ;alienShotYr FCB $1B ;alienShotYc FCB $25 ;alienShotSize FCB $FC ;alienShotDelta FCB $00 ;shotPicEnd ;1B80 FCB $01 ;shotSync FCB $FF ;tmp2081 FCB $FF ;numAliens DataSaucer: FCB $00 ;saucerStart FCB $00 ;saucerActive FCB $00 ;saucerHit FCB $20 FCB $64 ;saucerPriLocLSB FCB $1D FCB $D0 FCB $29 ;saucerPriPicMSB FCB $18 FCB $02 FCB $54 ;sauScoreLSB FCB $1D FCB $00 ;shotCountLSB ;1B90 FCB $08 FCB $00 ;tillSaucerLSB FCB $06 FCB $00 ;waitStartLoop FCB $00 ;soundPort3 FCB $01 ;changeFleetSnd FCB $40 ;curtimesltsnd FCB $00 ;fleetSndReload FCB $01 ;soundPort5 FCB $00 ;extraHold FCB $00 ;tilt FCB $10 ;fleetSndHold FCB $9E FCB $00 FCB $20 FCB $1C ; These don't need to be copied over to RAM I believe this to be a mistake. The constant at 01E4 is C0, ; which is the size of this mirror with the added sprite. It should be A0. I believe there was a macro ; to size this area and later the splash screens where put in. Some of the data spilled over into this ; and the macro automatically included it. No harm. AlienSprCYA: ; Alien sprite type C pulling upside down Y ; ........ ; **...... ; ..*..... ; ...****. ; ..*.*... ; **..*... ; ...*.... ; .*.**... ; *.****.. ; ...*.**. ; ..****** ; ..****** ; ...*.**. ; *.****.. ; .*.**... ; ........ ; ;1BA0: FCB $00,$03,$04,$78,$14,$13,$08,$1A,$3D,$68,$FC,$FC,$68,$3D,$1A,$00 AlienSprCYA1: ;1BB0: FCB $00,$00,$01,$B8,$98,$A0,$1B,$10,$FF,$00,$A0,$1B,$00,$00,$00,$00 ;--------------------------- End of initialization copy ------------------------- ShotExtraC: ; Shot descriptor for splash shooting the extra "C" ;1BC0: FCB $00,$10,$00 ; FCB $0E,$05 FDB AlienShootC ; Changed these two bytes from original $0E,$05 ($050E) jump address to new location FCB $00,$00,$00,$00,$00,$07,$D0,$1C,$C8,$9B,$03 AlienSprCYB: ; Alien sprite C pulling upside down Y. Note the difference between this and the first picutre ; above. The Y is closer to the ship. This gives the effect of the Y kind of "sticking" in the ; animation. ; ........ ; ........ ; **...... ; ..*..... ; ...****. ; ..*.*... ; **.*.... ; *..**... ; .*.***.. ; *.**.**. ; .*.***** ; .*.***** ; *.**.**. ; .*.***.. ; *..**... ; ........ ; ;Nothing points to this ??? does get copied From routine at $18d9,copies 256 bytes from ROM $1B00 to RAM $2000, so $1BD0 will actually be $20D0??? Check ;1BD0: FCB $00,$00,$03,$04,$78,$14,$0B,$19,$3A,$6D,$FA,$FA,$6D,$3A,$19,$00 ; More RAM initialization copied by 18D9 ;1BE0: FCB $00,$00,$00,$00,$00,$00,$00,$00,$00,$01,$00,$00,$01,$74,$1F,$00 ;1BF0: FCB $80,$00,$00,$00,$00,$00,$1C,$2F,$00,$00,$1C,$27,$00,$00,$1C,$39 ; From $1B00 to $1BFF is copied to RAM $2000 to $20FF (I think these are where the symbol tables should be loaded from) ;Alien Images AlienSprA: ; Alien sprite type A,B, and C at positions 0 ; ........ ........ ........ ; ........ ........ ........ ; *..***.. ........ ........ ; *..****. ...****. ........ ; .*.****. *.***... *..**... ; .***.**. .*****.* .*.***.. ; ..**.*** ..**.**. *.**.**. ; .*.***** ..****.. .*.***** ; .*.***** ..****.. .*.***** ; ..**.*** ..****.. *.**.**. ; .***.**. ..**.**. .*.***.. ; .*.****. .*****.* *..**... ; *..****. *.***... ........ ; *..***.. ...****. ........ ; ........ ........ ........ ; ........ ........ ........ ;1C00: FCB $00,$00,$39,$79,$7A,$6E,$EC,$FA,$FA,$EC,$6E,$7A,$79,$39,$00,$00 ;1C10: FCB $00,$00,$00,$78,$1D,$BE,$6C,$3C,$3C,$3C,$6C,$BE,$1D,$78,$00,$00 ;1C20: FCB $00,$00,$00,$00,$19,$3A,$6D,$FA,$FA,$6D,$3A,$19,$00,$00,$00,$00 AlienSprB: ; Alien sprite type A,B, and C at positions 1 ; ........ ........ ........ ; ........ ........ ........ ; ...***.. ........ ........ ; .*.****. .***.... ........ ; *******. ...**... .*.**... ; *.**.**. .*****.* *.****.. ; ..**.*** *.**.**. ...*.**. ; .*.***** *.****.. ..****** ; .*.***** ..****.. ..****** ; ..**.*** *.****.. ...*.**. ; *.**.**. *.**.**. *.****.. ; *******. .*****.* .*.**... ; .*.****. ...**... ........ ; ...***.. .***.... ........ ; ........ ........ ........ ; ........ ........ ........ ;1C30: FCB $00,$00,$38,$7A,$7F,$6D,$EC,$FA,$FA,$EC,$6D,$7F,$7A,$38,$00,$00 ;1C40: FCB $00,$00,$00,$0E,$18,$BE,$6D,$3D,$3C,$3D,$6D,$BE,$18,$0E,$00,$00 ;1C50: FCB $00,$00,$00,$00,$1A,$3D,$68,$FC,$FC,$68,$3D,$1A,$00,$00,$00,$00 PlayerSprite: ; Player sprite ; ........ ; ........ ; ****.... ; *****... ; *****... ; *****... ; *****... ; *******. ; ******** ; *******. ; *****... ; *****... ; *****... ; *****... ; ****.... ; ........ ;1C60: FCB $00,$00,$0F,$1F,$1F,$1F,$1F,$7F,$FF,$7F,$1F,$1F,$1F,$1F,$0F,$00 PlrBlowupSprites: ; = ; ........ ; ..*..... ; *....... ; **..*... ; **...... ; ***..... ; **..**.* ; ****.... ; ****.*.. ; **...... ; ****.*.. ; *..*..*. ; ..*..... ; **...... ; ........ ; *....... ; ;1C70: FCB $00,$04,$01,$13,$03,$07,$B3,$0F,$2F,$03,$2F,$49,$04,$03,$00,$01 ; ; ......*. ; ...*.... ; *.*..... ; **...*.* ; .*.*.... ; **...... ; **.**.*. ; ****.... ; ***..*.. ; ***..*.. ; **.*.... ; **.*..*. ; ......*. ; ..*....* ; *...*... ; ...*..*. ;1C80: FCB $40,$08,$05,$A3,$0A,$03,$5B,$0F,$27,$27,$0B,$4B,$40,$84,$11,$48 PlayerShotSpr: ;1C90: FCB $0F ; ****.... ; FCB $F0 ; ****.... - Flipped this byte so shot displays normal ShotExploding: ; *..**..* ; ..****.. ; .******. ; *.****.. ; ..****.* ; .*****.. ; ..*****. ; *..**..* ;1C91: FCB $99,$3C,$7E,$3D,$BC,$3E,$7C,$99 Message10Pts: ; Ran out of space at 1DFE ; "=10 POINTS" ;1C99: FCB $27,$1B,$1A,$26,$0F,$0E,$08,$0D,$13,$12 MessageAdv: ; "*SCORE ADVANCE TABLE*" ;1CA3: FCB $28,$12,$02,$0E,$11,$04,$26,$00 ;1CAB: FCB $03,$15,$00,$0D,$02,$04,$26,$13 ;1CB3: FCB $00,$01,$0B,$04,$28 AReloadScoreTab: ; The tables at 1CB8 and 1AA1 control how fast shots are created. The speed is based ; on the upper byte of the player's score. For a score of less than or equal 0200 then ; the fire speed is 30. For a score less than or equal 1000 the shot speed is 10. Less ; than or equal 2000 the speed is 0B. Less than or equal 3000 is 08. And anything ; above 3000 is 07. ; ; 1AA1: 30 10 0B 08 ; 1AA5: 07 ; Fastest shot firing speed ; ;1CB8: FCB $02,$10,$20,$30 MessageTilt: ; "TILT" ;1CBC: FCB $13,$08,$0B,$13 AlienExplode: ; Alien exploding sprite ; ........ ; ...*.... ; *..*..*. ; .*...*.. ; ..*.*... ; *......* ; .*....*. ; ........ ; .*....*. ; *......* ; ..*.*... ; .*...*.. ; *..*..*. ; ...*.... ; ........ ; ........ ;1CC0: FCB $00,$08,$49,$22,$14,$81,$42,$00,$42,$81,$14,$22,$49,$08,$00,$00 ;SquiglyShot: ; = ; Squigly shot picture in 4 animation frames ; ..*...*. ; .*.*.*.* ; ....*... ; ...*...* ; ..*.*.*. ; .*...*.. ; ....*... ; .*.*.*.* ; ..*...*. ; .*...*.. ; ..*.*.*. ; ...*...* ;1CD0: FCB $44,$AA,$10,$88,$54,$22,$10,$AA,$44,$22,$54,$88 AShotExplo: ; Alien shot exploding ; .*.*..*. ; *.*.*... ; .*****.* ; ******.. ; .****.*. ; *.*..*.. ;1CDC: FCB $4A,$15,$BE,$3F,$5E,$25 PlungerShot: ; Alien shot ... the plunger looking one ; ..*..... ; ..****** ; ..*..... ; ....*... ; ..****** ; ....*... ; .....*.. ; ..****** ; .....*.. ; .......* ; ..****** ; .......* ;1CE2: FCB $04,$FC,$04,$10,$FC,$10,$20,$FC,$20,$80,$FC,$80 RollShot: ; Alien shot ... the rolling one ; ........ ; .******* ; ........ ; ..*..*.. ; .******* ; .*..*... ; ........ ; .******* ; ........ ; ...*..*. ; .******* ; ....*..* ;1CEE: FCB $00,$FE,$00,$24,$FE,$12,$00,$FE,$00,$48,$FE,$90 MessagePlayUY: ;1CFA: FCB $0F,$0B,$00,$29 ; "PLAy" with an upside down 'Y' for splash screen ;1CFE: FCB $00,$00 ColFireTable: ; This table decides which column a shot will fall from. The column number is read from the ; table (1-11) and the pointer increases for the shot type. For instance, the "squiggly" shot ; will fall from columns in this order: 0B, 01, 06, 03. If you play the game you'll see that ; order. ; ; The "plunger" shot uses index 00-0F (inclusive) ; The "squiggly" shot uses index 06-14 (inclusive) ; The "rolling" shot targets the player ;1D00: FCB $01,$07,$01,$01,$01,$04,$0B,$01,$06,$03,$01,$01,$0B,$09,$02,$08 ;1D10: FCB $02,$0B,$04,$07,$0A ; ; This appears to be part of the column-firing table, but it is never used. ; Perhaps this was originally intended for the "rolling" shot but then the ; "rolling" was change to target the player specifically. ;1D15: FCB $05,$02,$05,$04,$06,$07,$08,$0A,$06,$0A,$03 ShieldImage: ; Shield image pattern. 2 x 22 = 44 bytes. ; ;************.... ;*************... ;**************.. ;***************. ;**************** ;..************** ;...************* ;....************ ;....************ ;....************ ;....************ ;....************ ;....************ ;....************ ;...************* ;..************** ;**************** ;**************** ;***************. ;**************.. ;*************... ;************.... ; ;1D20: FCB $FF,$0F,$FF,$1F,$FF,$3F,$FF,$7F,$FF,$FF,$FC,$FF,$F8,$FF,$F0,$FF,$F0,$FF,$F0,$FF,$F0,$FF ; Original Data ;1D36: FCB $F0,$FF,$F0,$FF,$F0,$FF,$F8,$FF,$FC,$FF,$FF,$FF,$FF,$FF,$FF,$7F,$FF,$3F,$FF,$1F,$FF,$0F ; Original Data TabPosSaScor: ;1D4C: FCB $05,$10,$15,$30 ; Table of possible saucer scores TabCorStrPrnt: ;1D50: FCB $94,$97,$9A,$9D ; Table of corresponding string prints for each possible score SaucerScrTab: ; 208D points here to the score given when the saucer is shot. It advances ; every time the player-shot is removed. The code wraps after 15, but there ; are 16 values in this table. This is a bug in the code at 044E (thanks to ; Colin Dooley for finding this). ; ; Thus the one and only 300 comes up every 15 shots (after an initial 8). ;1D54: FCB $10,$05,$05,$10,$15,$10,$10,$05,$30,$10,$10,$10,$05,$15,$10,$05 SpriteSaucer: ; ........ ; ........ ; ........ ; ........ ; ..*..... ; ..**.... ; .****... ; ***.**.. ; .*****.. ; ..*****. ; ..*.***. ; .******. ; .******. ; ..*.***. ; ..*****. ; .*****.. ; ***.**.. ; .****... ; ..**.... ; ..*..... ; ........ ; ........ ; ........ ; ........ ;1D64: FCB $00,$00,$00,$00,$04,$0C,$1E,$37,$3E,$7C,$74,$7E,$7E,$74,$7C,$3E,$37,$1E,$0C,$04,$00,$00,$00,$00 SpriteSaucerExp: ;........ ;.*...*.. ;........ ;*.*..*.* ;......*. ;...*.... ;...**..* ;*.****.. ;.**.**.* ;..****.. ;.**.**.. ;*.***... ;....*... ;...*..*. ;.*...**. ;.**.**.* ;*.***... ;...**..* ;...*.... ;.*....*. ;....*..* ;...*.... ;........ ;........ ;1D7C: FCB $00,$22,$00,$A5,$40,$08,$98,$3D,$B6,$3C,$36,$1D,$10,$48,$62,$B6,$1D,$98,$08,$42,$90,$08,$00,$00 SaucSoreStr: ;1D94: FCB $26,$1F,$1A ; _50 ;1D97: FCB $1B,$1A,$1A ; 100 ;1D9A: FCB $1B,$1F,$1A ; 150 ;1D9D: FCB $1D,$1A,$1A ; 300 AlienScores: ; Score table for hitting alien type ;1DA0: FCB $10 ; Bottom 2 rows ;1DA1: FCB $20 ; Middle row ;1DA2: FCB $30 ; Highest row AlienStartTable: ; Starting Y coordinates for aliens at beginning of rounds. The first round is initialized to $78 at 07EA. ; After that this table is used for 2nd, 3rd, 4th, 5th, 6th, 7th, 8th, and 9th. The 10th starts over at ; 1DA3 (60). ;1DA3: FCB $60 ;1DA4: FCB $50 ;1DA5: FCB $48 ;1DA6: FCB $48 ;1DA7: FCB $48 ;1DA8: FCB $40 ;1DA9: FCB $40 ;1DAA: FCB $40 MessagePlayY: ;1DAB: FCB $0F,$0B,$00,$18 ; "PLAY" with normal Y MessageInvaders: ; "SPACE INVADERS" ;1DAF: FCB $12,$0F,$00,$02,$04,$26,$26,$08,$0D,$15,$00,$03,$04,$11,$12 ScoreAdvTable: ; Tables used to draw "SCORE ADVANCE TABLE" information ;1DBE: FCB $0E,$2C,$68,$1D ; Flying Saucer ;1DC2: FCB $0C,$2C,$20,$1C ; Alien C, sprite 0 ;1DC6: FCB $0A,$2C,$40,$1C ; Alien B, sprite 1 ;1DCA: FCB $08,$2C,$00,$1C ; Alien A, sprite 0 ;1DCE: FCB $FF ; End of list ; AlienScoreTable: ;1DCF: FCB $0E,$2E,$E0,$1D ; "=? MYSTERY" ;1DD3: FCB $0C,$2E,$EA,$1D ; "=30 POINTS" ;1DD7: FCB $0A,$2E,$F4,$1D ; "=20 POINTS" ;1DDB: FCB $08,$2E,$99,$1C ; "=10 POINTS" ;1DDF: FCB $FF ; End of list MessageMyst: ;1DE0: FCB $27,$38,$26,$0C,$18,$12,$13,$04,$11,$18 ; "=? MYSTERY" Message30Pts: ;1DEA: FCB $27,$1D,$1A,$26,$0F,$0E,$08,$0D,$13,$12 ; "=30 POINTS" Message20Pts: ;1DF4: FCB $27,$1C,$1A,$26,$0F,$0E,$08,$0D,$13,$12 ; "=20 POINTS" ; Ran out of space here. The "=10" message is up at 1C99. That keeps ; the font table firmly at 1E00. ;1DFE: FCB $00,$00 ; Padding to put font table at 1E00 ;============================================================= Characters: ; 8 byte sprites ; The screen is turned so rotate these pictures counter-clockwise. ; Some of the font characters at the end were never needed. The ROM overwrites these characters with ; data near the end. For instance, 1F90 would be a character but has the "INSERT COIN" message. The "?" ; character is at 1FC0 and is used in messages as is 1FF8 "-". The "light colored" tiles in the grid below ; show the character slots that have been repurposed. ;1E00: FCB $00,$1F,$24,$44,$24,$1F,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1E08: FCB $00,$7F,$49,$49,$49,$36,$00,$00 ; *****... *******. .*****.. *******. *******. *******. .*****.. *******. ;1E10: FCB $00,$3E,$41,$41,$41,$22,$00,$00 ; ..*..*.. *..*..*. *.....*. *.....*. *..*..*. ...*..*. *.....*. ...*.... ;1E18: FCB $00,$7F,$41,$41,$41,$3E,$00,$00 ; ..*...*. *..*..*. *.....*. *.....*. *..*..*. ...*..*. *.....*. ...*.... ;1E20: FCB $00,$7F,$49,$49,$49,$41,$00,$00 ; ..*..*.. *..*..*. *.....*. *.....*. *..*..*. ...*..*. *.*...*. ...*.... ;1E28: FCB $00,$7F,$48,$48,$48,$40,$00,$00 ; *****... .**.**.. .*...*.. .*****.. *.....*. ......*. ***...*. *******. ;1E30: FCB $00,$3E,$41,$41,$45,$47,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1E38: FCB $00,$7F,$08,$08,$08,$7F,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1E40: FCB $00,$00,$41,$7F,$41,$00,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1E48: FCB $00,$02,$01,$01,$01,$7E,$00,$00 ; ........ .*...... *******. *******. *******. *******. .*****.. *******. ;1E50: FCB $00,$7F,$08,$14,$22,$41,$00,$00 ; *.....*. *....... ...*.... *....... .....*.. ....*... *.....*. ...*..*. ;1E58: FCB $00,$7F,$01,$01,$01,$01,$00,$00 ; *******. *....... ..*.*... *....... ...**... ...*.... *.....*. ...*..*. ;1E60: FCB $00,$7F,$20,$18,$20,$7F,$00,$00 ; *.....*. *....... .*...*.. *....... .....*.. ..*..... *.....*. ...*..*. ;1E68: FCB $00,$7F,$10,$08,$04,$7F,$00,$00 ; ........ .******. *.....*. *....... *******. *******. .*****.. ....**.. ;1E70: FCB $00,$3E,$41,$41,$41,$3E,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1E78: FCB $00,$7F,$48,$48,$48,$30,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1E80: FCB $00,$3E,$41,$45,$42,$3D,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1E88: FCB $00,$7F,$48,$4C,$4A,$31,$00,$00 ; .*****.. *******. .*..**.. ......*. .******. ..*****. *******. **...**. ;1E90: FCB $00,$32,$49,$49,$49,$26,$00,$00 ; *.....*. ...*..*. *..*..*. ......*. *....... .*...... .*...... ..*.*... ;1E98: FCB $00,$40,$40,$7F,$40,$40,$00,$00 ; *.*...*. ..**..*. *..*..*. *******. *....... *....... ..**.... ...*.... ;1EA0: FCB $00,$7E,$01,$01,$01,$7E,$00,$00 ; .*....*. .*.*..*. *..*..*. ......*. *....... .*...... .*...... ..*.*... ;1EA8: FCB $00,$7C,$02,$01,$02,$7C,$00,$00 ; *.****.. *...**.. .**..*.. ......*. .******. ..*****. *******. **...**. ;1EB0: FCB $00,$7F,$02,$0C,$02,$7F,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1EB8: FCB $00,$63,$14,$08,$14,$63,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1EC0: FCB $00,$60,$10,$0F,$10,$60,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1EC8: FCB $00,$43,$45,$49,$51,$61,$00,$00 ; .....**. **....*. .*****.. ........ **...*.. .*....*. ..**.... .*..***. ;1ED0: FCB $00,$3E,$45,$49,$51,$3E,$00,$00 ; ....*... *.*...*. *.*...*. *....*.. *.*...*. *.....*. ..*.*... *...*.*. ;1ED8: FCB $00,$00,$21,$7F,$01,$00,$00,$00 ; ****.... *..*..*. *..*..*. *******. *..*..*. *..*..*. ..*..*.. *...*.*. ;1EE0: FCB $00,$23,$45,$49,$49,$31,$00,$00 ; ....*... *...*.*. *...*.*. *....... *..*..*. *..**.*. *******. *...*.*. ;1EE8: FCB $00,$42,$41,$49,$59,$66,$00,$00 ; .....**. *....**. .*****.. ........ *...**.. .**..**. ..*..... .***..*. ;1EF0: FCB $00,$0C,$14,$24,$7F,$04,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1EF8: FCB $00,$72,$51,$51,$51,$4E,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1F00: FCB $00,$1E,$29,$49,$49,$46,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1F08: FCB $00,$40,$47,$48,$50,$60,$00,$00 ; .****... ......*. .**.**.. *...**.. ...*.... ........ ........ ..*.*... ;1F10: FCB $00,$36,$49,$49,$49,$36,$00,$00 ; *..*.*.. ***...*. *..*..*. *..*..*. ..*.*... *.....*. ........ ..*.*... ;1F18: FCB $00,$31,$49,$49,$4A,$3C,$00,$00 ; *..*..*. ...*..*. *..*..*. *..*..*. .*...*.. .*...*.. ........ ..*.*... ;1F20: FCB $00,$08,$14,$22,$41,$00,$00,$00 ; *..*..*. ....*.*. *..*..*. .*.*..*. *.....*. ..*.*... ........ ..*.*... ;1F28: FCB $00,$00,$41,$22,$14,$08,$00,$00 ; .**...*. .....**. .**.**.. ..****.. ........ ...*.... ........ ..*.*... ;1F30: FCB $00,$00,$00,$00,$00,$00,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1F38: FCB $00,$14,$14,$14,$14,$14,$00,$00 ; ........ ........ ........ ........ ........ ........ ........ ........ ;1F40: FCB $00,$22,$14,$7F,$14,$22,$00,$00 ; ........ ........ ;1F48: FCB $00,$03,$04,$78,$04,$03,$00,$00 ; .*...*.. **...... ; ..*.*... ..*..... ; *******. ...****. ; ..*.*... ..*..... ; .*...*.. **...... ; ........ ........ ; ........ ........ MessageP1or2: ;1F50: FCB $24,$1B,$26,$0E,$11,$26,$1C,$26 ; "<1 OR 2 PLAYERS> " ;1F58: FCB $0F,$0B,$00,$18,$04,$11,$12,$25 ;1F60: FCB $26,$26 Message1Coin: ;1F62: FCB $28,$1B,$26,$0F,$0B,$00,$18,$04 ; "*1 PLAYER 1 COIN " ;1F6A: FCB $11,$26,$26,$1B,$26,$02,$0E,$08 ;1F72: FCB $0D,$26 DemoCommands: ; (1=Right, 2=Left) ;1F74: FCB $01,$01,$00,$00,$01,$00,$02,$01,$00,$02,$01,$00 AlienSprCA: ; Small alien pushing Y back onto screen ; .....**. ; ....*... ; ****.... ; ....*... ; .....**. ; ....**.. ; ...**... ; .*.**... ; *.****.. ; ...*.**. ; ..****** ; ..****** ; ...*.**. ; *.****.. ; .*.**... ; ........ ;1F80: FCB $60,$10,$0F,$10,$60,$30,$18,$1A,$3D,$68,$FC,$FC,$68,$3D,$1A,$00 MessageCoin: ;1F90: FCB $08,$0D,$12,$04,$11,$13,$26,$26,$02,$0E,$08,$0D ; "INSERT COIN" CreditTable: ;1F9C: FCB $0D,$2A,$50,$1F ; "<1 OR 2 PLAYERS> " to screen at 2A0D CreditTable1: ;1FA0: FCB $0A,$2A,$62,$1F ; "*1 PLAYER 1 COIN " to screen at 2A0A ;1FA4: FCB $07,$2A,$E1,$1F ; "*2 PLAYERS 2 COINS" to screen at 2A07 ;1FA8: FCB $FF ; Terminates "table print" MessageCredit: ;1FA9: FCB $02,$11,$04,$03,$08,$13,$26 ; "CREDIT " (with space on the end) AlienSprCB: ; ........ ; .....**. ; ....*... ; ****.... ; ....*... ; .....**. ; ...***.. ; *..**... ; .*.***.. ; *.**.**. ; .*.***** ; .*.***** ; *.**.**. ; .*.***.. ; *..**... ; ........ ;1FB0: FCB $00,$60,$10,$0F,$10,$60,$38,$19,$3A,$6D,$FA,$FA,$6D,$3A,$19,$00 ;1FC0: FCB $00,$20,$40,$4D,$50,$20,$00,$00 ; "?" ;1FC8: FCB $00 SplshScrnAnm3: ; Splash screen animation structure 3 ; 00 Image form (increments each draw) ; 00 Delta X ; FF Delta Y is -1 ; B8 X coordinate ; FF Y starting coordiante ; 1F80 Base image (small alien with Y) ; 10 Size of image (16 bytes) ; 97 Target Y coordiante ; 00 Reached Y flag ; 1F80 Base iamge (small alien with Y) ; ;1FC9: FCB $00,$00,$FF,$B8,$FF,$80,$1F,$10,$97,$00,$80,$1F SplshScrnAnm4: ; Splash screen animation structure 4 ; 00 Image form (increments each draw) ; 00 Delta X ; 01 Delta Y is 1 ; D0 X coordinate ; 22 Y starting coordiante ; 1C20 Base image (small alien) ; 10 Size of image (16 bytes) ; 94 Target Y coordiante ; 00 Reached Y flag ; 1C20 Base iamge (small alien) ; ;1FD5: FCB $00,$00,$01,$D0,$22,$20,$1C,$10,$94,$00,$20,$1C Message2Coins: ;1FE1: FCB $28,$1C,$26,$0F,$0B,$00,$18,$04 ; "*2 PLAYERS 2 COINS" ;1FE9: FCB $11,$12,$26,$1C,$26,$02,$0E,$08 ;1FF1: FCB $0D,$12 MessagePush: ;1FF3: FCB $0F,$14,$12,$07,$26 ; "PUSH " (with space on the end) ;1FF8: FCB $00,$08,$08,$08,$08,$08,$00,$00 ; 3F:"-" ************************************************************************************************************** ; SI address $2000 - gets overwritten with a block copy from 1B00-1BBF to 2000-20BF waitOnDraw FCB $01 FCB $00 alienIsExploding FCB $00 expAlienTimer FCB $10 alienRow FCB $00 alienanimframe FCB $00 alienCurIndex FCB $00 refAlienDYr FCB $00 refAlienDXr FCB $02 refAlienYr FCB $78 refAlienXr FCB $38 alienPosLSB FCB $78 FCB $38 rackDirection FCB $00 rackDownDelta FCB $F8 FCB $00 ;2010 - Game object 0: Move/draw the player firstgameobj FCB $00 obj0TimerLSB FCB $80 obj0TimerExtra FCB $00 FCB $8E ;GameObj0 address pointer LSB - gets overwritten with a block copy from 1B00-1BBF to 2000-20BF FCB $02 ;GameObj0 address pointer MSB - gets overwritten with a block copy from 1B00-1BBF to 2000-20BF playerAlive FCB $FF FCB $05 FCB $0C plyrSprPicL FCB $60 FCB $1C playerYr FCB $20 playerXr FCB $30 FCB $10 nextDemoCmd FCB $01 hidMessSeq FCB $00 FCB $00 ;2020 - Game object 1: Move/draw the player shot objStruct1 FCB $00 FCB $00 FCB $00 FCB $BB ;GameObj1 address pointer LSB - gets overwritten with a block copy from 1B00-1BBF to 2000-20BF FCB $03 ;GameObj1 address pointer MSB - gets overwritten with a block copy from 1B00-1BBF to 2000-20BF plyrShotStatus FCB $00 FCB $10 plyShotStruct FCB $90 FCB $1C obj1CoorYr FCB $28 obj1CoorXr FCB $30 FCB $01 shotDeltaX FCB $04 fireBounce FCB $00 FCB $FF FCB $FF ;2030 - Game object 2: Allien rolling-shot (targets player specifically) objStruct2 FCB $00 FCB $00 obj2TimerExtra FCB $02 FCB $76 ;GameObj2 address pointer LSB - gets overwritten with a block copy from 1B00-1BBF to 2000-20BF FCB $04 ;GameObj2 address pointer MSB - gets overwritten with a block copy from 1B00-1BBF to 2000-20BF rolShotdatstct FCB $00 rolShotStepCnt FCB $00 FCB $00 rolShotCFirLSB FCB $00 FCB $00 FCB $04 FCB $EE FCB $1C FCB $00 FCB $00 FCB $03 ;2040 - Game object 3: Alien plunger-shot objStruct3 FCB $00 FCB $00 FCB $00 FCB $B6 ;GameObj3 address pointer LSB - gets overwritten with a block copy from 1B00-1BBF to 2000-20BF FCB $04 ;GameObj3 address pointer MSB - gets overwritten with a block copy from 1B00-1BBF to 2000-20BF pluShotdata FCB $00 pluShotStepCnt FCB $00 FCB $01 pluShotCFirLSB FCB $00 FCB $1D FCB $04 FCB $E2 FCB $1C FCB $00 FCB $00 FCB $03 ;2050 - Game object 4: Flying Saucer OR squiggly shot objStruct4 FCB $00 FCB $00 FCB $00 FCB $82 ;GameObj4 address pointer LSB - gets overwritten with a block copy from 1B00-1BBF to 2000-20BF FCB $06 ;GameObj4 address pointer MSB - gets overwritten with a block copy from 1B00-1BBF to 2000-20BF squShotStatus FCB $00 squShotStepCnt FCB $00 FCB $01 squShotCFirLSB FCB $06 FCB $1D FCB $04 FCB $D0 FCB $1C FCB $00 FCB $00 FCB $03 ;2060 FCB $FF collision FCB $00 exploAliendes FCB $C0 FCB $1C expAlienYr FCB $00 FCB $00 FCB $10 playerDataMSB FCB $21 playerOK FCB $01 enableAlienFire FCB $00 alienFireDelay FCB $30 flag1Alien FCB $00 temp206C FCB $12 invaded FCB $00 skipPlunger FCB $00 FCB $00 ;2070 otherShot1 FCB $0F otherShot2 FCB $0B vblankStatus FCB $00 aShotStatus FCB $18 FCB $26 FCB $0F aShotCFirLSB FCB $0B FCB $00 aShotBlowCnt FCB $18 aShotImageLSB FCB $04 FCB $11 alienShotYr FCB $24 alienShotYc FCB $1B alienShotSize FCB $25 alienShotDelta FCB $FC shotPicEnd FCB $00 ;2080 shotSync FCB $01 tmp2081 FCB $FF numAliens FCB $FF saucerStart FCB $00 saucerActive FCB $00 saucerHit FCB $00 FCB $20 saucerPriLocLSB FCB $64 FCB $1D FCB $D0 saucerPriPicMSB FCB $29 FCB $18 FCB $02 sauScoreLSB FCB $54 FCB $1D shotCountLSB FCB $00 ;2090 FCB $08 tillSaucerLSB FCB $00 FCB $06 waitStartLoop FCB $00 soundPort3 FCB $00 changeFleetSnd FCB $01 curtimesltsnd FCB $40 fleetSndReload FCB $00 soundPort5 FCB $01 extraHold FCB $00 tilt FCB $00 fleetSndHold FCB $10 FCB $9E FCB $00 FCB $20 FCB $1C ;20A0 FCB $00 FCB $03 FCB $04 FCB $78 FCB $14 FCB $13 FCB $08 FCB $1A FCB $3D FCB $68 FCB $FC FCB $FC FCB $68 FCB $3D FCB $1A FCB $00 ;20B0 FCB $00 FCB $00 FCB $01 FCB $B8 FCB $98 FCB $A0 FCB $1B FCB $10 FCB $FF FCB $00 FCB $A0 FCB $1B FCB $00 FCB $00 FCB $00 FCB $00 ;20C0 isrDelay FCB $00 isrSplashTask FCB $10 splashAnForm FCB $00 FCB $0E FCB $05 xyimagedesc FCB $00 FCB $00 splashImageLSB FCB $00 FCB $00 FCB $00 splashTargetY FCB $07 splashReached FCB $D0 splashImRestLSB FCB $1C FCB $C8 twoPlayers FCB $9B aShotReloadRate FCB $03 ;20D0 FCB $00 FCB $00 FCB $03 FCB $04 FCB $78 FCB $14 FCB $0B FCB $19 FCB $3A FCB $6D FCB $FA FCB $FA FCB $6D FCB $3A FCB $19 FCB $00 ;20E0 FCB $00 FCB $00 FCB $00 FCB $00 FCB $00 player1Ex FCB $00 FCB $00 player1Alive FCB $00 FCB $00 suspendPlay FCB $01 coinSwitch FCB $00 numCoins FCB $00 splashAnimate FCB $01 demoCmdPtrLSB FCB $74 FCB $1F gameMode FCB $00 ;20F0 FCB $80 adjustScore FCB $00 scoreDeltaLSB FCB $00 pointscrdelta FCB $00 hiscoredesc FCB $00 curhiscorU2dig FCB $00 FCB $1C FCB $2F P1ScorL FCB $00 FCB $00 FCB $1C FCB $27 P2ScorL FCB $00 FCB $00 FCB $1C FCB $39 ******************************************************************************************************* ; ORG $2100 StartAlienStr RMB $42 ; $2100 Start of alien structures (pobably should fix this pointer) Ply1ShieldBuff RMB $B9 ; $2142 p1RefAlienDX FCB $00 ; $21fb p1RefAlienY FDB $0000 ; $21fc p1RackCnt FCB $00 ; $21FE p1ShipsRem FCB $00 ; $21ff Ply2DataArea RMB $42 ; $2200 Ply2ShieldBuff RMB $B9 ; $2242 p2RefAlienDX FCB $00 ; $22fb p2RefAlienYr FDB $0000 ; $22fc p2RackCnt FCB $00 ; $22fe p2ShipsRem FCB $00 ; $22ff *Symbol Table Variables used in Space Invaders, also faster if we access them in the Direct Page in RAM Memroy BYTES 0 to 255 Restore LDA #$3C * back to normal memory pointer LDX #$FFA4 * mapped to $8000-$9FFF, in blocks upto $F7FFF LDB #4 * 4 pages to load ! STA ,X+ INCA * next page DECB BNE < PULS U,D,Y,X * Restore the Registers ANDCC #$AF * Enable interrupts RTS * return back to BASIC BinToDec: ;Binary to Decimal ; LDA Binary CLR DECNUM Check99 CMPA #99 ; Greater than 99 (Decimal)? BHI TooBig OKAY CLRB TENS SUBA #10 BLO ALIGN INCB BRA TENS ALIGN LSLB LSLB LSLB LSLB STB TEMP UNITS ADDA #10 ORA TEMP STA DECNUM+1 BRA Finish TooBig SUBA #100 INC DECNUM Bra Check99 TEMP FCB 00 DECNUM FCB 00,00 Finish RTS ; Message ; A=0,B=1,C=3 ; 0=26=$1A, 1=27=$1B, ... ; Space = 38=$26 ORG $7000 ;7000 FCB 18,7,14,19,38 ; SHOT ;7005 FCB 38,38,38,38,38 ;700a FCB 21,1,11,0,13,10,38 ; VBLANK * autostart ORG $0167 JMP START Final END START * set exec address * end of file -