mirror of https://github.com/AbePralle/FGB.git
4716 lines
118 KiB
NASM
4716 lines
118 KiB
NASM
;class.asm
|
|
;gfx.asm
|
|
;map.asm
|
|
;object.asm
|
|
;music.asm
|
|
;---------------------------------------------------------------------
|
|
; user.asm
|
|
; 12.18.99 by Abe Pralle
|
|
;---------------------------------------------------------------------
|
|
|
|
INCLUDE "Source/defs.inc"
|
|
INCLUDE "Source/start.inc"
|
|
INCLUDE "Source/items.inc"
|
|
|
|
MIN_VBLANKS EQU 2
|
|
|
|
INTCLK EQU $83
|
|
EXTCLK EQU $82
|
|
|
|
|
|
|
|
|
|
SECTION "User",ROM0
|
|
|
|
;---------------------------------------------------------------------
|
|
; Subroutine: UserMain
|
|
; Description: Sets up game then goes into main loop
|
|
;---------------------------------------------------------------------
|
|
UserMain::
|
|
call InitGfx
|
|
|
|
;set up initial location + hero
|
|
;initial level 0009 (demo intro)
|
|
ld a,LEVELSTATEBANK
|
|
ld [$ff70],a
|
|
xor a
|
|
ld [curLevelStateIndex],a
|
|
|
|
xor a
|
|
ldio [mapState],a
|
|
ldio [mapState+1],a
|
|
|
|
;initial level index
|
|
ld a,((INITIALMAP>>8) & $ff)
|
|
ld [curLevelIndex+1],a
|
|
ld a,(INITIALMAP & $ff)
|
|
ld [curLevelIndex],a
|
|
|
|
ld hl,$1102
|
|
call SetJoinMap
|
|
call SetRespawnMap
|
|
|
|
ld a,[heroesUsed]
|
|
or INITTYPE0
|
|
ld [heroesUsed],a
|
|
|
|
;hero 0 initial setup
|
|
ld a,(INITHERO0 & $ff)
|
|
ld [hero0_class],a
|
|
ld a,((INITHERO0>>8)&$ff)
|
|
ld [hero0_class+1],a
|
|
ld a,(INITLOC0 & $ff)
|
|
ld [hero0_enterLevelLocation],a
|
|
ld a,((INITLOC0>>8) & $ff)
|
|
ld [hero0_enterLevelLocation+1],a
|
|
ld a,INITEXIT0
|
|
ld [hero0_enterLevelFacing],a
|
|
ld a,INITTYPE0
|
|
ld [hero0_type],a
|
|
xor a
|
|
ld [hero0_health],a
|
|
|
|
;hero 1 initial setup
|
|
ld a,(INITHERO1 & $ff)
|
|
ld [hero1_class],a
|
|
ld a,((INITHERO1>>8)&$ff)
|
|
ld [hero1_class+1],a
|
|
ld a,(INITLOC1 & $ff)
|
|
ld [hero1_enterLevelLocation],a
|
|
ld a,((INITLOC1>>8) & $ff)
|
|
ld [hero1_enterLevelLocation+1],a
|
|
ld a,INITEXIT1
|
|
ld [hero1_enterLevelFacing],a
|
|
ld a,INITTYPE1
|
|
ld [hero1_type],a
|
|
xor a
|
|
ld [hero1_health],a
|
|
|
|
ld a,1
|
|
ld [timeToChangeLevel],a
|
|
|
|
jr .loadNextLevel
|
|
|
|
.game ld a,[timeToChangeLevel]
|
|
or a
|
|
jr z,.continue
|
|
|
|
;switch levels
|
|
ld b,METHOD_DIE
|
|
call IterateAllLists
|
|
ld a,15
|
|
call SetupFadeToBlack
|
|
call WaitFade
|
|
|
|
.loadNextLevel
|
|
call LoadNextLevel
|
|
call UpdateObjects
|
|
call GentleCameraAdjust
|
|
call levelCheckRAM
|
|
call RedrawMap
|
|
ld a,15
|
|
call SetupFadeFromBlack
|
|
call WaitFade
|
|
;call LighteningIn
|
|
|
|
.continue
|
|
call MainLoopUpdateObjects
|
|
call GentleCameraAdjust
|
|
|
|
call levelCheckRAM
|
|
.afterLevelCheck
|
|
call RedrawMap
|
|
jr .game
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: SetActiveROM
|
|
; Argument: a - desired ROM bank
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
; Description: Switches to a new ROM bank by writing the new bank
|
|
; number to [$2100]. Also saves the bank in
|
|
; [curROMBank].
|
|
;---------------------------------------------------------------------
|
|
SetActiveROM::
|
|
ldio [curROMBank],a
|
|
ld [$2100],a
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: LongCall
|
|
; Arguments: stack:
|
|
; +0: parameter de
|
|
; +2: parameter af
|
|
; +4: empty (for addr of routine)
|
|
; +6: empty (return addr from routine)
|
|
; +8: af - old ROM bank
|
|
; +10: return addr from LongCall
|
|
; a - bank of routine
|
|
; de - address of routine
|
|
; Returns: Whatever routine normally returns (excluding F).
|
|
; Alters: af, hl
|
|
; Description:
|
|
;---------------------------------------------------------------------
|
|
LongCall::
|
|
ldio [curROMBank],a ;switch to ROM containing routine
|
|
ld [$2100],a
|
|
|
|
push hl
|
|
ld hl,sp+6 ;point hl to empty spot in stack
|
|
ld [hl],e
|
|
inc hl
|
|
ld [hl],d ;save addr of method
|
|
inc hl
|
|
ld de,.returnAddress
|
|
ld [hl],e ;save return addr of method
|
|
inc hl
|
|
ld [hl],d
|
|
pop hl
|
|
pop de
|
|
pop af
|
|
ret ;call method
|
|
|
|
.returnAddress
|
|
ld [longCallTempA],a
|
|
pop af ;get old ROM bank
|
|
ldio [curROMBank],a
|
|
ld [$2100],a
|
|
ld a,[longCallTempA]
|
|
|
|
ret ;return from this routine
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: LongCallNoArgs
|
|
; Arguments: a - RAM bank of routine
|
|
; hl - address of routine
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
; Description: Quick & dirty long call (handles bank swapping)
|
|
; for routines that don't require arguments or return
|
|
; values.
|
|
;---------------------------------------------------------------------
|
|
LongCallNoArgs::
|
|
push bc
|
|
|
|
ld b,a
|
|
ld a,[curROMBank]
|
|
push af ;save old bank
|
|
|
|
ld a,b
|
|
ld [curROMBank],a ;swap in new bank
|
|
ld [$2100],a
|
|
|
|
ld bc,.returnAddress
|
|
push bc
|
|
|
|
jp hl
|
|
.returnAddress
|
|
|
|
pop af ;retrieve old bank & swap it in
|
|
ld [curROMBank],a
|
|
ld [$2100],a
|
|
|
|
pop bc
|
|
ret
|
|
|
|
WaitFade::
|
|
call FadeStep
|
|
|
|
;took out distorts synchronization
|
|
;send a null control byte
|
|
;ld a,LNULL
|
|
;call ExchangeByte
|
|
;call HandleRemoteInput
|
|
|
|
call CheckSkip
|
|
ld a,[specialFX]
|
|
and FX_FADE
|
|
jr nz,WaitFade
|
|
|
|
;allow last palette to install
|
|
.waitInstall
|
|
ldio a,[paletteBufferReady] ;wait for vblank
|
|
or a
|
|
jr nz,.waitInstall
|
|
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: CheckSpecialFX
|
|
; Alters: af
|
|
; Description: Performs any special effects upkeep
|
|
;---------------------------------------------------------------------
|
|
CheckSpecialFX:
|
|
ld a,[specialFX]
|
|
and FX_FADE
|
|
jr z,.done
|
|
|
|
call FadeStep
|
|
ret
|
|
|
|
.done
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Subroutine: UpdateObjTimers
|
|
; Alters: af
|
|
; Description: Updates timers used for object movement
|
|
;---------------------------------------------------------------------
|
|
UpdateObjTimers::
|
|
ld a,[objTimerBase]
|
|
inc a
|
|
ld [objTimerBase],a
|
|
|
|
;put 2 LSB bits in 4:3
|
|
rlca
|
|
rlca
|
|
rlca
|
|
and %00011000
|
|
ld [objTimer60ths],a
|
|
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Subroutine: UpdateHeroTimers
|
|
; Alters: af
|
|
; Description: Updates timers used for hero/hero bullet movement
|
|
;---------------------------------------------------------------------
|
|
UpdateHeroTimers:
|
|
ld a,[heroTimerBase]
|
|
inc a
|
|
ld [heroTimerBase],a
|
|
|
|
;put 2 LSB bits in 4:3
|
|
rlca
|
|
rlca
|
|
rlca
|
|
and %00011000
|
|
ld [heroTimer60ths],a
|
|
|
|
ret
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Subroutine: VWait
|
|
; Description: Waits until a vertical blank occurs and then returns
|
|
;---------------------------------------------------------------------
|
|
VWait::
|
|
.vwait halt ;stop processor until interrupt
|
|
nop ;pad to prevent weird stuff after halt
|
|
|
|
;interrupt happened, let's look at our flag
|
|
ldio a,[vblankFlag]
|
|
or a ;a=a|a; tells us if vblankFlag is 1 or 0
|
|
jr z,.vwait ;not the vblank flag after all
|
|
|
|
;If we're here then a vblank has occurred. Get on with our lives.
|
|
xor a ;clear vblank flag
|
|
ldio [vblankFlag],a
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Subroutine: OnVBlank
|
|
; Description: Called every vertical blank. Sets "vblankFlag" to
|
|
; 1 to indicate a vertical blank has occurred
|
|
;---------------------------------------------------------------------
|
|
OnVBlank::
|
|
push af ;save all registers on stack
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
ld a,1 ;set "vblankFlag" to 1
|
|
ldio [vblankFlag],a
|
|
|
|
;store current VRAM bank in use
|
|
ld a,[$ff00+$4f]
|
|
push af
|
|
|
|
;update timers
|
|
ld hl,vblankTimer
|
|
inc [hl]
|
|
|
|
;is the backBuffer ready for DMA blit to screen?
|
|
ldio a,[backBufferReady]
|
|
or a
|
|
jr z,.checkDMALoad ;wasn't ready
|
|
|
|
ld a,[displayType]
|
|
or a
|
|
jr nz,.afterVblankCheck ;cinema type
|
|
|
|
ldio a,[vblankTimer]
|
|
ld hl,vblankTemp
|
|
sub [hl]
|
|
ldio [vblanksPerUpdate],a
|
|
|
|
;update no faster than 1-frame-every-so-many-blanks clocks
|
|
cp MIN_VBLANKS
|
|
jr c,.checkDMALoad ;too soon to update
|
|
|
|
;MUST SYNCHRONIZE WITH REMOTE TO DYNAMICALLY CHANGE THIS
|
|
;jr z,.rightOn ;exactly right
|
|
;updating too many objects; decrease iterateNumObjects
|
|
;ld hl,iterateNumObjects
|
|
;ld a,[hl]
|
|
;cp 17
|
|
;jr c,.rightOn ;don't risk going below 16 per update
|
|
|
|
;dec [hl]
|
|
|
|
.rightOn
|
|
ldio a,[vblankTimer] ;save time of this screen update
|
|
ldio [vblankTemp],a
|
|
|
|
.afterVblankCheck
|
|
ld hl,updateTimer
|
|
inc [hl]
|
|
|
|
xor a
|
|
ldio [backBufferReady],a ;signal we've blitted backbuffer
|
|
|
|
;set VRAM bank to 0
|
|
xor a
|
|
ld [$ff00+$4f],a
|
|
|
|
;use DMA to transfer sprite OAM buffer to VRAM
|
|
call SpriteDMAHandler
|
|
|
|
;Initiate DMA transfer of backbuffer to VRAM
|
|
|
|
ld hl,backBuffer
|
|
ld a,h
|
|
ld [$ff00+$51],a ;high byte of source
|
|
ld a,l
|
|
ld [$ff00+$52],a ;low byte of source
|
|
ldio a,[backBufferDestHighByte]
|
|
ld [$ff00+$53],a ;high byte of dest
|
|
xor a
|
|
ld [$ff00+$54],a ;low byte of dest
|
|
ld a,37 ;copy (37+1)*16 = 608 bytes
|
|
ld [$ff00+$55],a ;store length to start DMA
|
|
|
|
;set VRAM bank to 1 (tile attributes)
|
|
ld a,1
|
|
ld [$ff00+$4f],a
|
|
|
|
;Initiate DMA transfer of tile attributes to VRAM
|
|
ld hl,attributeBuffer
|
|
ld a,h
|
|
ld [$ff00+$51],a ;high byte of source
|
|
ld a,l
|
|
ld [$ff00+$52],a ;low byte of source
|
|
ldio a,[backBufferDestHighByte]
|
|
ld [$ff00+$53],a ;high byte of dest
|
|
xor a
|
|
ld [$ff00+$54],a ;low byte of dest
|
|
ld a,37 ;copy (37+1)*16 = 608 bytes
|
|
ld [$ff00+$55],a ;store length to start DMA
|
|
|
|
ld a,[desiredPixelOffset_x]
|
|
ld [curPixelOffset_x],a
|
|
ld a,[desiredPixelOffset_y]
|
|
ld [curPixelOffset_y],a
|
|
jr .scroll
|
|
|
|
.checkDMALoad
|
|
ldio a,[dmaLoad]
|
|
or a
|
|
jr z,.scroll
|
|
|
|
.checkLoadBank0
|
|
bit 0,a
|
|
jr z,.checkLoadBank1
|
|
|
|
.loadBank0
|
|
res 0,a
|
|
ldio [dmaLoad],a ;indicate we've loaded it
|
|
xor a
|
|
ld [$ff4f],a
|
|
|
|
;Initiate DMA transfer to bank 0 VRAM
|
|
ld a,[dmaLoadSrc0+1]
|
|
ld [$ff00+$51],a ;high byte of source
|
|
ld a,[dmaLoadSrc0]
|
|
ld [$ff00+$52],a ;low byte of source
|
|
ld a,[dmaLoadDest0+1]
|
|
ld [$ff00+$53],a ;high byte of dest
|
|
ld a,[dmaLoadDest0]
|
|
ld [$ff00+$54],a ;low byte of dest
|
|
ld a,[dmaLoadLen0] ;copy (N+1)*16 bytes
|
|
ld [$ff00+$55],a ;store length to start DMA
|
|
|
|
.checkLoadBank1
|
|
ldio a,[dmaLoad]
|
|
bit 1,a
|
|
jr z,.scroll
|
|
|
|
.loadBank1
|
|
res 1,a
|
|
ldio [dmaLoad],a ;indicate we've loaded it
|
|
ld a,1
|
|
ld [$ff4f],a
|
|
|
|
;Initiate DMA transfer to bank 1 VRAM
|
|
ld a,[dmaLoadSrc1+1]
|
|
ld [$ff00+$51],a ;high byte of source
|
|
ld a,[dmaLoadSrc1]
|
|
ld [$ff00+$52],a ;low byte of source
|
|
ld a,[dmaLoadDest1+1]
|
|
ld [$ff00+$53],a ;high byte of dest
|
|
ld a,[dmaLoadDest1]
|
|
ld [$ff00+$54],a ;low byte of dest
|
|
ld a,[dmaLoadLen1] ;copy (N+1)*16 bytes
|
|
ld [$ff00+$55],a ;store length to start DMA
|
|
|
|
.scroll ;Adjust scroll coordinates
|
|
ld hl,lineZeroHorizontalOffset
|
|
ldio a,[jiggleDuration]
|
|
or a
|
|
jr z,.useNormalScreenOffsets
|
|
|
|
;jiggle the screen!
|
|
dec a
|
|
ldio [jiggleDuration],a
|
|
|
|
;determine jiggle type
|
|
ld a,[jiggleType]
|
|
or a
|
|
jr nz,.takeoffJiggle
|
|
|
|
.normalJiggle
|
|
;pick a "random value"
|
|
ld a,[jiggleLoc]
|
|
add 17
|
|
ld [jiggleLoc],a
|
|
ld b,a ;store random value in b
|
|
and %00000111 ;random x offset
|
|
ldio [baseHorizontalOffset],a
|
|
add [hl] ;add lineZeroHorizontalOffset
|
|
ldio [$ff43],a
|
|
swap b
|
|
ld a,%00000111 ;random y offset
|
|
and b
|
|
ldio [$ff42],a
|
|
jr .done
|
|
|
|
.takeoffJiggle
|
|
ld a,[jiggleLoc]
|
|
add 17
|
|
ld [jiggleLoc],a
|
|
ld b,a ;store random value in b
|
|
and %11 ;x offset
|
|
ldio [$ff43],a
|
|
swap b
|
|
ld a,%11
|
|
and b
|
|
ld b,a
|
|
ld a,[curPixelOffset_y]
|
|
add b
|
|
cp 8
|
|
jr c,.takeoffVOkay
|
|
|
|
ld a,7
|
|
|
|
.takeoffVOkay
|
|
ldio [$ff42],a
|
|
jr .done
|
|
|
|
.useNormalScreenOffsets
|
|
ld a,[curPixelOffset_x]
|
|
ldio [baseHorizontalOffset],a
|
|
add [hl] ;add lineZeroHorizontalOffset
|
|
ldio [$ff43],a
|
|
ld a,[curPixelOffset_y]
|
|
ldio [$ff42],a
|
|
|
|
;new palette to copy?
|
|
ldio a,[paletteBufferReady]
|
|
or a
|
|
jr z,.afterPaletteCopy
|
|
|
|
ld c,16 ;c is count
|
|
ld hl,fadeCurPalette
|
|
ld a,%10000000
|
|
ld [$ff68],a
|
|
ld [$ff6a],a
|
|
.paletteCopyBG
|
|
ld a,[hl+] ;copy sets of 4 bytes to avoid minute but
|
|
ld [$ff69],a ;killer delays decrementing the counter
|
|
ld a,[hl+]
|
|
ld [$ff69],a
|
|
ld a,[hl+]
|
|
ld [$ff69],a
|
|
ld a,[hl+]
|
|
ld [$ff69],a
|
|
dec c
|
|
jr nz,.paletteCopyBG
|
|
|
|
ld c,16 ;c is count
|
|
.paletteCopyFG
|
|
ld a,[hl+]
|
|
ld [$ff6b],a
|
|
ld a,[hl+]
|
|
ld [$ff6b],a
|
|
ld a,[hl+]
|
|
ld [$ff6b],a
|
|
ld a,[hl+]
|
|
ld [$ff6b],a
|
|
dec c
|
|
jr nz,.paletteCopyFG
|
|
|
|
xor a
|
|
ldio [paletteBufferReady],a
|
|
|
|
.afterPaletteCopy
|
|
|
|
.done
|
|
;update music
|
|
ldio a,[musicEnabled]
|
|
and %10000
|
|
jr z,.doneMusic
|
|
|
|
ld hl,musicNoteCountdown
|
|
dec [hl]
|
|
call z,PlayMusic
|
|
|
|
.doneMusic
|
|
;restore VRAM bank that was in use
|
|
pop af
|
|
ld [$ff00+$4f],a
|
|
|
|
pop hl ;restore all regs from stack
|
|
pop de
|
|
pop bc
|
|
pop af
|
|
reti ;return from interrupt
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Subroutine: OnHBlank
|
|
; Description: Called after every line is drawn.
|
|
;---------------------------------------------------------------------
|
|
OnHBlank::
|
|
push af
|
|
push bc
|
|
push hl
|
|
|
|
OnHBlank_afterPush:
|
|
ldio a,[$ff41] ;get stat register
|
|
bit 2,a ;equal to lyc?
|
|
jp z,FinishHBlankPlayingSample
|
|
|
|
.continue
|
|
ld hl,hblankFlag
|
|
bit 0,[hl] ;turning window on or off?
|
|
jr nz,OnHBlank_turnOffWindow
|
|
|
|
;turn on window
|
|
bit 1,[hl] ;allowed to?
|
|
jr nz,.turnOn
|
|
jp z,FinishHBlankPlayingSample
|
|
.turnOn
|
|
set 0,[hl]
|
|
ldio a,[hblankWinOff]
|
|
ld [$ff45],a ;reset lyc to win off pos
|
|
ld hl,$ff40 ;turn window on
|
|
set 5,[hl]
|
|
|
|
OnHBlank_SetBGColorBlack:
|
|
;set background palette 0, color zero to black
|
|
ld b,0
|
|
ld c,$68
|
|
ld hl,$ff69
|
|
ld a,%10000000 ;specification 0
|
|
ld [c],a
|
|
ld [hl],b
|
|
ld [hl],b
|
|
|
|
;ld a,%10000110 ;white
|
|
;ld [c],a
|
|
;ld [hl],h
|
|
;ld [hl],h
|
|
|
|
ld a,%10001000 ;palette 1
|
|
ld [c],a
|
|
ld [hl],b
|
|
ld [hl],b
|
|
|
|
ld a,%10010000 ;palette 2
|
|
ld [c],a
|
|
ld [hl],b
|
|
ld [hl],b
|
|
|
|
ld a,%10011000 ;palette 3
|
|
ld [c],a
|
|
ld [hl],b
|
|
ld [hl],b
|
|
|
|
ld a,%10100000 ;palette 4
|
|
ld [c],a
|
|
ld [hl],b
|
|
ld [hl],b
|
|
|
|
ld a,%10101000 ;palette 5
|
|
ld [c],a
|
|
ld [hl],b
|
|
ld [hl],b
|
|
|
|
ld a,%10110000 ;palette 6
|
|
ld [c],a
|
|
ld [hl],b
|
|
ld [hl],b
|
|
|
|
ld a,%10111000 ;palette 7
|
|
ld [c],a
|
|
ld [hl],b
|
|
ld [hl],b
|
|
|
|
jp FinishHBlankPlayingSample
|
|
|
|
OnHBlank_turnOffWindow:
|
|
res 0,[hl]
|
|
ldio a,[hblankWinOn]
|
|
ld [$ff45],a ;reset lyc to win on pos
|
|
ld hl,$ff40 ;turn window off
|
|
res 5,[hl]
|
|
|
|
;restore background palette 0, color zero
|
|
push de
|
|
ld c,$68
|
|
ld hl,mapColor
|
|
ld a,[hl+]
|
|
ld d,[hl]
|
|
ld e,a
|
|
ld hl,$ff69
|
|
|
|
HBlankSetBGColorCDEHL_DEPushed:
|
|
ld a,%10000000 ;specification 0
|
|
ld [c],a
|
|
ld [hl],e
|
|
ld [hl],d
|
|
|
|
ld a,%10001000 ;specification 1
|
|
ld [c],a
|
|
ld [hl],e
|
|
ld [hl],d
|
|
|
|
ld a,%10010000 ;specification 2
|
|
ld [c],a
|
|
ld [hl],e
|
|
ld [hl],d
|
|
|
|
ld a,%10011000 ;specification 3
|
|
ld [c],a
|
|
ld [hl],e
|
|
ld [hl],d
|
|
|
|
ld a,%10100000 ;specification 4
|
|
ld [c],a
|
|
ld [hl],e
|
|
ld [hl],d
|
|
|
|
ld a,%10101000 ;specification 5
|
|
ld [c],a
|
|
ld [hl],e
|
|
ld [hl],d
|
|
|
|
ld a,%10110000 ;specification 6
|
|
ld [c],a
|
|
ld [hl],e
|
|
ld [hl],d
|
|
|
|
ld a,%10111000 ;specification 7
|
|
ld [c],a
|
|
ld [hl],e
|
|
ld [hl],d
|
|
|
|
;ld a,%10000110 ;white
|
|
;ld [c],a
|
|
;ld a,[fadeCurPalette+6]
|
|
;ld [hl],a
|
|
;ld a,[fadeCurPalette+7]
|
|
;ld [hl],a
|
|
|
|
pop de
|
|
|
|
.done
|
|
jp FinishHBlankPlayingSample
|
|
|
|
;---------------------------------------------------------------------
|
|
; Subroutine: CinemaOnHBlank
|
|
; Description: Called after every line is drawn.
|
|
;---------------------------------------------------------------------
|
|
CinemaOnHBlank::
|
|
push af
|
|
push bc
|
|
push hl
|
|
|
|
ldio a,[$ff41] ;get stat register
|
|
bit 2,a ;equal to lyc?
|
|
jp z,FinishHBlankPlayingSample
|
|
|
|
.continue
|
|
ld hl,hblankFlag
|
|
bit 0,[hl] ;turning window on or off?
|
|
jr nz,.turnOff
|
|
|
|
;turn on window
|
|
bit 1,[hl] ;allowed to?
|
|
jr nz,.turnOn
|
|
jp z,FinishHBlankPlayingSample
|
|
.turnOn
|
|
set 0,[hl]
|
|
ldio a,[hblankWinOff]
|
|
ld [$ff45],a ;reset lyc to win off pos
|
|
ld hl,$ff40 ;turn window on
|
|
set 5,[hl]
|
|
jp FinishHBlankPlayingSample
|
|
|
|
.turnOff
|
|
res 0,[hl]
|
|
ldio a,[hblankWinOn]
|
|
ld [$ff45],a ;reset lyc to win on pos
|
|
ld hl,$ff40 ;turn window off
|
|
res 5,[hl]
|
|
|
|
.done
|
|
jp FinishHBlankPlayingSample
|
|
|
|
;---------------------------------------------------------------------
|
|
; Subroutine: HOffsetOnHBlank
|
|
; Description: Called after every line is drawn. Sets each line's
|
|
; horizontal offset from horizontalOffset[144]
|
|
;---------------------------------------------------------------------
|
|
HOffsetOnHBlank::
|
|
push af
|
|
push bc
|
|
push hl
|
|
|
|
;save current RAM bank
|
|
ldio a,[$ff70]
|
|
push af
|
|
|
|
ld a,TILEINDEXBANK
|
|
ldio [$ff70],a
|
|
;add to the horizontal offset for next line
|
|
ld c,$44 ;get $ff44
|
|
ld a,[c] ;ypos of refresh (index to array)
|
|
dec c ;c=$43 ($ff43)
|
|
inc a
|
|
ld l,a
|
|
ld h,((horizontalOffset>>8)&$ff)
|
|
ldio a,[baseHorizontalOffset]
|
|
add [hl]
|
|
ld [c],a ;xpos of screen
|
|
|
|
.afterAdjustOffsets
|
|
pop af
|
|
ldio [$ff70],a
|
|
jp OnHBlank_afterPush
|
|
|
|
;---------------------------------------------------------------------
|
|
; Subroutine: SeasonsOnHBlank
|
|
; Description: Called after every line is drawn. Sets color 0 (all
|
|
; bg palettes) to white on line 143 and to brown ($888)
|
|
; on line 39.
|
|
;---------------------------------------------------------------------
|
|
SeasonsOnHBlank::
|
|
push af
|
|
push bc
|
|
push hl
|
|
|
|
ld a,[amShowingDialog]
|
|
or a
|
|
jr nz,.seasonsWindow
|
|
|
|
;jr z,.normal
|
|
;ld a,[hblankFlag]
|
|
;and %11
|
|
;jr nz,.seasonsWindow
|
|
|
|
.normal
|
|
ldio a,[$ff44] ;ypos
|
|
cp 143
|
|
jr z,.toWhite
|
|
cp 39
|
|
jp nz,.seasonsWindow
|
|
|
|
.toBrown
|
|
push de
|
|
ld c,$68
|
|
ld hl,fadeCurPalette+62
|
|
ld a,[hl+]
|
|
ld d,[hl]
|
|
ld e,a
|
|
ld hl,$ff69
|
|
jp HBlankSetBGColorCDEHL_DEPushed
|
|
|
|
.toWhite
|
|
push de
|
|
ld c,$68
|
|
ld hl,mapColor
|
|
ld a,[hl+]
|
|
ld d,[hl]
|
|
ld e,a
|
|
ld hl,$ff69
|
|
jp HBlankSetBGColorCDEHL_DEPushed
|
|
|
|
.seasonsWindow
|
|
ldio a,[$ff41] ;get stat register
|
|
bit 2,a ;equal to lyc?
|
|
jp z,FinishHBlankPlayingSample
|
|
|
|
.continue
|
|
ld hl,hblankFlag
|
|
bit 0,[hl] ;turning window on or off?
|
|
jr nz,.turnOff
|
|
|
|
;turn on window
|
|
bit 1,[hl] ;allowed to?
|
|
jr z,.toBrown
|
|
;jr nz,.turnOn
|
|
;jp FinishHBlankPlayingSample
|
|
.turnOn
|
|
set 0,[hl]
|
|
ldio a,[hblankWinOff]
|
|
ld [$ff45],a ;reset lyc to win off pos
|
|
ld hl,$ff40 ;turn window on
|
|
set 5,[hl]
|
|
jp OnHBlank_SetBGColorBlack
|
|
|
|
.turnOff
|
|
res 0,[hl]
|
|
ldio a,[hblankWinOn]
|
|
ld [$ff45],a ;reset lyc to win on pos
|
|
ld hl,$ff40 ;turn window off
|
|
res 5,[hl]
|
|
|
|
jr .toBrown
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: GetInput
|
|
; Description: Swaps in the link code bank, passes the work along to
|
|
; HandleInput, and swaps in the object bank before
|
|
; ending.
|
|
;---------------------------------------------------------------------
|
|
GetInput::
|
|
LONGCALLNOARGS HandleInput
|
|
|
|
;if the heroes are idle then set both inputs to zero
|
|
ld a,[heroesIdle]
|
|
or a
|
|
ret z
|
|
|
|
xor a
|
|
ld [curJoy0],a
|
|
ld [curJoy1],a
|
|
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: CheckPause
|
|
; Description: Checks for pause button being pressed
|
|
;---------------------------------------------------------------------
|
|
CheckPause:
|
|
ld hl,dialogJoyIndex
|
|
ld a,[hl]
|
|
push af
|
|
|
|
ld a,[heroesPresent]
|
|
cp %11
|
|
jr z,.checkBoth
|
|
|
|
dec a
|
|
ld [hl],a
|
|
ld a,[myJoy]
|
|
bit JOY_START_BIT,a
|
|
jr z,.done
|
|
jr .checkModes
|
|
|
|
.checkBoth
|
|
ld [hl],0
|
|
ld a,[curInput0]
|
|
bit JOY_START_BIT,a
|
|
jr nz,.checkModes
|
|
|
|
ld [hl],1
|
|
ld a,[curInput1]
|
|
bit JOY_START_BIT,a
|
|
jr z,.done
|
|
|
|
.checkModes
|
|
ld a,[amShowingDialog]
|
|
or a
|
|
jr nz,.done
|
|
|
|
ld a,[displayType]
|
|
or a
|
|
jr nz,.done
|
|
|
|
ld a,[amChangingMap]
|
|
or a
|
|
jr nz,.done
|
|
|
|
ld a,[heroesIdle]
|
|
or a
|
|
jr nz,.done
|
|
|
|
;(show as dialog)
|
|
ld a,[dialogSettings]
|
|
push af
|
|
call SetPressBDialog
|
|
call SaveIdle
|
|
|
|
ld de,.afterPause
|
|
call SetDialogSkip
|
|
|
|
ld a,BANK(GeneratePauseMessage)
|
|
ld hl,GeneratePauseMessage
|
|
call LongCallNoArgs
|
|
|
|
ld a,1
|
|
ld [amShowingDialog],a
|
|
ld de,backBuffer+63
|
|
|
|
call ShowDialogAtTopCommon
|
|
call DialogWaitInputZero
|
|
|
|
.wait
|
|
call UpdateObjects
|
|
call RedrawMap
|
|
call SetDialogJoy
|
|
bit JOY_SELECT_BIT,[hl]
|
|
jr nz,.selectPressed
|
|
call CheckDialogContinue
|
|
or a
|
|
jr z,.wait
|
|
|
|
.afterPause
|
|
call ClearDialog
|
|
|
|
call RestoreIdle
|
|
pop af
|
|
ld [dialogSettings],a
|
|
.done
|
|
pop af
|
|
ld [dialogJoyIndex],a
|
|
ret
|
|
|
|
.selectPressed
|
|
call ClearDialog
|
|
call RestoreIdle
|
|
pop af
|
|
ld [dialogSettings],a
|
|
pop af
|
|
ld [dialogJoyIndex],a
|
|
|
|
ld a,EXIT_D
|
|
ld [hero0_enterLevelFacing],a
|
|
ld [hero1_enterLevelFacing],a
|
|
|
|
ld hl,respawnMap
|
|
ld a,[hl+]
|
|
ld [curLevelIndex],a
|
|
ld a,[hl+]
|
|
ld [curLevelIndex+1],a
|
|
ld a,1
|
|
ld [timeToChangeLevel],a
|
|
ret
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: UpdateObjects
|
|
; Description: Waits for a vertical blank and then updates the
|
|
; objects
|
|
;---------------------------------------------------------------------
|
|
MainLoopUpdateObjects:
|
|
call VWait ;wait for a vertical blank
|
|
ld a,1
|
|
ld [checkInputInMainLoop],a
|
|
call GetInput
|
|
call CheckPause
|
|
xor a
|
|
ld [checkInputInMainLoop],a
|
|
jr .doneWithInput
|
|
|
|
.vwait call VWait ;wait for a vertical blank
|
|
|
|
.doneWithInput
|
|
call CheckSpecialFX
|
|
|
|
;wait for previous frame to be drawn
|
|
ldio a,[backBufferReady]
|
|
or a
|
|
jp nz,.vwait
|
|
jr UpdateObjectsCommon
|
|
|
|
UpdateObjects::
|
|
call VWait ;wait for a vertical blank
|
|
call GetInput
|
|
|
|
.vwait call VWait ;wait for a vertical blank
|
|
|
|
.doneWithInput
|
|
call CheckSpecialFX
|
|
|
|
;wait for previous frame to be drawn
|
|
ldio a,[backBufferReady]
|
|
or a
|
|
jp nz,.vwait
|
|
|
|
UpdateObjectsCommon:
|
|
ld a,[displayType]
|
|
or a
|
|
ret nz ;done if cinema type
|
|
|
|
;check each object
|
|
ld b,METHOD_CHECK
|
|
ld a,[iterateNumObjects]
|
|
ld c,a
|
|
call IterateMaxObjects
|
|
|
|
;update the timers that hero/hero bullets base moves off of
|
|
call UpdateHeroTimers
|
|
|
|
;make sure we check the hero & hero's bullets each time
|
|
ld hl,hero0_index
|
|
call .checkHeroObjects
|
|
ld hl,hero1_index
|
|
call .checkHeroObjects
|
|
|
|
ret
|
|
|
|
.checkHeroObjects
|
|
ld a,[hl+] ;heroX_index
|
|
or a
|
|
ret z ;this hero doesn't exist
|
|
|
|
ld c,a
|
|
call GetFirst
|
|
ld b,METHOD_CHECK
|
|
;ld e,[hl] ;heroX_objectL
|
|
;inc hl
|
|
;ld d,[hl] ;heroX_objectH
|
|
;inc hl
|
|
call CallMethod
|
|
|
|
inc hl
|
|
inc hl
|
|
|
|
ld a,[hl] ;heroX_bulletIndex
|
|
ld c,a
|
|
call GetFirst
|
|
call IterateList
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: RedrawMap
|
|
; Description: Redraws the map for the next vblank
|
|
;---------------------------------------------------------------------
|
|
RedrawMap::
|
|
call RestrictCameraToBounds
|
|
call ScrollToCamera
|
|
|
|
call DrawMapToBackBuffer
|
|
call UpdateEnvEffect
|
|
|
|
ld a,1
|
|
ldio [backBufferReady],a ;signal we're ready for DMA
|
|
|
|
ret
|
|
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: HandleRemoteInput
|
|
; Arguments: a - remoteInput
|
|
; Returns: a - remoteInput
|
|
; zflag - set if must repeat
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
HandleRemoteInput::
|
|
bit 7,a ;test bit 7
|
|
ret z ;got the BCB
|
|
|
|
;handle the link command code we received.
|
|
call HandleLCC
|
|
|
|
;indicate we need to try exchanging our control bytes again
|
|
ld a,1
|
|
or a ;don't set zero flag
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: HandleLCC
|
|
; Arguments: a - value of the Link Command Code.
|
|
; Returns: nothing.
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
HandleLCC:
|
|
.checkLGETGAMESTATE
|
|
cp LGETGAMESTATE
|
|
jr nz,.checkLGETMAPINDEX
|
|
|
|
;LGETGAMESTATE
|
|
LONGCALLNOARGS TransmitGameState
|
|
ret
|
|
|
|
.checkLGETMAPINDEX
|
|
cp LGETMAPINDEX
|
|
jr nz,.checkLUPDATESTATE
|
|
|
|
;LGETMAPINDEX
|
|
ld a,[displayType]
|
|
or a
|
|
jr nz,.returnDifferentMapCode ;cinema type
|
|
ld a,[amChangingMap]
|
|
or a
|
|
jr nz,.returnWaitCode
|
|
ld a,[canJoinMap]
|
|
;or a
|
|
;jr z,.returnWaitCode
|
|
cp 2
|
|
jr z,.returnDifferentMapCode ;do this map independently
|
|
|
|
ld a,[curLevelStateIndex]
|
|
jr .returnMapIndex
|
|
.returnWaitCode
|
|
ld a,$ff
|
|
jr .returnMapIndex
|
|
.returnDifferentMapCode
|
|
ld a,$fe
|
|
.returnMapIndex
|
|
call TransmitByte
|
|
ret
|
|
|
|
.checkLUPDATESTATE
|
|
cp LUPDATESTATE
|
|
jr nz,.checkLUPDATEHERO
|
|
|
|
;LUPDATESTATE
|
|
push hl
|
|
ld a,LEVELSTATEBANK
|
|
ld [$ff70],a
|
|
call ReceiveByte
|
|
ld h,((levelState>>8) & $ff)
|
|
ld l,a
|
|
call ReceiveByte
|
|
ld [hl],a
|
|
pop hl
|
|
|
|
ret
|
|
|
|
.checkLUPDATEHERO
|
|
cp LUPDATEHERO
|
|
jr nz,.checkLYANKPLAYER
|
|
|
|
;LUPDATEHERO
|
|
call ReceiveByte
|
|
ld [heroesUsed],a
|
|
ret
|
|
|
|
.checkLYANKPLAYER
|
|
cp LYANKPLAYER
|
|
jr nz,.checkLRESYNCHRONIZE
|
|
|
|
;LYANKPLAYER
|
|
;get exit direction
|
|
call ReceiveByte
|
|
ld [hero0_enterLevelFacing],a
|
|
|
|
;get new map coordinates
|
|
push bc
|
|
push af
|
|
call ReceiveByte
|
|
ld c,a
|
|
call ReceiveByte
|
|
ld b,a
|
|
ld a,[hero0_enterLevelFacing]
|
|
and 7
|
|
ld [hero0_enterLevelFacing],a
|
|
ld [hero1_enterLevelFacing],a
|
|
pop af
|
|
cp 8
|
|
jr nc,.changeMap
|
|
|
|
ld a,[curLevelIndex]
|
|
cp c
|
|
jr nz,.changeMap
|
|
ld a,[curLevelIndex+1]
|
|
cp b
|
|
jr z,.noChangeMap
|
|
|
|
.changeMap
|
|
ld a,c
|
|
ld [curLevelIndex],a ;map index lobyte
|
|
ld a,b
|
|
ld [curLevelIndex+1],a ;map index lobyte
|
|
ld a,1
|
|
ld [timeToChangeLevel],a
|
|
|
|
.noChangeMap ;already on this map
|
|
pop bc
|
|
ret
|
|
|
|
.checkLRESYNCHRONIZE
|
|
cp LRESYNCHRONIZE
|
|
jr nz,.checkLGETRANDOMSEED
|
|
|
|
LONGCALLNOARGS GuestContinueSynchronization
|
|
ret
|
|
|
|
.checkLGETRANDOMSEED
|
|
cp LGETRANDOMSEED
|
|
jr nz,.checkLTERMINATE
|
|
|
|
ldio a,[randomLoc]
|
|
call TransmitByte
|
|
ret
|
|
|
|
.checkLTERMINATE
|
|
cp LTERMINATE
|
|
jr nz,.checkLSYNCHRONIZE
|
|
|
|
ld a,[amLinkMaster]
|
|
cp 1
|
|
jr z,.terminateMaster
|
|
|
|
;terminate slave
|
|
rst $00
|
|
.terminateMaster
|
|
ld a,$fe
|
|
ld [amLinkMaster],a
|
|
ret
|
|
|
|
.checkLSYNCHRONIZE
|
|
cp LSYNCHRONIZE
|
|
jr nz,.checkLLOCKHEROES
|
|
|
|
;LSYNCHRONIZE
|
|
;am I ready to synchronize?
|
|
ld a,[checkInputInMainLoop]
|
|
or a
|
|
jr z,.notReadyToSync
|
|
ld a,[amShowingDialog]
|
|
or a
|
|
jr nz,.readyIfCinema
|
|
ld a,[canJoinMap]
|
|
or a
|
|
jr z,.notReadyToSync
|
|
jr .readyToSynchronize
|
|
|
|
.readyIfCinema
|
|
ld a,[displayType]
|
|
or a
|
|
jr nz,.readyToSynchronize
|
|
|
|
;not ready
|
|
.notReadyToSync
|
|
ld a,LSYNCHWAIT
|
|
call TransmitByte
|
|
ret
|
|
|
|
.readyToSynchronize
|
|
ld a,LSYNCHREADY
|
|
call TransmitByte
|
|
LONGCALLNOARGS HostSynchronize
|
|
ret
|
|
|
|
.checkLLOCKHEROES
|
|
cp LLOCKHEROES
|
|
jr nz,.checkLNOLINK
|
|
|
|
call ReceiveByte ;receive lock or unlock from remote
|
|
ld [heroesLocked],a
|
|
ret
|
|
|
|
.checkLNOLINK
|
|
cp LNOLINK
|
|
jr nz,.checkLLINKTEST
|
|
ret
|
|
|
|
.checkLLINKTEST
|
|
cp LLINKTEST
|
|
jr nz,.checkLCHANGEAPPXMAP
|
|
|
|
LONGCALLNOARGS LinkTest
|
|
ret
|
|
|
|
.checkLCHANGEAPPXMAP
|
|
cp LCHANGEAPPXMAP
|
|
jr nz,.checkLADDINVITEM
|
|
|
|
LONGCALLNOARGS NewAppxLocation
|
|
ret
|
|
|
|
.checkLADDINVITEM
|
|
cp LADDINVITEM
|
|
jr nz,.checkLREMINVITEM
|
|
|
|
push bc
|
|
call ReceiveByte ;get bc (inventory item)
|
|
ld c,a
|
|
call ReceiveByte
|
|
ld b,a
|
|
|
|
;duplicate code also in map.asm
|
|
push hl
|
|
call PointHLToInventory
|
|
or c
|
|
ld [hl],a
|
|
pop hl
|
|
pop bc
|
|
ret
|
|
|
|
.checkLREMINVITEM
|
|
cp LREMINVITEM
|
|
jr nz,.checkLUPDATEMEMORY
|
|
|
|
;duplicate code also in map.asm
|
|
push bc
|
|
call ReceiveByte ;get bc (inventory item)
|
|
ld c,a
|
|
call ReceiveByte
|
|
ld b,a
|
|
|
|
;duplicate code also in map.asm
|
|
push hl
|
|
call PointHLToInventory
|
|
xor $ff
|
|
or c
|
|
xor $ff
|
|
ld [hl],a
|
|
pop hl
|
|
pop bc
|
|
ret
|
|
|
|
.checkLUPDATEMEMORY
|
|
cp LUPDATEMEMORY
|
|
jr nz,.unknown
|
|
|
|
push hl
|
|
call ReceiveByte ;memory bank of data
|
|
ldio [$ff70],a
|
|
call ReceiveByte ;address
|
|
ld l,a
|
|
call ReceiveByte
|
|
ld h,a
|
|
call ReceiveByte ;new value
|
|
ld [hl],a
|
|
pop hl
|
|
ret
|
|
|
|
.unknown
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routines: LLTransmitByte
|
|
; Arguments: a - byte to send
|
|
; [amLinkMaster] - should be set to 1 or 0
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
; Note: DON'T CALL THIS DIRECTLY. It should only be called
|
|
; via TransmitByte and ReceiveByte.
|
|
; Description: Low-level routine to transmit a byte via the link
|
|
; cable.
|
|
;---------------------------------------------------------------------
|
|
LLTransmitByte:
|
|
di ;be sure we're not interrupted
|
|
ldio [$ff01],a ;store the data to transfer
|
|
ld a,INTCLK
|
|
ldio [$ff02],a ;let's do it
|
|
.waitForCompletion
|
|
ldio a,[$ff02] ;12
|
|
and $80 ; 8
|
|
jr nz,.waitForCompletion ; 8
|
|
|
|
;go into slave mode to await next byte
|
|
ld a,EXTCLK ; 8
|
|
ldio [$ff02],a ;12 TOTAL = 48
|
|
|
|
;clear pending interrupts
|
|
xor a
|
|
ei
|
|
ld [$ff0f],a
|
|
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routines: LLReceiveByte
|
|
; Arguments: [amLinkMaster] - should be set to 1 or 0
|
|
; Returns: a - byte received from linked GameBoy.
|
|
; Alters: af
|
|
; Note: DON'T CALL THIS DIRECTLY. It should only be called
|
|
; via TransmitByte and ReceiveByte.
|
|
; Description: Low-level routine to receive a byte via the link
|
|
; cable. If no signal within nine seconds aborts link
|
|
; connection and:
|
|
; Master - continues game as slaveless master ($fe).
|
|
; Slave - restarts game.
|
|
;---------------------------------------------------------------------
|
|
LLReceiveByte:
|
|
push bc
|
|
push de
|
|
|
|
ld de,$0C00 ;abort timer counts down $C00 * 256 times
|
|
ld c,0 ;(about 1.25 sec per $100)
|
|
.waitForCompletion
|
|
ldio a,[$ff02]
|
|
and $80
|
|
jr z,.gotValue
|
|
|
|
;update our abort timer
|
|
dec c
|
|
jr nz,.waitForCompletion
|
|
|
|
dec de
|
|
xor a
|
|
cp d
|
|
jr nz,.waitForCompletion
|
|
cp e
|
|
jr nz,.waitForCompletion
|
|
jr .linkLost
|
|
|
|
.gotValue
|
|
ldio a,[$ff01] ;return the value we received
|
|
|
|
;kill some time before we let ourselves change to master
|
|
;to give the current master time to change to slave
|
|
push af
|
|
.killOuter
|
|
ld a,6
|
|
.killTime
|
|
dec a ; 4
|
|
jr nz,.killTime ; 4/8 TOTAL = 8, *6=48
|
|
pop af
|
|
jr .done
|
|
|
|
.linkLost
|
|
ld a,[amLinkMaster]
|
|
or a
|
|
jr z,.amSlave
|
|
|
|
cp 1
|
|
jr z,.amMaster
|
|
|
|
;link already lost, restore stack & jump to bail-out address
|
|
;if not 0
|
|
ld a,[linkBailOut] ;get bail-out address
|
|
ld e,a
|
|
ld a,[linkBailOut+1]
|
|
ld d,a
|
|
xor a
|
|
cp d
|
|
jr nz,.bail
|
|
cp e
|
|
jr nz,.bail
|
|
|
|
;null bail address; just quit
|
|
jr .done
|
|
|
|
.bail
|
|
;restore the stack before jumping to the bail address
|
|
ld a,[linkBailOut+2]
|
|
ld l,a
|
|
ld a,[linkBailOut+3]
|
|
ld h,a
|
|
ld sp,hl
|
|
push de
|
|
ret
|
|
|
|
.amMaster
|
|
ld a,$fe
|
|
ld [amLinkMaster],a
|
|
|
|
call RemoveRemoteHero
|
|
jr .done
|
|
|
|
.amSlave
|
|
rst $00 ;restart the game
|
|
|
|
.done
|
|
pop de
|
|
pop bc
|
|
ret
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: TransmitByte
|
|
; Argument: a - byte to send
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
; Description: Transmits a byte via the link cable to a remote GBC.
|
|
;---------------------------------------------------------------------
|
|
TransmitByte::
|
|
push bc
|
|
ld b,a
|
|
|
|
;If I was the last one to transmit, receive a byte as an
|
|
;acknowledgement.
|
|
ld a,[lastLinkAction]
|
|
or a
|
|
jr z,.afterOrientStream
|
|
|
|
call LLReceiveByte
|
|
|
|
.afterOrientStream
|
|
ld a,1 ;our action this time is to transmit
|
|
ld [lastLinkAction],a
|
|
|
|
ld a,b
|
|
call LLTransmitByte
|
|
|
|
.done
|
|
pop bc
|
|
ret
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: ReceiveByte
|
|
; Arguments: None.
|
|
; Returns: a - byte received.
|
|
; Alters: af
|
|
; Description: Receives a byte from a remote machine.
|
|
;---------------------------------------------------------------------
|
|
ReceiveByte::
|
|
;If necessary we'll transmit a value to avoid receiving twice
|
|
;in a row.
|
|
ld a,[lastLinkAction]
|
|
or a
|
|
jr nz,.afterOrientStream
|
|
|
|
ld a,1
|
|
call LLTransmitByte
|
|
|
|
.afterOrientStream
|
|
xor a ;our action this time is to receive
|
|
ld [lastLinkAction],a
|
|
|
|
call LLReceiveByte
|
|
ret
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: ExchangeByte
|
|
; Argument: a - byte to send
|
|
; Returns: a - byte received in return
|
|
; Alters: af
|
|
; Description: Uses value of [amLinkMaster] (Master should be $01 and
|
|
; Slave should be $00) to determine sequencing of
|
|
; TransmitByte and ReceiveByte.
|
|
;---------------------------------------------------------------------
|
|
ExchangeByte::
|
|
push bc
|
|
ld b,a
|
|
|
|
ld a,[amLinkMaster]
|
|
or a
|
|
jr z,.slaveExchange
|
|
|
|
;masterExchange
|
|
ld a,b
|
|
call TransmitByte
|
|
call ReceiveByte
|
|
jr .done
|
|
|
|
.slaveExchange
|
|
call ReceiveByte
|
|
ld c,a
|
|
ld a,b
|
|
call TransmitByte
|
|
ld a,c
|
|
|
|
.done
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: TransmitData
|
|
; Arguments: a - RAM bank to switch to (may/may not make a diff)
|
|
; bc - number of bytes to transmit
|
|
; hl - starting address of data to transmit
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
TransmitData::
|
|
push bc
|
|
push hl
|
|
|
|
ldio [$ff70],a
|
|
|
|
.loop ld a,[hl+]
|
|
call TransmitByte
|
|
dec bc
|
|
xor a
|
|
cp b
|
|
jr nz,.loop
|
|
cp c
|
|
jr nz,.loop
|
|
|
|
pop hl
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: ReceiveData
|
|
; Arguments: a - RAM bank to switch to (may/may not make a diff)
|
|
; bc - number of bytes to receive
|
|
; hl - starting address to store data
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
ReceiveData::
|
|
push bc
|
|
push hl
|
|
|
|
ldio [$ff70],a ;change RAM bank
|
|
|
|
.loop call ReceiveByte
|
|
ld [hl+],a
|
|
dec bc
|
|
xor a
|
|
cp b
|
|
jr nz,.loop
|
|
cp c
|
|
jr nz,.loop
|
|
|
|
pop hl
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: TransmitCompressedData
|
|
; Arguments: a - RAM bank to switch to (may/may not make a diff)
|
|
; bc - number of bytes to transmit
|
|
; hl - starting address of data to transmit
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
; Description: Packs bytes 8:1 into the backBuffer by combining
|
|
; bit[0] from every eight bytes into one byte. The
|
|
; left-most bit will be from byte[0] and the right-most
|
|
; from byte[7].
|
|
;---------------------------------------------------------------------
|
|
TransmitCompressedData::
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
ldio [$ff70],a
|
|
|
|
;divide len by 8 (>>3)
|
|
srl b
|
|
rr c
|
|
srl b
|
|
rr c
|
|
srl b
|
|
rr c
|
|
|
|
push bc ;save length
|
|
|
|
;----Compress the data into backBuffer------------------------
|
|
ld de,backBuffer ;compression buffer
|
|
.nextByte
|
|
push de ;push destination
|
|
ld de,8 ;d is compressed (0), e is times to loop
|
|
|
|
.nextBit
|
|
ld a,[hl+] ;get the next byte
|
|
rrca ;shift out bit[0] into the carry flag
|
|
rl d ;rotate onto right side of result
|
|
dec e
|
|
jr nz,.nextBit
|
|
|
|
ld a,d ;a has compressed byte
|
|
pop de ;pop destination
|
|
ld [de],a ;store the compressed byte
|
|
inc de
|
|
|
|
dec bc
|
|
xor a
|
|
cp b
|
|
jr nz,.nextByte
|
|
cp c
|
|
jr nz,.nextByte
|
|
|
|
pop bc ;done with compression, retrieve len
|
|
|
|
ld hl,backBuffer ;transmit the compressed data
|
|
xor a
|
|
call TransmitData
|
|
|
|
pop hl
|
|
pop de
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: ReceiveCompressedData
|
|
; Arguments: a - RAM bank to switch to (may/may not make a diff)
|
|
; bc - number of bytes to receive
|
|
; hl - starting address of data to receive
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
; Description: Receives numBytes/8 into the backBuffer, then unpacks
|
|
; each byte 1:8 to the data address.
|
|
;---------------------------------------------------------------------
|
|
ReceiveCompressedData::
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
push af ;save original RAM bank
|
|
push hl ;save original dest
|
|
|
|
;----Calculate numBytes/8 (>>3)-------------------------------
|
|
srl b
|
|
rr c
|
|
srl b
|
|
rr c
|
|
srl b
|
|
rr c
|
|
|
|
ld hl,backBuffer ;receive the compressed data
|
|
xor a
|
|
call ReceiveData
|
|
|
|
pop hl ;retrieve original dest
|
|
pop af ;retrieve original RAM bank
|
|
ldio [$ff70],a
|
|
|
|
;----Decompress the data into the destination buffer----------
|
|
ld de,backBuffer ;decompression buffer
|
|
.nextByte
|
|
ld a,[de] ;get a byte from the buffer
|
|
inc de
|
|
push de ;push source
|
|
ld d,a ;store compressed byte in d
|
|
ld e,8 ;e is times to loop
|
|
|
|
.nextBit
|
|
xor a
|
|
rl d ;get a bit off left side of compressed
|
|
rla ;shift onto right side of uncompressed
|
|
ld [hl+],a ;store in final destination
|
|
dec e
|
|
jr nz,.nextBit
|
|
|
|
pop de ;pop source
|
|
|
|
dec bc
|
|
xor a
|
|
cp b
|
|
jr nz,.nextByte
|
|
cp c
|
|
jr nz,.nextByte
|
|
|
|
pop hl
|
|
pop de
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: CheckSimultaneousLCC
|
|
; Argument: a - byte received from link exchange
|
|
; Returns: not z flag - this machine must repeat its command
|
|
; z flag set - everything is okay
|
|
; Alters: af
|
|
; Description: Called after one machine has sent the first byte of
|
|
; a Link Command Code. If the remote machine
|
|
; coincidentally sent its own LCC at the same time then
|
|
; the Slave must delay its own LCC until the Master's
|
|
; LCC has been handled.
|
|
;---------------------------------------------------------------------
|
|
CheckSimultaneousLCC::
|
|
bit 7,a ;see if we got a LCC or a BCB.
|
|
ret z ;Got a BCB, everything's cool.
|
|
|
|
;Got a LCC. If I'm the Master then the other machine must
|
|
;yield to me so everything's still cool.
|
|
push af
|
|
ld a,[amLinkMaster]
|
|
or a
|
|
jr z,.amSlave
|
|
pop af
|
|
xor a
|
|
ret
|
|
|
|
.amSlave
|
|
;I must yield to the Master. Handle his request and then
|
|
;return a value indicating I must repeat my own request.
|
|
pop af
|
|
call HandleLCC
|
|
ld a,1
|
|
or a
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: YankRemotePlayer
|
|
; Arguments: a - exit to set to (>7 means &7 and don't check
|
|
; to see if you're on the map)
|
|
;
|
|
; hl - map to set to in BCD (e.g. $0205)
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
; Description: Interrupts whatever the other player is doing and
|
|
; sets him to come to the specified map. Also sets
|
|
; the join/respawn map to the map to set to.
|
|
;---------------------------------------------------------------------
|
|
YankRemotePlayer::
|
|
push af
|
|
call UpdateState
|
|
call SetJoinMap
|
|
call SetRespawnMap
|
|
pop af
|
|
|
|
;verify link established
|
|
push hl
|
|
ld hl,amLinkMaster
|
|
bit 7,[hl]
|
|
pop hl
|
|
ret nz
|
|
|
|
push af
|
|
push hl
|
|
ld a,2 ;two additional things on stack
|
|
ld hl,.linkBailAddress
|
|
call SetLinkBailOutAddress
|
|
|
|
pop hl
|
|
|
|
.sendYank
|
|
ld a,LYANKPLAYER
|
|
call ExchangeByte
|
|
call CheckSimultaneousLCC
|
|
jr nz,.sendYank
|
|
pop af
|
|
|
|
;remote machine is listening; send map coords
|
|
call TransmitByte ;exit dir
|
|
ld a,l
|
|
call TransmitByte
|
|
ld a,h
|
|
call TransmitByte
|
|
.linkBailAddress
|
|
push hl
|
|
ld hl,0
|
|
xor a
|
|
call SetLinkBailOutAddress
|
|
pop hl
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: RemoveRemoteHero
|
|
; Arguments: None.
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
; Description: Deletes the remote hero and removes traces of its
|
|
; presence.
|
|
;---------------------------------------------------------------------
|
|
RemoveRemoteHero::
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
;determine which is the remote hero
|
|
ld a,[amLinkMaster]
|
|
or a
|
|
jr z,.amSlave
|
|
|
|
.amMaster
|
|
ld hl,hero1_index
|
|
jr .determinedRemoteHero
|
|
|
|
.amSlave
|
|
ld hl,hero0_index
|
|
|
|
.determinedRemoteHero
|
|
;remove remote hero from map if present
|
|
ld a,[hl] ;heroX_index
|
|
or a
|
|
jr z,.done ;not here
|
|
|
|
;save the current ROM & switch in the object ROM
|
|
ldio a,[curROMBank]
|
|
push af
|
|
ld a,CLASSROM
|
|
call SetActiveROM
|
|
|
|
;----remove this one's heroesPresent flag----
|
|
ld a,[hl] ;heroX_index
|
|
push af
|
|
|
|
ld a,l
|
|
and 16
|
|
IF HERODATASIZE!=16
|
|
fix this
|
|
ENDC
|
|
swap a ;a is now 1 or 0
|
|
add 1 ;now 2 or 1
|
|
xor $ff ;now ~2 or ~1
|
|
ld b,a
|
|
ld a,[heroesPresent]
|
|
and b
|
|
ld [heroesPresent],a
|
|
|
|
;push hl
|
|
;ld de,(HERODATA_TYPE - HERODATA_INDEX)
|
|
;add hl,de
|
|
;ld a,[hl] ;heroX_type
|
|
;xor $ff
|
|
;ld b,a
|
|
;ld a,[heroesPresent]
|
|
;and b
|
|
;ld [heroesPresent],a
|
|
;pop hl
|
|
xor a
|
|
ld [hl],a ;heroX_index
|
|
pop af
|
|
|
|
ld c,a
|
|
call GetFirst
|
|
ld b,METHOD_DIE
|
|
call CallMethod
|
|
|
|
;restore the old ROM
|
|
pop af
|
|
call SetActiveROM
|
|
|
|
.done
|
|
pop hl
|
|
pop de
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: DebugMesg
|
|
; Argument: hl - pointer to a gtx structure
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
; Description: Displays the given gtx string in a window for a
|
|
; brief period.
|
|
;---------------------------------------------------------------------
|
|
DebugMesg::
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
ldio a,[$ff40]
|
|
and %10000000
|
|
jr z,.done
|
|
|
|
ld d,h
|
|
ld e,l
|
|
xor a
|
|
|
|
;-------Show Dialog At Top-------------------
|
|
push af
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
ld b,0 ;lines to skip at top
|
|
ld a,[curROMBank]
|
|
call ShowDialogCommon
|
|
|
|
xor a
|
|
ldh [$ff4a], a ;set window y position
|
|
|
|
ld a,[de] ;number of lines
|
|
rlca ;times 8 = pixels for window
|
|
rlca
|
|
rlca
|
|
add 7
|
|
ld [hblankWinOff],a
|
|
|
|
ld a,143
|
|
ld [hblankWinOn],a
|
|
|
|
ld a,[hblankFlag]
|
|
bit 1,a
|
|
jr nz,.afterSetLYC
|
|
|
|
ld a,143
|
|
ld [$ff45],a ;lyc
|
|
ld a,[hblankFlag]
|
|
|
|
.afterSetLYC
|
|
or %10 ;allow window to show
|
|
ld [hblankFlag],a
|
|
|
|
pop hl
|
|
pop de
|
|
pop bc
|
|
pop af
|
|
;-------Show Dialog At Top-------------------
|
|
|
|
ld bc,16384
|
|
call TimerDelay
|
|
call ClearDialog
|
|
|
|
.done
|
|
pop hl
|
|
pop de
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: DebugVal
|
|
; Argument: a - byte to show as a hex value
|
|
; Returns: Nothing.
|
|
; Alters: Nothing.
|
|
; Description: Displays the given value in hex using a Dialog
|
|
; window for a brief period. Game must be on.
|
|
;---------------------------------------------------------------------
|
|
DebugVal::
|
|
push af
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
ld b,a ;save byte
|
|
|
|
ldio a,[$ff40]
|
|
and %10000000
|
|
jr z,.done
|
|
|
|
ld hl,$c200
|
|
ld [hl],1
|
|
inc hl
|
|
ld [hl],1
|
|
inc hl
|
|
ld [hl],2
|
|
inc hl
|
|
ld a,b
|
|
ld c,0
|
|
swap a
|
|
and $0f
|
|
add 200
|
|
ld [hl+],a
|
|
ld a,b
|
|
and $0f
|
|
add 200
|
|
ld [hl+],a
|
|
ld de,$c200
|
|
xor a
|
|
|
|
;-------Show Dialog At Top-------------------
|
|
push af
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
ld b,0 ;lines to skip at top
|
|
ld a,[curROMBank]
|
|
call ShowDialogCommon
|
|
|
|
xor a
|
|
ldh [$ff4a], a ;set window y position
|
|
|
|
ld a,[de] ;number of lines
|
|
rlca ;times 8 = pixels for window
|
|
rlca
|
|
rlca
|
|
add 7
|
|
ld [hblankWinOff],a
|
|
|
|
ld a,143
|
|
ld [hblankWinOn],a
|
|
|
|
ld a,[hblankFlag]
|
|
bit 1,a
|
|
jr nz,.afterSetLYC
|
|
|
|
ld a,143
|
|
ld [$ff45],a ;lyc
|
|
ld a,[hblankFlag]
|
|
|
|
.afterSetLYC
|
|
or %10 ;allow window to show
|
|
ld [hblankFlag],a
|
|
|
|
pop hl
|
|
pop de
|
|
pop bc
|
|
pop af
|
|
;-------Show Dialog At Top-------------------
|
|
|
|
ld bc,8192 ;16384
|
|
call TimerDelay
|
|
call ClearDialog
|
|
|
|
.done
|
|
pop hl
|
|
pop de
|
|
pop bc
|
|
pop af
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: TimerDelay
|
|
; Argument: bc - number of 16384's of a second to delay
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
TimerDelay::
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
ld hl,$ff04
|
|
ld d,0 ;prev value of time
|
|
ld [hl],0
|
|
.wait
|
|
ld a,[hl]
|
|
ld e,a
|
|
xor d ;equal to prev value?
|
|
jr z,.wait
|
|
|
|
ld d,e
|
|
dec bc
|
|
xor a
|
|
cp b
|
|
jr nz,.wait
|
|
cp c
|
|
jr nz,.wait
|
|
|
|
pop hl
|
|
pop de
|
|
pop bc
|
|
ret
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: SetDialogSkip
|
|
; SetDialogForward
|
|
; Arguments: de - position to skip or fast forward to
|
|
; Alters: af
|
|
; Description: Saves the stack pointer (minus the space that this
|
|
; function call is taking up) and sets the go-to
|
|
; address for fast forwarding (any button pressed)
|
|
; or skipping (start button pressed).
|
|
;---------------------------------------------------------------------
|
|
SetDialogSkip::
|
|
push hl
|
|
ld hl,sp+4 ;stack ptr pos once returned from this fn
|
|
ld a,e
|
|
ld [levelCheckSkip+2],a ;set skip forward addr
|
|
ld a,d
|
|
ld [levelCheckSkip+3],a
|
|
jr SetDialogCommon
|
|
|
|
SetDialogForward::
|
|
push hl
|
|
ld hl,sp+4 ;stack ptr pos once returned from this fn
|
|
ld a,e ;set fast forward addr
|
|
ld [levelCheckSkip],a
|
|
ld a,d
|
|
ld [levelCheckSkip+1],a
|
|
jr SetDialogCommon
|
|
|
|
SetSkipStackPos::
|
|
push hl
|
|
ld hl,sp+4 ;stack ptr pos once returned from this fn
|
|
|
|
SetDialogCommon:
|
|
ld a,l
|
|
ld [levelCheckStackPos],a
|
|
ld a,h
|
|
ld [levelCheckStackPos+1],a
|
|
pop hl
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: ClearDialogSkipForward
|
|
; Arguments: None.
|
|
; Alters: af,hl
|
|
;---------------------------------------------------------------------
|
|
ClearDialogSkipForward::
|
|
call ClearDialog
|
|
call ClearSkipForward
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: ClearSkipForward
|
|
; Arguments: None.
|
|
; Alters: af,hl
|
|
;---------------------------------------------------------------------
|
|
ClearSkipForward::
|
|
ld de,0
|
|
call SetDialogSkip
|
|
call SetDialogForward
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: SetRespawnMap
|
|
; Arguments: hl - map to go to after hero dies
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
SetRespawnMap::
|
|
ld a,l
|
|
ld [respawnMap],a
|
|
ld a,h
|
|
ld [respawnMap+1],a
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: SetJoinMap
|
|
; Arguments: hl - map to join from 2nd player title screen
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
SetJoinMap::
|
|
ld a,l
|
|
ld [joinMap],a
|
|
ld a,h
|
|
ld [joinMap+1],a
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: SetLinkBailOutAddress
|
|
; Arguments: a - number of additional words to add to stack
|
|
; pointer
|
|
; hl - saves stack pointer and bailout address into
|
|
; [linkBailOut0...3].
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
SetLinkBailOutAddress::
|
|
push hl
|
|
push af
|
|
;save bail out address
|
|
ld a,l
|
|
ld [linkBailOut],a
|
|
ld a,h
|
|
ld [linkBailOut+1],a
|
|
|
|
;save stack pointer (plus a*2 + 6 for things pushed on stack)
|
|
ld hl,sp+6
|
|
pop af
|
|
or a
|
|
jr z,.afterAdjustStackPointer
|
|
|
|
.adjustStackPointer
|
|
inc hl
|
|
inc hl
|
|
dec a
|
|
jr nz,.adjustStackPointer
|
|
|
|
.afterAdjustStackPointer
|
|
ld a,l
|
|
ld [linkBailOut+2],a
|
|
ld a,h
|
|
ld [linkBailOut+3],a
|
|
|
|
pop hl
|
|
ret
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: RemoveSpriteObjectsFromMap
|
|
;---------------------------------------------------------------------
|
|
RemoveSpriteObjectsFromMap:
|
|
;----remove all objects from the map--------------------------
|
|
ld a,OBJLISTBANK
|
|
ldio [$ff70],a
|
|
ld hl,objExists+1
|
|
.removeObjectsLoop
|
|
push hl
|
|
ld a,[hl]
|
|
or a
|
|
jr z,.removeObjects_continue
|
|
ld a,l
|
|
call IndexToPointerDE
|
|
ld a,[numClasses]
|
|
ld b,a
|
|
inc b
|
|
ld h,((objClassLookup>>8) & $ff)
|
|
ld a,[hl]
|
|
cp b ;this obj a regular creature?
|
|
jr nc,.removeObjects_continue
|
|
|
|
call GetFacing
|
|
bit 7,a ;sprite?
|
|
jr z,.restoreObjListBank ;no
|
|
ld c,a
|
|
call RemoveFromMap
|
|
.restoreObjListBank
|
|
ld a,OBJLISTBANK
|
|
ldio [$ff70],a
|
|
.removeObjects_continue
|
|
pop hl
|
|
inc hl
|
|
ld a,h
|
|
cp ((objExists>>8) & $ff) + 1
|
|
jr nz,.removeObjectsLoop
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: GetMethodAddrFromPointer
|
|
; Arguments: hl - pointer to method address
|
|
; Returns: hl - pointer to method
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
GetMethodAddrFromPointer::
|
|
PUSHROM
|
|
ld a,BANK(classTable)
|
|
call SetActiveROM
|
|
|
|
ld a,[hl+]
|
|
ld h,[hl]
|
|
ld l,a
|
|
|
|
POPROM
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: InstallHBlankHandler
|
|
; Arguments: hl - pointer to hblank handler
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
InstallHBlankHandler::
|
|
di
|
|
ld a,l
|
|
ld [hblankVector+1],a ;+0 is opcode "jp"
|
|
ld a,h
|
|
ld [hblankVector+2],a
|
|
ei
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: LockRemoteHeroesUsed
|
|
; Arguments: None.
|
|
; Returns: z - lock attempt unsuccessful
|
|
; nz - success
|
|
; Alters: af
|
|
; Description: Informs the remote machine that this machine is
|
|
; messing with the heroes to prevent collisions
|
|
;---------------------------------------------------------------------
|
|
LockRemoteHeroesUsed::
|
|
ld a,[amLinkMaster]
|
|
bit 7,a
|
|
jr nz,.lockOkay
|
|
|
|
.sendLock
|
|
ld a,[heroesLocked]
|
|
or a
|
|
jr nz,.returnFailure
|
|
ld a,LLOCKHEROES
|
|
call ExchangeByte
|
|
call CheckSimultaneousLCC
|
|
jr nz,.sendLock
|
|
|
|
ld a,1 ;send "locked" to remote
|
|
call TransmitByte
|
|
|
|
.lockOkay
|
|
ld a,1
|
|
ld [heroesLocked],a
|
|
or a
|
|
ret
|
|
|
|
.returnFailure
|
|
xor a
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: UpdateRemoteHeroesUsed
|
|
; Arguments: None.
|
|
; Returns: Nothing.
|
|
; Alters: af
|
|
; Description: Sends current [heroesUsed] to the remote machine
|
|
; if present. Also unlocks heroesUsed if locked.
|
|
;---------------------------------------------------------------------
|
|
UpdateRemoteHeroesUsed::
|
|
ld a,[amLinkMaster]
|
|
bit 7,a
|
|
jr z,.updateHeroesUsed
|
|
|
|
;no link, free lock
|
|
xor a
|
|
ld [heroesLocked],a
|
|
ret
|
|
|
|
.updateHeroesUsed
|
|
ld a,LUPDATEHERO
|
|
call ExchangeByte
|
|
call CheckSimultaneousLCC
|
|
jr nz,.updateHeroesUsed ;must repeat
|
|
|
|
ld a,[heroesUsed]
|
|
call TransmitByte
|
|
|
|
ld a,[heroesLocked]
|
|
or a
|
|
ret z ;not locked don't bother
|
|
|
|
.sendLock
|
|
ld a,LLOCKHEROES
|
|
call ExchangeByte
|
|
call CheckSimultaneousLCC
|
|
jr nz,.sendLock
|
|
|
|
xor a ;send "unlocked" to remote
|
|
ld [heroesLocked],a
|
|
call TransmitByte
|
|
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: PlaySample
|
|
; Arguments: a - bank of sample
|
|
; hl - start address of sample
|
|
; Alters: af,hl
|
|
; Description:
|
|
;---------------------------------------------------------------------
|
|
PlaySample::
|
|
;start sample playing
|
|
di
|
|
ldio [sampleBank],a
|
|
xor a
|
|
ldio [samplePlaying],a
|
|
ei
|
|
|
|
ld a,l
|
|
ldio [sampleAddress],a
|
|
ld a,h
|
|
ldio [sampleAddress+1],a
|
|
ld a,$0f
|
|
ldio [sampleMask],a
|
|
ld a,$86
|
|
ldio [$ff1a],a ;sound 3 enable
|
|
ld hl,$ff26
|
|
set 2,[hl]
|
|
ld a,1
|
|
ldio [samplePlaying],a
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; FinishHBlankPlayingSample
|
|
;---------------------------------------------------------------------
|
|
FinishHBlankPlayingSample::
|
|
ldio a,[samplePlaying]
|
|
or a
|
|
jr z,.done
|
|
|
|
.nextSample
|
|
ld hl,sampleAddress
|
|
ld a,[hl+]
|
|
ld h,[hl]
|
|
ld l,a
|
|
.loadNextSample
|
|
ldio a,[sampleBank]
|
|
ld [$2100],a
|
|
ldio a,[sampleMask]
|
|
xor $ff
|
|
ldio [sampleMask],a
|
|
cp $0f
|
|
jr nz,.getLeftNibble
|
|
|
|
.getRightNibble
|
|
and a,[hl]
|
|
inc hl
|
|
jr .normalData ;will never be special code
|
|
|
|
.getLeftNibble
|
|
and a,[hl] ;next sample byte
|
|
|
|
.checkForSpecialCodes
|
|
cp $80
|
|
jr nz,.normalData
|
|
|
|
;code indicating termination or link follows
|
|
inc hl
|
|
ld a,[hl+]
|
|
or a
|
|
jr nz,.link
|
|
|
|
ld [samplePlaying],a
|
|
ld a,$ff
|
|
ld [$ff24],a ;full volume both channels
|
|
jr .restoreBank
|
|
|
|
.link ld a,[hl+]
|
|
ld [sampleBank],a
|
|
ld a,[hl+]
|
|
ld h,[hl]
|
|
ld l,a
|
|
jr .loadNextSample
|
|
|
|
.normalData
|
|
ld b,a
|
|
swap a
|
|
or b
|
|
or $88
|
|
ldio [$ff24],a ;sound enabled/volume
|
|
|
|
.saveNewAddress
|
|
ld a,l
|
|
ldio [sampleAddress],a
|
|
ld a,h
|
|
ldio [sampleAddress+1],a
|
|
|
|
.restoreBank
|
|
ldio a,[curROMBank]
|
|
ld [$2100],a
|
|
|
|
.done
|
|
pop hl
|
|
pop bc
|
|
pop af
|
|
reti
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: Lookup8
|
|
; Arguments: a - index
|
|
; hl - start lookup table
|
|
; Alters: af,hl
|
|
; Returns: a - table[a]
|
|
;---------------------------------------------------------------------
|
|
Lookup8::
|
|
add l
|
|
ld l,a
|
|
ld a,h
|
|
adc 0
|
|
ld h,a
|
|
ld a,[hl]
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: Lookup16
|
|
; Arguments: a - index
|
|
; hl - start lookup table
|
|
; Alters: af,hl
|
|
; Returns: hl - table[a*2]
|
|
;---------------------------------------------------------------------
|
|
Lookup16::
|
|
push de
|
|
ld e,a
|
|
ld d,0
|
|
sla e
|
|
rl d
|
|
add hl,de
|
|
ld a,[hl+]
|
|
ld h,[hl]
|
|
ld l,a
|
|
pop de
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: LookupIndexOfData8
|
|
; Arguments: a - data
|
|
; hl - start of lookup table
|
|
; Alters: af,hl
|
|
; Returns: a - where table[a] is original data a.
|
|
; Data must exist!
|
|
;---------------------------------------------------------------------
|
|
LookupIndexOfData8::
|
|
push bc
|
|
|
|
ld c,0
|
|
.findLoop
|
|
cp [hl]
|
|
jr z,.done
|
|
|
|
inc c
|
|
inc hl
|
|
jr .findLoop
|
|
|
|
.done
|
|
ld a,c
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: FindFirstBGLoc
|
|
; Arguments: hl - bg class to find (e.g. classClearanceBG)
|
|
; Returns: hl - first location of class
|
|
; a - 0 on failure
|
|
; zflag - or a
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
FindFirstBGLoc::
|
|
push bc
|
|
push de
|
|
|
|
push hl
|
|
pop bc
|
|
call FindClassIndex
|
|
ld b,a
|
|
|
|
ld a,MAPBANK
|
|
ld [$ff70],a
|
|
|
|
;setup de with first out-of-bounds index
|
|
ld a,[mapTotalSize]
|
|
ld e,a
|
|
ld a,[mapTotalSize+1]
|
|
ld d,a
|
|
|
|
ld hl,map
|
|
.lookForFirst
|
|
ld a,[hl+]
|
|
cp b
|
|
jr z,.foundIt
|
|
|
|
;hl < de?
|
|
ld a,h
|
|
cp d
|
|
jr c,.lookForFirst
|
|
|
|
ld a,l
|
|
cp e
|
|
jr c,.lookForFirst
|
|
|
|
;not found
|
|
xor a
|
|
jr .done
|
|
|
|
.foundIt
|
|
dec hl
|
|
ld a,1
|
|
or a
|
|
|
|
.done
|
|
pop de
|
|
pop bc
|
|
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: LinkTransmitMemoryLocation
|
|
; Arguments: a - RAM bank
|
|
; hl - memory location to transmit to the remote
|
|
; GameBoy
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
LinkTransmitMemoryLocation::
|
|
push bc
|
|
ld b,a
|
|
|
|
ld a,[amLinkMaster]
|
|
bit 7,a
|
|
jr nz,.done ;not linked
|
|
|
|
.requestTransmit
|
|
ld a,LUPDATEMEMORY
|
|
call ExchangeByte
|
|
call CheckSimultaneousLCC
|
|
jr nz,.requestTransmit
|
|
|
|
ld a,b
|
|
ldio [$ff70],a
|
|
call TransmitByte
|
|
ld a,l
|
|
call TransmitByte
|
|
ld a,h
|
|
call TransmitByte
|
|
ld a,[hl]
|
|
call TransmitByte
|
|
|
|
.done
|
|
pop bc
|
|
ret
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Link Code
|
|
;---------------------------------------------------------------------
|
|
SECTION "LinkCodeSection",ROMX
|
|
linkCode:
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: HandleInput
|
|
; Description: Polls the buttons and stores the input in [curInput0].
|
|
; Then performs some network code and sets up [curJoy0],
|
|
; [curJoy1], [curInput0], and [curInput1]. curJoyX may
|
|
; be set to zero elsewhere to prevent the heroes from
|
|
; moving but curInput0 and curInput1 will always be the
|
|
; original values.
|
|
;
|
|
; $80 - 7:Start
|
|
; $40 - 6:Select
|
|
; $20 - 5:B
|
|
; $10 - 4:A
|
|
; $08 - 3:Down
|
|
; $04 - 2:Up
|
|
; $02 - 1:Left
|
|
; $01 - 0:Right
|
|
;
|
|
; A linked Master should read its input from [curJoy0].
|
|
; A linked Slave should read its input from [curJoy1].
|
|
; To make this easier [myJoy] is set up to be the
|
|
; appropriate one.
|
|
;
|
|
; If no link connection established, attempt a link
|
|
; connection.
|
|
;
|
|
; If no link:
|
|
; Convert curInput0->Joy1 (result will be same as
|
|
; encoding then decoding curInput).
|
|
;
|
|
; If link connection established:
|
|
; Exchange Control Byte with linked machine. If
|
|
; Command Code gotten in return, perform requested
|
|
; action and then repeat attempt to exchange control
|
|
; byte.
|
|
;
|
|
; If Master: curInput0->Joy1, remoteInput->Joy2
|
|
; If Slave: curInput0->Joy2, remoteInput->Joy1
|
|
;---------------------------------------------------------------------
|
|
HandleInput:
|
|
ld a,$20
|
|
ldio [$ff00],a ;select P14
|
|
ldio a,[$ff00]
|
|
ldio a,[$ff00] ;wait a few cycles
|
|
cpl
|
|
and $0f
|
|
ld b,a ;b has direction info
|
|
ld a,$10 ;select P15
|
|
ld [$ff00],a
|
|
ldio a,[$ff00] ;wait mo
|
|
ldio a,[$ff00] ;wait mo
|
|
ldio a,[$ff00] ;wait mo
|
|
ldio a,[$ff00] ;wait mo
|
|
ldio a,[$ff00] ;wait mo
|
|
ldio a,[$ff00] ;wait mo
|
|
cpl
|
|
and $0f
|
|
swap a
|
|
or b ;a has all buttons
|
|
ld [curInput0],a
|
|
ld a,$30 ;deselect P14 and P15
|
|
ldio [$ff00],a
|
|
|
|
;--if no link, attempt link connection ----------------------
|
|
ld a,[amLinkMaster]
|
|
cp $ff
|
|
jr z,.afterLinkAttempt ;l1101 attempts link as slave
|
|
cp $fe ;attempt link as master?
|
|
jr nz,.afterLinkAttempt
|
|
|
|
.attemptLinkAsMaster
|
|
ld a,$55
|
|
ldio [$ff01],a ;exchange data = $55
|
|
ld a,INTCLK ;ready to xchg, use my clock
|
|
ldio [$ff02],a
|
|
|
|
.waitMasterLinkAttempt
|
|
ldio a,[$ff02]
|
|
and $80
|
|
jr nz,.waitMasterLinkAttempt
|
|
|
|
;see what we got
|
|
ldio a,[$ff01]
|
|
cp $aa
|
|
jr nz,.afterLinkAttempt
|
|
|
|
;found a Slave! That means I'm the Master!
|
|
ld a,EXTCLK ;switch to receive mode
|
|
ldio [$ff02],a
|
|
ld a,1
|
|
ld [amLinkMaster],a
|
|
ld [lastLinkAction],a
|
|
|
|
.afterLinkAttempt
|
|
ld a,[amLinkMaster]
|
|
bit 7,a
|
|
jr z,.amLinked
|
|
|
|
;no link, convert curInput->curJoy0.
|
|
ld a,[curInput0]
|
|
call ConvertInput
|
|
ld [curInput0],a
|
|
ld [curJoy0],a
|
|
ld [myJoy],a
|
|
xor a
|
|
ld [curJoy1],a
|
|
ld [curInput1],a
|
|
ret
|
|
|
|
.amLinked
|
|
|
|
IF 0
|
|
;----every second or so, get the remote machine's
|
|
;----random seed and make sure it matches ours.
|
|
;----Resynchronize the machines if not
|
|
ld a,[amLinkMaster]
|
|
cp 1
|
|
jr nz,.exchangeBCB ;only master can initiate
|
|
|
|
ld a,[updateTimer] ;not time yet
|
|
and 31
|
|
cp 31
|
|
jr nz,.exchangeBCB
|
|
|
|
;don't do it if can't join map
|
|
ld a,[canJoinMap]
|
|
or a
|
|
jr z,.exchangeBCB
|
|
|
|
.getRandomSeed
|
|
ld a,LGETRANDOMSEED ;send the request
|
|
call ExchangeByte
|
|
call CheckSimultaneousLCC
|
|
jr nz,.getRandomSeed ;must repeat
|
|
|
|
;get the actual seed
|
|
call ReceiveByte
|
|
ld b,a
|
|
ld a,[randomLoc]
|
|
cp b ;got same in return?
|
|
jr z,.exchangeBCB ;continue if yes
|
|
|
|
;on same map?
|
|
ld a,[heroesPresent]
|
|
cp %11
|
|
jr nz,.exchangeBCB ;not on same map
|
|
|
|
.resynchronize
|
|
;resynchronize machines
|
|
ld a,LRESYNCHRONIZE
|
|
call ExchangeByte
|
|
call CheckSimultaneousLCC
|
|
jr nz,.resynchronize
|
|
LONGCALLNOARGS HostResynchronize
|
|
ENDC
|
|
|
|
.exchangeBCB
|
|
;----exchange Button Control Bytes with remote machine--------
|
|
ld a,[curInput0]
|
|
call EncodeInput
|
|
call ExchangeByte
|
|
call HandleRemoteInput
|
|
jr nz,.exchangeBCB ;must repeat
|
|
|
|
;--Decode Button Control Bytes--------------------------------
|
|
push af ;save remoteInput
|
|
ld a,[amLinkMaster]
|
|
or a
|
|
jr z,.amLinkSlave
|
|
|
|
;I am the master. curInput0->curJoy0, remoteInput->curJoy1
|
|
pop af ;retrieve remoteInput
|
|
call DecodeInput
|
|
ld [curInput1],a
|
|
ld [curJoy1],a
|
|
ld a,[curInput0]
|
|
call ConvertInput
|
|
ld [curInput0],a
|
|
ld [curJoy0],a
|
|
ld [myJoy],a
|
|
ret
|
|
|
|
.amLinkSlave
|
|
ld a,[curInput0]
|
|
call ConvertInput
|
|
ld [curInput1],a
|
|
ld [curJoy1],a
|
|
ld [myJoy],a
|
|
|
|
pop af ;retrieve remoteInput
|
|
call DecodeInput
|
|
ld [curInput0],a
|
|
ld [curJoy0],a
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: ConvertInput
|
|
; Argument: a - uncoded button control byte
|
|
; Returns: a - converted BCB, same as calling Encode then
|
|
; Decode but quicker
|
|
; Description: To simulate uncoded->encoded->decoded, forces buttons
|
|
; 7:4 to have A/B or Start/Select info but not both
|
|
; (A/B has priority).
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
ConvertInput:
|
|
push bc
|
|
|
|
ld b,a ;save original
|
|
and %00110000 ;has A/B info?
|
|
ld a,b ;restore original
|
|
jr z,.done ;if no AB info then conv. complete
|
|
and %00111111 ;has AB info; clear out start/sel
|
|
|
|
.done
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: EncodeInput
|
|
; Argument: a - uncoded button control byte
|
|
; 7:st 6:se 5:b 4:a 3:0=DULR
|
|
; Returns: a - coded button control byte
|
|
; 7=0
|
|
; 6=0 - 5:4 = ba
|
|
; =1 - 5:4 = te (start, select)
|
|
; 3:0 = DULR
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
EncodeInput:
|
|
push bc
|
|
push hl
|
|
|
|
ld b,a ;save original
|
|
and %00001111
|
|
ld c,a ;save D-PAD
|
|
|
|
ld h,((encodeControlByteTable>>8) & $ff)
|
|
ld a,b
|
|
swap a
|
|
and %00001111
|
|
add (encodeControlByteTable & $ff)
|
|
ld l,a
|
|
|
|
ld a,[hl] ;get encoded buttons
|
|
or c ;combine with D-PAD
|
|
|
|
pop hl
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: DecodeInput
|
|
; Argument: a - coded Control Byte
|
|
; Returns: a - uncoded button Control Byte
|
|
; Alters: af
|
|
;---------------------------------------------------------------------
|
|
DecodeInput:
|
|
push bc
|
|
push hl
|
|
|
|
ld b,a ;save original
|
|
and %00001111
|
|
ld c,a ;save D-PAD
|
|
|
|
ld h,((decodeControlByteTable>>8) & $ff)
|
|
ld a,b
|
|
swap a
|
|
and %00000111
|
|
add (decodeControlByteTable & $ff)
|
|
ld l,a
|
|
|
|
ld a,[hl] ;get decoded buttons
|
|
or c ;combine with D-PAD
|
|
|
|
pop hl
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: TransmitGameState
|
|
; Arguments: none
|
|
; Returns: none
|
|
; Alters: af
|
|
; Description: Transmits relevant game state data to Slave
|
|
; Includes hero data so slave's classes will be set
|
|
; appropriately
|
|
;---------------------------------------------------------------------
|
|
TransmitGameState:
|
|
push bc
|
|
push hl
|
|
|
|
ld hl,.linkBailAddress
|
|
xor a
|
|
call SetLinkBailOutAddress
|
|
|
|
ld hl,gameState
|
|
ld bc,5
|
|
xor a ;RAM bank doesn't matter
|
|
call TransmitData
|
|
|
|
ld hl,levelState
|
|
ld bc,256
|
|
ld a,LEVELSTATEBANK
|
|
call TransmitData
|
|
|
|
ld hl,inventory
|
|
ld bc,16
|
|
xor a
|
|
call TransmitData
|
|
|
|
ld hl,flightCode
|
|
ld bc,256
|
|
ld a,FLIGHTCODEBANK
|
|
call TransmitData
|
|
|
|
ld hl,hero0_data
|
|
ld bc,HERODATASIZE * 2
|
|
xor a
|
|
call TransmitData
|
|
|
|
ld a,[appomattoxMapIndex]
|
|
call TransmitByte
|
|
|
|
;send the joinMap
|
|
ld a,[joinMap]
|
|
call TransmitByte
|
|
ld a,[joinMap+1]
|
|
call TransmitByte
|
|
|
|
.linkBailAddress
|
|
ld hl,0
|
|
xor a
|
|
call SetLinkBailOutAddress
|
|
|
|
pop hl
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: HostSynchronize
|
|
; Arguments: none
|
|
; Returns: none
|
|
; Alters: af
|
|
; Description: Transmits relevant map data to Guest
|
|
;---------------------------------------------------------------------
|
|
HostResynchronize:
|
|
push bc
|
|
push hl
|
|
jr HostSynchronizeCommon
|
|
|
|
HostSynchronize:
|
|
push bc
|
|
push hl
|
|
|
|
ld hl,.nogo
|
|
xor a
|
|
call SetLinkBailOutAddress
|
|
|
|
ld a,1 ;make sure it's looking at 1x1
|
|
ldio [curObjWidthHeight],a
|
|
call ReceiveByte ;guest's requested exit direction
|
|
ld hl,$0101 ;start looking at (1,1)
|
|
call FindExitLocation
|
|
xor a ;didn't find it if $0000
|
|
cp h
|
|
jr nz,.goAhead
|
|
cp l
|
|
jr nz,.goAhead
|
|
|
|
;nogo
|
|
xor a
|
|
call TransmitByte
|
|
.nogo
|
|
ld hl,0
|
|
xor a
|
|
call SetLinkBailOutAddress
|
|
|
|
pop hl
|
|
pop bc
|
|
ret
|
|
|
|
.goAhead
|
|
call ConvertLocHLToXY
|
|
ld a,1
|
|
call TransmitByte
|
|
|
|
ldio a,[mapState] ;level state
|
|
call TransmitByte
|
|
ldio a,[mapState+1]
|
|
call TransmitByte
|
|
|
|
;make sure backbuffer doesn't redraw during hero loading
|
|
xor a
|
|
ldio [backBufferReady],a
|
|
|
|
;activate heroes 0 and 1
|
|
ld a,[hero0_index]
|
|
or a
|
|
jr nz,.hero0_active
|
|
|
|
ld a,1
|
|
ld [hero0_index],a
|
|
|
|
.hero0_active
|
|
ld a,[hero1_index]
|
|
or a
|
|
jr nz,.hero1_active
|
|
|
|
ld a,1
|
|
ld [hero1_index],a
|
|
.hero1_active
|
|
call HostExchangeHeroData
|
|
|
|
;set hero 1 to joystick 1
|
|
push hl
|
|
ld a,[hero1_type]
|
|
ld hl,heroJoyIndex
|
|
or [hl]
|
|
ld [hl],a
|
|
pop hl
|
|
|
|
inc hl
|
|
ld b,h
|
|
ld c,l
|
|
PREPLONGCALL .afterSetupHero
|
|
LONGCALL PrepSetupHeroBC ;bc is correct hero data
|
|
.afterSetupHero
|
|
|
|
HostSynchronizeCommon:
|
|
xor a
|
|
ld [backBufferReady],a
|
|
|
|
;wait on ready signal from guest
|
|
.waitReady
|
|
call ReceiveByte
|
|
cp LSYNCHREADY
|
|
jr nz,.waitReady
|
|
|
|
;send all my hero data ;re-send all my hero data
|
|
ld hl,hero0_data
|
|
ld bc,HERODATASIZE*2
|
|
xor a
|
|
call TransmitData
|
|
|
|
ld a,[randomLoc] ;current random value/seed
|
|
call TransmitByte
|
|
|
|
ld a,[heroesIdle]
|
|
call TransmitByte
|
|
|
|
LONGCALLNOARGS RemoveSpriteObjectsFromMap
|
|
|
|
;----transmit map---------------------------------------------
|
|
ld hl,map
|
|
ld bc,4096
|
|
ld a,MAPBANK
|
|
call TransmitData
|
|
|
|
;fadeCurPalette
|
|
ld hl,fadeCurPalette
|
|
ld bc,128
|
|
xor a
|
|
call TransmitData
|
|
|
|
;gamePalette
|
|
ld hl,gamePalette
|
|
ld bc,128
|
|
ld a,FADEBANK
|
|
call TransmitData
|
|
|
|
;first 16 bytes of level check RAM
|
|
ld hl,levelCheckRAM
|
|
ld bc,16
|
|
xor a
|
|
call TransmitData
|
|
|
|
;spriteOAMdata
|
|
ld hl,spriteOAMBuffer
|
|
ld bc,160
|
|
xor a
|
|
call TransmitData
|
|
|
|
ld hl,headTable ;headTable - linked list head
|
|
ld bc,256
|
|
ld a,OBJLISTBANK
|
|
call TransmitData
|
|
|
|
ld hl,objExists ;objExists, FOF table
|
|
ld bc,512
|
|
ld a,OBJLISTBANK
|
|
call TransmitCompressedData
|
|
|
|
ld a,[numClasses]
|
|
call TransmitByte
|
|
|
|
;bc = numClasses times two + 2
|
|
ld a,[numClasses]
|
|
ld b,0
|
|
ld c,a
|
|
sla c
|
|
rl b
|
|
inc bc
|
|
inc bc
|
|
|
|
ld hl,classLookup ;objExists, FOF table
|
|
;ld bc,512 numClasses * 2
|
|
ld a,OBJLISTBANK
|
|
call TransmitData
|
|
|
|
ld hl,fgTileMap
|
|
ld a,[numClasses]
|
|
ld b,0
|
|
ld c,a
|
|
ld a,OBJLISTBANK
|
|
call TransmitData
|
|
|
|
ld hl,objClassLookup ;class indices for each obj
|
|
ld bc,256
|
|
ld a,OBJLISTBANK
|
|
call TransmitData
|
|
|
|
ld hl,associatedIndex
|
|
ld bc,256
|
|
ld a,OBJLISTBANK
|
|
call TransmitData
|
|
|
|
ld hl,spritesUsed
|
|
ld bc,40
|
|
ld a,OBJLISTBANK
|
|
call TransmitCompressedData
|
|
|
|
;---------------transmit used objects-------------------------
|
|
ld a,OBJLISTBANK
|
|
ld [$ff70],a
|
|
ld de,objExists+1
|
|
.transmitUsedObject
|
|
ld a,[de] ;is this object used?
|
|
or a
|
|
jr z,.afterTransmitUsedObject ;not used
|
|
|
|
PREPLONGCALL .afterCvtToPtr
|
|
ld a,e ;get object index
|
|
LONGCALL IndexToPointerHL ;cvt to ptr
|
|
.afterCvtToPtr
|
|
ld bc,16
|
|
ld a,OBJBANK
|
|
call TransmitData
|
|
ld a,OBJLISTBANK
|
|
ld [$ff70],a
|
|
|
|
.afterTransmitUsedObject
|
|
inc de
|
|
ld a,e
|
|
or a
|
|
jr nz,.transmitUsedObject
|
|
|
|
ld a,[numFreeSprites]
|
|
call TransmitByte
|
|
|
|
ld a,[firstFreeObj]
|
|
call TransmitByte
|
|
|
|
ld a,[randomLoc]
|
|
call TransmitByte
|
|
|
|
ld a,[guardAlarm]
|
|
call TransmitByte
|
|
|
|
;ld a,[dialogBank]
|
|
;call TransmitByte
|
|
|
|
ld a,[respawnMap]
|
|
call TransmitByte
|
|
ld a,[respawnMap+1]
|
|
call TransmitByte
|
|
|
|
ldio a,[mapState] ;level state
|
|
call TransmitByte
|
|
ldio a,[mapState+1]
|
|
call TransmitByte
|
|
|
|
ld hl,levelVars
|
|
ld bc,64
|
|
xor a ;RAM bank doesn't matter
|
|
call TransmitData
|
|
|
|
ld hl,musicBank
|
|
ld bc,64
|
|
xor a ;RAM bank doesn't matter
|
|
call TransmitData
|
|
|
|
ld hl,musicStack
|
|
ld bc,128
|
|
ld a,MUSICBANK
|
|
call TransmitData
|
|
|
|
ldio a,[musicEnabled]
|
|
call TransmitByte
|
|
|
|
LONGCALLNOARGS LinkRemakeLists
|
|
|
|
.done
|
|
pop hl
|
|
pop bc
|
|
ret
|
|
|
|
HostExchangeHeroData:
|
|
;send out my hero data
|
|
ld a,[amLinkMaster]
|
|
or a
|
|
jr z,.slave_sendHero1_recvHero0
|
|
|
|
.master_sendHero0_recvHero1
|
|
ld hl,hero1_data
|
|
push hl
|
|
ld hl,hero0_data
|
|
jr .setHeroSequence
|
|
|
|
.slave_sendHero1_recvHero0
|
|
ld hl,hero0_data
|
|
push hl
|
|
ld hl,hero1_data
|
|
|
|
.setHeroSequence
|
|
;send my hero data
|
|
ld bc,HERODATASIZE
|
|
xor a
|
|
call TransmitData
|
|
|
|
;get remote hero data
|
|
pop hl
|
|
xor a
|
|
call ReceiveData
|
|
|
|
ret
|
|
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: LinkTest
|
|
; Arguments: none
|
|
; Returns: none
|
|
; Alters: af
|
|
; Description: Fills 4k of memory with repeating pattern $00-$ff,
|
|
; transmits that data to Slave, receives that data
|
|
; back, compares it to the original pattern, and
|
|
; finally prints out a Debug message of 1 for equal
|
|
; or 0 for not equal.
|
|
;---------------------------------------------------------------------
|
|
LinkTest:
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
di
|
|
;fill $d000-$cfff with repeating pattern $00-$ff
|
|
ld a,1
|
|
ld [$ff70],a
|
|
|
|
ld hl,$d000
|
|
xor a
|
|
.fill
|
|
ld [hl+],a
|
|
inc a
|
|
jr nz,.fill
|
|
ld a,h
|
|
cp $e0
|
|
ld a,0 ;avoid setting z flag here
|
|
jr nz,.fill
|
|
|
|
ld a,4
|
|
.transfer32k
|
|
push af
|
|
ld a,1
|
|
ld bc,$1000
|
|
ld hl,$d000
|
|
call TransmitData
|
|
|
|
ld a,1
|
|
call ReceiveData
|
|
pop af
|
|
dec a
|
|
jr nz,.transfer32k
|
|
|
|
;compare received data to original pattern
|
|
ld b,0 ;comparing to $00
|
|
ld hl,$d000
|
|
.compare
|
|
ld a,[hl+]
|
|
cp b
|
|
jr nz,.notEqual
|
|
inc b
|
|
ld a,h
|
|
cp $e0
|
|
jr nz,.compare
|
|
|
|
.equal
|
|
xor a
|
|
ldio [$ff0f],a
|
|
ei
|
|
ld a,1
|
|
call DebugVal
|
|
jr .infi
|
|
|
|
.notEqual
|
|
xor a
|
|
ldio [$ff0f],a
|
|
ei
|
|
xor a
|
|
call DebugVal
|
|
|
|
dec hl
|
|
ld a,h
|
|
call DebugVal
|
|
ld a,l
|
|
call DebugVal
|
|
di
|
|
.infi jr .infi
|
|
|
|
pop hl
|
|
pop de
|
|
pop bc
|
|
ret
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: GeneratePauseMessage
|
|
; Arguments: none
|
|
; Returns: none
|
|
; Alters: all
|
|
; Description: Creates appropriate Pause Message
|
|
;---------------------------------------------------------------------
|
|
GeneratePauseMessage:
|
|
ld a,[dialogBank]
|
|
push af
|
|
|
|
;wait until the backbuffer is blitted
|
|
.waitBackBuffer
|
|
ldio a,[backBufferReady]
|
|
or a
|
|
jr z,.canMessWithBackBuffer
|
|
|
|
call VWait
|
|
jr .waitBackBuffer
|
|
|
|
.canMessWithBackBuffer
|
|
;generate gtx at backbuffer+$800
|
|
|
|
;de = level name index
|
|
ld a,[curLevelStateIndex]
|
|
ld d,0
|
|
ld e,a ;level index * 2
|
|
sla e
|
|
rl d
|
|
ld hl,levelNames
|
|
add hl,de
|
|
ld a,[hl+]
|
|
ld e,a
|
|
ld a,[hl+]
|
|
ld d,a
|
|
ld hl,backBuffer
|
|
ld a,BANK(levelNames)
|
|
ld [dialogBank],a
|
|
call ClearGTXLine
|
|
call WriteGTXLine
|
|
|
|
;write char name
|
|
push hl
|
|
push hl
|
|
LDHL_CURHERODATA HERODATA_TYPE
|
|
ld a,[hl]
|
|
ld c,$ff
|
|
.findCharIndex
|
|
inc c
|
|
rrca
|
|
bit 7,a
|
|
jr z,.findCharIndex
|
|
ld a,c
|
|
ld hl,heroNames
|
|
call .lookupIndexAToHLToDE
|
|
pop hl
|
|
call ClearGTXLine
|
|
call WriteGTXLine
|
|
pop hl
|
|
|
|
;write map coordinates
|
|
add sp,-24 ;create temporary buffer on stack
|
|
push hl
|
|
ld hl,sp+2
|
|
ld d,h
|
|
ld e,l
|
|
ld [hl],16 ;spaces to center
|
|
inc hl
|
|
ld [hl],4 ;number of chars
|
|
inc hl
|
|
ld a,[curLevelStateIndex]
|
|
push af
|
|
and $f
|
|
add 210 ;'A' character
|
|
ld [hl+],a
|
|
pop af
|
|
ld [hl],$f9 ;hyphen
|
|
inc hl
|
|
swap a
|
|
and $f
|
|
cp 10
|
|
jr c,.oneDigit
|
|
ld [hl],201 ;'1'
|
|
inc hl
|
|
sub 10
|
|
.oneDigit
|
|
add 200 ;get digit '0'-'9'
|
|
ld [hl+],a
|
|
ld [hl],0 ;blank space may or may not be used
|
|
pop hl
|
|
push de
|
|
call WriteGTXLine ;de is line to write
|
|
pop de
|
|
|
|
;write map coordinates if landing zone------------------------
|
|
ld c,2 ;# of items in inventory screen
|
|
|
|
push bc ;see if we're at an LZ
|
|
ld a,[curLevelStateIndex]
|
|
call .getFlightCodeBCFromMapA
|
|
pop bc
|
|
jr z,.afterWriteCoords
|
|
|
|
push hl
|
|
ld h,d
|
|
ld l,e
|
|
ld [hl],8 ;spaces to center
|
|
inc hl
|
|
ld [hl],4 ;number of chars
|
|
inc hl
|
|
push bc
|
|
ld a,[curLevelStateIndex]
|
|
call .getFlightCodeBCFromMapA
|
|
call .writeCoordsBC
|
|
pop bc
|
|
pop hl
|
|
call ClearGTXLine
|
|
call WriteGTXLine ;de is line to write
|
|
inc c
|
|
.afterWriteCoords
|
|
add sp,24 ;free up temporary stack space
|
|
|
|
;write inventory
|
|
call GetFirstInventoryIndex
|
|
cp $ff
|
|
jr z,.afterWriteInventory
|
|
|
|
;blank line
|
|
push af
|
|
ld de,blankLine
|
|
call ClearGTXLine
|
|
call WriteGTXLine
|
|
inc c
|
|
pop af
|
|
|
|
;remaining inventory
|
|
.nextInventoryItem
|
|
ld b,a ;obj index in b
|
|
push af
|
|
push hl
|
|
ld hl,itemNames
|
|
call .lookupIndexAToHLToDE
|
|
pop hl
|
|
|
|
inc de
|
|
ld a,[de]
|
|
dec de
|
|
or a
|
|
jr z,.skipItem ;an upgrade or something
|
|
|
|
call ClearGTXLine
|
|
push hl
|
|
call WriteGTXLine
|
|
|
|
;write object color
|
|
ld a,b
|
|
add (itemColors&$ff)
|
|
ld e,a
|
|
ld a,((itemColors>>8)&$ff)
|
|
adc 0
|
|
ld d,a
|
|
ld a,[de]
|
|
or 8
|
|
pop de
|
|
push hl
|
|
ld hl,$300
|
|
add hl,de
|
|
ld [hl],a
|
|
pop hl
|
|
inc c
|
|
.skipItem
|
|
pop af
|
|
call GetNextInventoryIndex
|
|
cp $ff
|
|
jr nz,.nextInventoryItem
|
|
|
|
.afterWriteInventory
|
|
inc c
|
|
ld de,restart_message
|
|
call ClearGTXLine
|
|
call WriteGTXLine
|
|
|
|
ld a,c
|
|
ld [backBuffer+63],a ;#lines
|
|
|
|
pop af
|
|
ld [dialogBank],a
|
|
ret
|
|
|
|
.getFlightCodeBCFromMapA
|
|
;returns flight code in bc, zflag
|
|
push de
|
|
push hl
|
|
ld de,3
|
|
|
|
ld b,a
|
|
ld a,FLIGHTCODEBANK
|
|
ldio [$ff70],a
|
|
ld hl,flightCode
|
|
ld c,[hl] ;number of flight codes
|
|
add hl,de ;start at byte 2/3 (map index)
|
|
.findFlightCode
|
|
ld a,[hl]
|
|
cp b
|
|
jr z,.getFlightCodeMatch
|
|
add hl,de
|
|
dec c
|
|
jr nz,.findFlightCode
|
|
|
|
xor a ;not found
|
|
pop hl
|
|
pop de
|
|
ret
|
|
|
|
.getFlightCodeMatch
|
|
dec hl
|
|
ld b,[hl]
|
|
dec hl
|
|
ld c,[hl]
|
|
pop hl
|
|
pop de
|
|
ld a,1
|
|
or a
|
|
ret
|
|
|
|
.lookupIndexAToHLToDE
|
|
rlca
|
|
add l
|
|
ld l,a
|
|
ld a,h
|
|
adc 0
|
|
ld h,a
|
|
ld a,[hl+]
|
|
ld d,[hl]
|
|
ld e,a
|
|
ret
|
|
|
|
.writeCoordsBC
|
|
ld a,b
|
|
swap a
|
|
call .writeCoordA
|
|
ld a,b
|
|
call .writeCoordA
|
|
ld a,c
|
|
swap a
|
|
call .writeCoordA
|
|
ld a,c
|
|
call .writeCoordA
|
|
ret
|
|
|
|
.writeCoordA
|
|
and $f
|
|
add $c0 ;first of 8 coordinate symbols
|
|
ld [hl+],a
|
|
ret
|
|
|
|
restart_message:
|
|
GTXSTRINGC "SELECTgoRESTART"
|
|
blankLine:
|
|
DB 0,0
|
|
|
|
;---------------------------------------------------------------------
|
|
; Routine: CopyMessageDEToHL
|
|
; Arguments: de - pointer to message
|
|
; hl - pointer to dest
|
|
; Returns: none
|
|
; Alters: all
|
|
; Description: First byte at [de] is number of bytes to copy from
|
|
; [de] to [hl]. Must be >= 1!
|
|
;---------------------------------------------------------------------
|
|
CopyMessageDEToHL:
|
|
ld a,[de]
|
|
inc de
|
|
ld c,a
|
|
.loop ld a,[de]
|
|
inc de
|
|
ld [hl+],a
|
|
dec c
|
|
jr nz,.loop
|
|
ret
|
|
|
|
levelNames:
|
|
DW L0000_Name,L0100_Name,L0200_Name,L0300_Name
|
|
DW L0400_Name,L0500_Name,L0600_Name,L0700_Name
|
|
DW L0800_Name,L0900_Name,L1000_Name,L1100_Name
|
|
DW L1200_Name,L1300_Name,L1400_Name,L1500_Name
|
|
|
|
DW L0001_Name,L0101_Name,L0201_Name,L0301_Name
|
|
DW L0401_Name,L0501_Name,L0601_Name,L0701_Name
|
|
DW L0801_Name,L0901_Name,L1001_Name,L1101_Name
|
|
DW L1201_Name,L1301_Name,L1401_Name,L1501_Name
|
|
|
|
DW L0002_Name,L0102_Name,L0202_Name,L0302_Name
|
|
DW L0402_Name,L0502_Name,L0602_Name,L0702_Name
|
|
DW L0802_Name,L0902_Name,L1002_Name,L1102_Name
|
|
DW L1202_Name,L1302_Name,L1402_Name,L1502_Name
|
|
|
|
DW L0003_Name,L0103_Name,L0203_Name,L0303_Name
|
|
DW L0403_Name,L0503_Name,L0603_Name,L0703_Name
|
|
DW L0803_Name,L0903_Name,L1003_Name,L1103_Name
|
|
DW L1203_Name,L1303_Name,L1403_Name,L1503_Name
|
|
|
|
DW L0004_Name,L0104_Name,L0204_Name,L0304_Name
|
|
DW L0404_Name,L0504_Name,L0604_Name,L0704_Name
|
|
DW L0804_Name,L0904_Name,L1004_Name,L1104_Name
|
|
DW L1204_Name,L1304_Name,L1404_Name,L1504_Name
|
|
|
|
DW L0005_Name,L0105_Name,L0205_Name,L0305_Name
|
|
DW L0405_Name,L0505_Name,L0605_Name,L0705_Name
|
|
DW L0805_Name,L0905_Name,L1005_Name,L1105_Name
|
|
DW L1205_Name,L1305_Name,L1405_Name,L1505_Name
|
|
|
|
DW L0006_Name,L0106_Name,L0206_Name,L0306_Name
|
|
DW L0406_Name,L0506_Name,L0606_Name,L0706_Name
|
|
DW L0806_Name,L0906_Name,L1006_Name,L1106_Name
|
|
DW L1206_Name,L1306_Name,L1406_Name,L1506_Name
|
|
|
|
DW L0007_Name,L0107_Name,L0207_Name,L0307_Name
|
|
DW L0407_Name,L0507_Name,L0607_Name,L0707_Name
|
|
DW L0807_Name,L0907_Name,L1007_Name,L1107_Name
|
|
DW L1207_Name,L1307_Name,L1407_Name,L1507_Name
|
|
|
|
DW L0008_Name,L0108_Name,L0208_Name,L0308_Name
|
|
DW L0408_Name,L0508_Name,L0608_Name,L0708_Name
|
|
DW L0808_Name,L0908_Name,L1008_Name,L1108_Name
|
|
DW L1208_Name,L1308_Name,L1408_Name,L1508_Name
|
|
|
|
DW L0009_Name,L0109_Name,L0209_Name,L0309_Name
|
|
DW L0409_Name,L0509_Name,L0609_Name,L0709_Name
|
|
DW L0809_Name,L0909_Name,L1009_Name,L1109_Name
|
|
DW L1209_Name,L1309_Name,L1409_Name,L1509_Name
|
|
|
|
DW L0010_Name,L0110_Name,L0210_Name,L0310_Name
|
|
DW L0410_Name,L0510_Name,L0610_Name,L0710_Name
|
|
DW L0810_Name,L0910_Name,L1010_Name,L1110_Name
|
|
DW L1210_Name,L1310_Name,L1410_Name,L1510_Name
|
|
|
|
DW L0011_Name,L0111_Name,L0211_Name,L0311_Name
|
|
DW L0411_Name,L0511_Name,L0611_Name,L0711_Name
|
|
DW L0811_Name,L0911_Name,L1011_Name,L1111_Name
|
|
DW L1211_Name,L1311_Name,L1411_Name,L1511_Name
|
|
|
|
DW L0012_Name,L0112_Name,L0212_Name,L0312_Name
|
|
DW L0412_Name,L0512_Name,L0612_Name,L0712_Name
|
|
DW L0812_Name,L0912_Name,L1012_Name,L1112_Name
|
|
DW L1212_Name,L1312_Name,L1412_Name,L1512_Name
|
|
|
|
DW L0013_Name,L0113_Name,L0213_Name,L0313_Name
|
|
DW L0413_Name,L0513_Name,L0613_Name,L0713_Name
|
|
DW L0813_Name,L0913_Name,L1013_Name,L1113_Name
|
|
DW L1213_Name,L1313_Name,L1413_Name,L1513_Name
|
|
|
|
DW L0014_Name,L0114_Name,L0214_Name,L0314_Name
|
|
DW L0414_Name,L0514_Name,L0614_Name,L0714_Name
|
|
DW L0814_Name,L0914_Name,L1014_Name,L1114_Name
|
|
DW L1214_Name,L1314_Name,L1414_Name,L1514_Name
|
|
|
|
DW L0015_Name,L0115_Name,L0215_Name,L0315_Name
|
|
DW L0415_Name,L0515_Name,L0615_Name,L0715_Name
|
|
DW L0815_Name,L0915_Name,L1015_Name,L1115_Name
|
|
DW L1215_Name,L1315_Name,L1415_Name,L1515_Name
|
|
|
|
L0000_Name:
|
|
GTXSTRINGC "THEoHIVE"
|
|
L0001_Name:
|
|
GTXSTRINGC "MADoBEEoPASS"
|
|
L0002_Name:
|
|
GTXSTRINGC "BEEoCOUNTRY"
|
|
L0003_Name:
|
|
GTXSTRINGC "BIGoEDaSoSHEEPoFARM"
|
|
L0004_Name:
|
|
GTXSTRINGC "HILLoBROTHERSoRANCH"
|
|
L0005_Name:
|
|
GTXSTRINGC "TRAKKTORoCOUNTRY"
|
|
L0006_Name:
|
|
GTXSTRINGC "SUNSEToVILLAGE"
|
|
L0007_Name:
|
|
GTXSTRINGC "CRAZYoCROWoGULCH"
|
|
L0008_Name:
|
|
GTXSTRINGC "SNEAKYoTREEoVALLEY"
|
|
L0009_Name:
|
|
GTXSTRINGC "PROVOLONE"
|
|
L0010_Name:
|
|
GTXSTRINGC "MUENSTER"
|
|
L0011_Name:
|
|
GTXSTRINGC "HOUSEoOFoSEASONS"
|
|
L0012_Name:
|
|
GTXSTRINGC "COMMANDoCORE"
|
|
L0013_Name:
|
|
GTXSTRINGC "ITCHYoTRIGGER"
|
|
L0014_Name:
|
|
GTXSTRINGC "THEoFAULT"
|
|
L0015_Name:
|
|
GTXSTRINGC "LOOSEoJARGON"
|
|
|
|
L0100_Name:
|
|
L0101_Name:
|
|
L0102_Name:
|
|
GTXSTRINGC "RIVER"
|
|
L0103_Name:
|
|
GTXSTRINGC "LILoEDaSoSHEEPoFARM"
|
|
L0104_Name:
|
|
GTXSTRINGC "SPRING"
|
|
L0105_Name:
|
|
GTXSTRINGC "DARKERoFOREST"
|
|
L0106_Name:
|
|
GTXSTRINGC "DARKISHoFOREST"
|
|
L0107_Name:
|
|
GTXSTRINGC "KIDNAPoCORNERoeLZf"
|
|
L0108_Name:
|
|
GTXSTRINGC "SNAKEoPIToeLZf"
|
|
L0109_Name:
|
|
GTXSTRINGC "CHEDDAR"
|
|
L0110_Name:
|
|
GTXSTRINGC "HAVARTI"
|
|
L0111_Name:
|
|
GTXSTRINGC "GOBLINoPIT"
|
|
L0112_Name:
|
|
GTXSTRINGC "HIVEoENTRANCE"
|
|
L0113_Name:
|
|
GTXSTRINGC "FRIENDLYoFIRE"
|
|
L0114_Name:
|
|
GTXSTRINGC "PATHoOFoSHADOWS"
|
|
L0115_Name:
|
|
GTXSTRINGC "DOUBLEoENTENDRE"
|
|
|
|
L0200_Name:
|
|
GTXSTRINGC "FORBIDDENoVILLAGE"
|
|
L0201_Name:
|
|
GTXSTRINGC "STRANGEoMIST"
|
|
L0202_Name:
|
|
L0203_Name:
|
|
L0204_Name:
|
|
GTXSTRINGC "RIVER"
|
|
L0205_Name:
|
|
GTXSTRINGC "JACOBaSoBRIDGE"
|
|
L0206_Name:
|
|
L0207_Name:
|
|
L0208_Name:
|
|
L0209_Name:
|
|
L0210_Name:
|
|
GTXSTRINGC "RIVER"
|
|
L0211_Name:
|
|
GTXSTRINGC "THEoBLOWERS"
|
|
L0212_Name:
|
|
GTXSTRINGC "THEoBELFRY"
|
|
L0213_Name:
|
|
GTXSTRINGC "CLOSEoQUARTERS"
|
|
L0214_Name:
|
|
GTXSTRINGC "JAILoBREAK"
|
|
L0215_Name:
|
|
GTXSTRINGC "BRAINIAC"
|
|
|
|
L0300_Name:
|
|
GTXSTRINGC "WEIRDoMIST"
|
|
L0301_Name:
|
|
GTXSTRINGC "ODDoMIST"
|
|
L0302_Name:
|
|
GTXSTRINGC "BIZARREoMISToeLZf"
|
|
L0303_Name:
|
|
GTXSTRINGC "MISToGATE"
|
|
L0304_Name:
|
|
GTXSTRINGC "MUNGAoFUNGA"
|
|
L0305_Name:
|
|
GTXSTRINGC "TRILINGoBURROW"
|
|
L0306_Name:
|
|
GTXSTRINGC "TWOoGUNS"
|
|
L0307_Name:
|
|
GTXSTRINGC "OUTPOSTgoDESOLATION"
|
|
L0308_Name:
|
|
GTXSTRINGC "DRYoGULCH"
|
|
L0309_Name:
|
|
GTXSTRINGC "SHOOToOUT"
|
|
L0310_Name:
|
|
GTXSTRINGC "REDoBENDoeLZf"
|
|
L0311_Name:
|
|
GTXSTRINGC "HIGHoGROUND"
|
|
L0312_Name:
|
|
GTXSTRINGC "THEoSTING"
|
|
L0313_Name:
|
|
GTXSTRINGC "INSOMNIAC"
|
|
L0314_Name:
|
|
GTXSTRINGC "RUNAWAY"
|
|
L0315_Name:
|
|
GTXSTRINGC "INSOMNIAC"
|
|
|
|
L0400_Name:
|
|
GTXSTRINGC "OUTPOSTgoDARKFROST"
|
|
L0401_Name:
|
|
GTXSTRINGC "PITCHoBLACK"
|
|
L0402_Name:
|
|
GTXSTRINGC "WYRMoDUSK"
|
|
L0403_Name:
|
|
GTXSTRINGC "GROMMOLD"
|
|
L0404_Name:
|
|
GTXSTRINGC "MILDEWoSTIRRING"
|
|
L0405_Name:
|
|
GTXSTRINGC "WESTGATE"
|
|
L0406_Name:
|
|
GTXSTRINGC "PIGSoINoAoBLANKET"
|
|
L0407_Name:
|
|
GTXSTRINGC "BIOS"
|
|
L0408_Name:
|
|
GTXSTRINGC "HOToVALLEYoOFoDEATH"
|
|
L0409_Name:
|
|
GTXSTRINGC "TUMBLEWEED"
|
|
L0410_Name:
|
|
GTXSTRINGC "OUTPOSTgoDRAGON"
|
|
L0411_Name:
|
|
GTXSTRINGC "THEoCELL"
|
|
L0412_Name:
|
|
GTXSTRINGC "STONEHEADoHOUSE"
|
|
L0413_Name:
|
|
L0414_Name:
|
|
L0415_Name:
|
|
GTXSTRINGC "FGB"
|
|
|
|
L0500_Name:
|
|
GTXSTRINGC "PENGUINoSMACKDOWN"
|
|
L0501_Name:
|
|
GTXSTRINGC "WINTER"
|
|
L0502_Name:
|
|
GTXSTRINGC "CHILLVILLE"
|
|
L0503_Name:
|
|
GTXSTRINGC "OLDoHOSS"
|
|
L0504_Name:
|
|
GTXSTRINGC "NORTHGATE"
|
|
L0505_Name:
|
|
GTXSTRINGC "FLOWERoTOWER"
|
|
L0506_Name:
|
|
GTXSTRINGC "SOUTHGATE"
|
|
L0507_Name:
|
|
GTXSTRINGC "WETLANDS"
|
|
L0508_Name:
|
|
GTXSTRINGC "LAaSHANDAaS"
|
|
L0509_Name:
|
|
GTXSTRINGC "OUTBACK"
|
|
L0510_Name:
|
|
GTXSTRINGC "SUMMER"
|
|
L0511_Name:
|
|
L0512_Name:
|
|
GTXSTRINGC "LAaSHANDA"
|
|
L0513_Name:
|
|
L0514_Name:
|
|
L0515_Name:
|
|
GTXSTRINGC "FGB"
|
|
|
|
L0600_Name:
|
|
GTXSTRINGC "ICEoPLAINSoeLZf"
|
|
L0601_Name:
|
|
GTXSTRINGC "ICEWOLFoRUN"
|
|
L0602_Name:
|
|
GTXSTRINGC "KINGoCHUBBAaSoHALL"
|
|
L0603_Name:
|
|
GTXSTRINGC "IGNORANToHILLoPEOPLE"
|
|
L0604_Name:
|
|
GTXSTRINGC "TOWERoOFoPAIN"
|
|
L0605_Name:
|
|
GTXSTRINGC "EASTGATE"
|
|
L0606_Name:
|
|
GTXSTRINGC "ROCKoBOTTOMoBENDeLZf"
|
|
L0607_Name:
|
|
GTXSTRINGC "SWAMPYaSoLAIR"
|
|
L0608_Name:
|
|
GTXSTRINGC "GOONoHILL"
|
|
L0609_Name:
|
|
GTXSTRINGC "LASToRESORT"
|
|
L0610_Name:
|
|
GTXSTRINGC "BITTERoROCKoPARK"
|
|
L0611_Name:
|
|
L0612_Name:
|
|
GTXSTRINGC "TELEPORToCHAMBER"
|
|
L0613_Name:
|
|
L0614_Name:
|
|
L0615_Name:
|
|
GTXSTRINGC "FGB"
|
|
|
|
L0700_Name:
|
|
GTXSTRINGC "THEoPLANEoOFoNOTHING"
|
|
L0701_Name:
|
|
GTXSTRINGC "KIWIoKEEP"
|
|
L0702_Name:
|
|
GTXSTRINGC "BAToCAVE"
|
|
L0703_Name:
|
|
GTXSTRINGC "UNDERLAKEoCAVERNS"
|
|
L0704_Name:
|
|
GTXSTRINGC "STONEHEADoCANYONeLZf"
|
|
L0705_Name:
|
|
GTXSTRINGC "MONKEYoHOLLER"
|
|
L0706_Name:
|
|
GTXSTRINGC "SWAMPLACE"
|
|
L0707_Name:
|
|
GTXSTRINGC "THEoBARROWS"
|
|
L0708_Name:
|
|
GTXSTRINGC "BLACKoSWAMPoPASSAGE"
|
|
L0709_Name:
|
|
GTXSTRINGC "HERDoLANDS"
|
|
L0710_Name:
|
|
GTXSTRINGC "DONaToSHOOToTHEoCROC"
|
|
L0711_Name:
|
|
GTXSTRINGC "SPACEoSTNoAPOCALYPSE"
|
|
L0712_Name:
|
|
GTXSTRINGC "HICKoLANDINGoeLZf"
|
|
L0713_Name:
|
|
L0714_Name:
|
|
L0715_Name:
|
|
GTXSTRINGC "FGB"
|
|
|
|
L0800_Name:
|
|
GTXSTRINGC "JACOBaSoOTHERoBRIDGE"
|
|
L0801_Name:
|
|
GTXSTRINGC "KIWIoROAD"
|
|
L0802_Name:
|
|
GTXSTRINGC "WINDYoPASS"
|
|
L0803_Name:
|
|
GTXSTRINGC "DJINNoLAKE"
|
|
L0804_Name:
|
|
GTXSTRINGC "ELDERoROCKoCANYON"
|
|
L0805_Name:
|
|
GTXSTRINGC "MONKEYoSURPRISE"
|
|
L0806_Name:
|
|
GTXSTRINGC "SCAREDIEoRUN"
|
|
L0807_Name:
|
|
GTXSTRINGC "SPIDERoMOORE"
|
|
L0808_Name:
|
|
GTXSTRINGC "CROUTONoWARoCAMP"
|
|
L0809_Name:
|
|
GTXSTRINGC "GRAPESHOT"
|
|
L0810_Name:
|
|
GTXSTRINGC "CROUTONoGRAVESoeLZf"
|
|
L0811_Name:
|
|
GTXSTRINGC "GOBLINoMODULE"
|
|
L0812_Name:
|
|
GTXSTRINGC "WARPoZONE"
|
|
L0813_Name:
|
|
L0814_Name:
|
|
L0815_Name:
|
|
GTXSTRINGC "FGB"
|
|
|
|
L0900_Name:
|
|
GTXSTRINGC "OUTPOSTgoDELTA"
|
|
L0901_Name:
|
|
GTXSTRINGC "CROUTONoSLAVEoCAMP"
|
|
L0902_Name:
|
|
GTXSTRINGC "WIZARDaSoPASSoeLZf"
|
|
L0903_Name:
|
|
GTXSTRINGC "CORNVILLE"
|
|
L0904_Name:
|
|
GTXSTRINGC "SPIDERWEEDoCANYON"
|
|
L0905_Name:
|
|
GTXSTRINGC "MONKEYoLANDINGoeLZf"
|
|
L0906_Name:
|
|
GTXSTRINGC "MOOREoGATE"
|
|
L0907_Name:
|
|
GTXSTRINGC "ARROWHEADoCITY"
|
|
L0908_Name:
|
|
GTXSTRINGC "CANNONBALL"
|
|
L0909_Name:
|
|
GTXSTRINGC "CROSSROADS"
|
|
L0910_Name:
|
|
GTXSTRINGC "WARZONE"
|
|
L0911_Name:
|
|
GTXSTRINGC "ARTILLERYoMODULE"
|
|
L0912_Name:
|
|
GTXSTRINGC "MONKEYoHOMEWORLD"
|
|
L0913_Name:
|
|
L0914_Name:
|
|
L0915_Name:
|
|
GTXSTRINGC "FGB"
|
|
|
|
L1000_Name:
|
|
GTXSTRINGC "GYROaSoBASEoCAMPeLZf"
|
|
L1001_Name:
|
|
GTXSTRINGC "CACTUSoPETE"
|
|
L1002_Name:
|
|
GTXSTRINGC "OUTPOSTgoDINGLE"
|
|
L1003_Name:
|
|
GTXSTRINGC "THEoBEACHoeLZf"
|
|
L1004_Name:
|
|
GTXSTRINGC "PUMPKINoPATCH"
|
|
L1005_Name:
|
|
GTXSTRINGC "AUTUMN"
|
|
L1006_Name:
|
|
GTXSTRINGC "OUTPOSTgoDESPAIR"
|
|
L1007_Name:
|
|
GTXSTRINGC "BUBoANDoJEB"
|
|
L1008_Name:
|
|
GTXSTRINGC "GRENADIERoGRAVEYARD"
|
|
L1009_Name:
|
|
GTXSTRINGC "BLACKoMAGIC"
|
|
L1010_Name:
|
|
GTXSTRINGC "GRENADIERoCASTLE"
|
|
L1011_Name:
|
|
GTXSTRINGC "DOCKINGoBAY"
|
|
L1012_Name:
|
|
GTXSTRINGC "MONKEYoLIBRARY"
|
|
L1013_Name:
|
|
L1014_Name:
|
|
L1015_Name:
|
|
GTXSTRINGC "FGB"
|
|
|
|
L1100_Name:
|
|
L1101_Name:
|
|
L1102_Name:
|
|
L1103_Name:
|
|
L1104_Name:
|
|
L1105_Name:
|
|
L1106_Name:
|
|
L1107_Name:
|
|
L1108_Name:
|
|
L1109_Name:
|
|
L1110_Name:
|
|
GTXSTRINGC "FGB"
|
|
L1111_Name:
|
|
GTXSTRINGC "HULKoMODULE"
|
|
L1112_Name:
|
|
GTXSTRINGC "DUKEaSoDISCO"
|
|
L1113_Name:
|
|
L1114_Name:
|
|
L1115_Name:
|
|
|
|
L1200_Name:
|
|
L1201_Name:
|
|
L1202_Name:
|
|
L1203_Name:
|
|
L1204_Name:
|
|
L1205_Name:
|
|
L1206_Name:
|
|
L1207_Name:
|
|
L1208_Name:
|
|
L1209_Name:
|
|
L1210_Name:
|
|
L1211_Name:
|
|
GTXSTRINGC "FGB"
|
|
L1212_Name:
|
|
GTXSTRINGC "THEoBESToLAIDoPLANS`"
|
|
L1213_Name:
|
|
L1214_Name:
|
|
L1215_Name:
|
|
|
|
L1300_Name:
|
|
GTXSTRINGC "BCSoAPPOMATTOX"
|
|
L1301_Name:
|
|
GTXSTRINGC "FGB"
|
|
L1302_Name:
|
|
L1303_Name:
|
|
L1304_Name:
|
|
L1305_Name:
|
|
L1306_Name:
|
|
L1307_Name:
|
|
L1308_Name:
|
|
L1309_Name:
|
|
L1310_Name:
|
|
L1311_Name:
|
|
GTXSTRINGC "FGB"
|
|
L1312_Name:
|
|
GTXSTRINGC "GOBLINoCOUNTRY"
|
|
L1313_Name:
|
|
L1314_Name:
|
|
L1315_Name:
|
|
|
|
L1400_Name:
|
|
L1401_Name:
|
|
L1402_Name:
|
|
L1403_Name:
|
|
L1404_Name:
|
|
L1405_Name:
|
|
L1406_Name:
|
|
L1407_Name:
|
|
L1408_Name:
|
|
L1409_Name:
|
|
L1410_Name:
|
|
L1411_Name:
|
|
GTXSTRINGC "FGB"
|
|
L1412_Name:
|
|
GTXSTRINGC "HIGHoNOON"
|
|
L1413_Name:
|
|
L1414_Name:
|
|
L1415_Name:
|
|
|
|
L1500_Name:
|
|
L1501_Name:
|
|
L1502_Name:
|
|
L1503_Name:
|
|
L1504_Name:
|
|
;THE KIDNAPPING!
|
|
DB ((20-STRLEN("THEoKIDNAPPING "))/2),STRLEN("THEoKIDNAPPING ")
|
|
COUNTER SET 1
|
|
REPT STRLEN("THEoKIDNAPPING")
|
|
DB ((STRSUB("THEoKIDNAPPING",COUNTER,1)+145) & $ff)
|
|
COUNTER SET COUNTER+1
|
|
ENDR
|
|
DB 237 ;!
|
|
L1505_Name:
|
|
GTXSTRINGC "FGB"
|
|
L1506_Name:
|
|
L1507_Name:
|
|
L1508_Name:
|
|
L1509_Name:
|
|
L1510_Name:
|
|
L1511_Name:
|
|
GTXSTRINGC "FGB"
|
|
L1512_Name:
|
|
GTXSTRINGC "GENERATORoCORE"
|
|
L1513_Name:
|
|
L1514_Name:
|
|
L1515_Name:
|
|
GTXSTRINGC "FGB"
|
|
|
|
heroNames:
|
|
DW ba_name,bs_name,haiku_name,flour_name,flower_name,grenade_name
|
|
|
|
ba_name:
|
|
GTXSTRING "BA"
|
|
bs_name:
|
|
GTXSTRING "BS"
|
|
haiku_name:
|
|
GTXSTRING "HAIKU"
|
|
flour_name:
|
|
GTXSTRING "CAPTAINoFLOUR"
|
|
flower_name:
|
|
GTXSTRING "LADYoFLOWER"
|
|
grenade_name:
|
|
GTXSTRING "KINGoGRENADE"
|
|
|
|
itemNames:
|
|
DW item000_Name,item001_Name,item002_Name,item003_Name,item004_Name
|
|
DW item005_Name,item006_Name,item007_Name,item008_Name,item009_Name
|
|
DW item010_Name,item011_Name,item012_Name,item013_Name,item014_Name
|
|
DW item015_Name,item016_Name,item017_Name,item018_Name,item019_Name
|
|
DW item020_Name,item021_Name,item022_Name
|
|
|
|
itemColors:
|
|
DB 1,1,0,0,0,0,0,1,3,1 ; 0- 9
|
|
DB 5,0,0,5,0,0,1,6,5,3 ;10-19
|
|
DB 2,4,5,0,0,0,0,0,0,0 ;20-29
|
|
;0=Grey, 1=Red, 2=Blue, 3=Green, 4=Purple, 5=Yellow, 6=Brown/Orange, 7=Fuscia
|
|
|
|
item000_Name:
|
|
ITEMSTRING 246,"SNAKEoBITEoKIT"
|
|
item001_Name:
|
|
ITEMSTRING 254,"SPOREoMASK"
|
|
item002_Name:
|
|
;ITEMSTRING 250,"CODE0400"
|
|
DB 0,0
|
|
item003_Name:
|
|
;ITEMSTRING 250,"CODE0900"
|
|
DB 0,0
|
|
item004_Name:
|
|
;ITEMSTRING 250,"CODE1002"
|
|
DB 0,0
|
|
item005_Name:
|
|
;ITEMSTRING 250,"CODE1006"
|
|
DB 0,0
|
|
item006_Name:
|
|
;ITEMSTRING 250,"CODE0410"
|
|
DB 0,0
|
|
item007_Name:
|
|
ITEMSTRING 249,"BAToJUICE"
|
|
item008_Name:
|
|
ITEMSTRING 248,"SPACEoMONEY"
|
|
item009_Name:
|
|
ITEMSTRING 247,"SPACEoSODA"
|
|
item010_Name:
|
|
ITEMSTRING 247,"HONEY"
|
|
item011_Name:
|
|
ITEMSTRING 254,"WRANGLINGoIRON"
|
|
item012_Name:
|
|
DB 0,0 ;BS Shoot Fast
|
|
item013_Name:
|
|
ITEMSTRING 253,"BUGoSPRAY"
|
|
item014_Name:
|
|
DB 0,0 ;BA High Impact Bullets
|
|
item015_Name:
|
|
;ITEMSTRING 250,"CODE0307"
|
|
DB 0,0
|
|
item016_Name:
|
|
ITEMSTRING 250,"ALPHAoCLEARANCE"
|
|
item017_Name:
|
|
ITEMSTRING 250,"BETAoCLEARANCE"
|
|
item018_Name:
|
|
ITEMSTRING 250,"GAMMAoCLEARANCE"
|
|
item019_Name:
|
|
ITEMSTRING 250,"DELTAoCLEARANCE"
|
|
item020_Name:
|
|
ITEMSTRING 250,"EPSILONoCLEARANCE"
|
|
item021_Name:
|
|
ITEMSTRING 250,"ZETAoCLEARANCE"
|
|
item022_Name:
|
|
ITEMSTRING 251,"APPOMATTOXoKEY"
|
|
|
|
|
|
SECTION "MiscSection",ROMX
|
|
;---------------------------------------------------------------------
|
|
; Routine: NewAppxLocation
|
|
; Arguments: None.
|
|
; Returns: none
|
|
; Alters: af
|
|
; Description:
|
|
;---------------------------------------------------------------------
|
|
NewAppxLocation:
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
ld a,[curLevelStateIndex]
|
|
ld b,a
|
|
ld a,[appomattoxMapIndex]
|
|
cp b
|
|
jr nz,.checkAppear
|
|
|
|
;remove appomattox from current map
|
|
call ReceiveByte ;new map location
|
|
ld [appomattoxMapIndex],a
|
|
ld a,190
|
|
ldio [jiggleDuration],a
|
|
|
|
call FindFirstLight ;remove appx
|
|
ld c,a ;invisible wall index
|
|
ld d,0
|
|
ld a,[mapPitch]
|
|
ld e,a
|
|
ld a,b ;light index
|
|
call .drawLightRowTop
|
|
call .drawLightRowMiddle
|
|
call .drawLightRowMiddle
|
|
call .drawLightRowMiddle
|
|
call .drawLightRowMiddle
|
|
call .drawLightRowBottom
|
|
|
|
ld hl,EngineSound
|
|
call PlaySound
|
|
|
|
;link up exit to $4040 (+0 +0)
|
|
ld hl,mapExitLinks+EXIT_U*2
|
|
ld a,$40
|
|
ld [hl+],a
|
|
ld [hl],a
|
|
|
|
jr .done
|
|
|
|
.checkAppear
|
|
call ReceiveByte ;new map location
|
|
ld [appomattoxMapIndex],a
|
|
cp b ;appearing on my map
|
|
jr nz,.checkInAppx
|
|
|
|
;place appomattox on current map
|
|
ld a,190
|
|
ldio [jiggleDuration],a
|
|
|
|
call AddAppomattox
|
|
ld hl,EngineSound
|
|
call PlaySound
|
|
|
|
jr .done
|
|
|
|
.checkInAppx
|
|
ld a,b
|
|
cp $0d
|
|
jr nz,.checkAtAppxControls
|
|
|
|
;I'm inside the Appomattox!
|
|
ld a,190
|
|
ldio [jiggleDuration],a
|
|
|
|
ld hl,EngineSound
|
|
call PlaySound
|
|
|
|
ld a,[appomattoxMapIndex]
|
|
or a
|
|
jr nz,.landing
|
|
|
|
.takingOff
|
|
;link down exit to $4040 (+0 +0)
|
|
ld hl,mapExitLinks+EXIT_D*2
|
|
ld a,$40
|
|
ld [hl+],a
|
|
ld [hl],a
|
|
jr .done
|
|
|
|
.landing
|
|
;convert map index to 16-bit BCD index
|
|
ld b,a
|
|
and %1111
|
|
call NumberToBCD
|
|
ld d,a
|
|
ld a,b
|
|
swap a
|
|
and %1111
|
|
call NumberToBCD
|
|
ld e,a
|
|
|
|
ld hl,mapExitLinks+EXIT_D*2
|
|
ld [hl],e
|
|
inc hl
|
|
ld [hl],d
|
|
jr .done
|
|
|
|
.checkAtAppxControls
|
|
;if I was at the controls when my buddy took off, kick me off
|
|
cp $1e
|
|
jr nz,.done
|
|
|
|
ld hl,EngineSound
|
|
call PlaySound
|
|
|
|
ld a,$ff
|
|
ld [levelVars+6],a ;VAR_SELSTAGE in $1401
|
|
|
|
.done
|
|
pop hl
|
|
pop de
|
|
pop bc
|
|
ret
|
|
|
|
.drawLightRowTop
|
|
push hl
|
|
ld [hl+],a
|
|
inc a
|
|
ld [hl+],a
|
|
inc a
|
|
ld [hl+],a
|
|
inc a
|
|
ld [hl+],a
|
|
sub 3
|
|
pop hl
|
|
add hl,de
|
|
ret
|
|
|
|
.drawLightRowMiddle
|
|
push hl
|
|
push af
|
|
sub b
|
|
add 3
|
|
and 3
|
|
add b
|
|
ld [hl+],a
|
|
;xor a
|
|
ld a,c
|
|
ld [hl+],a
|
|
ld [hl+],a
|
|
pop af
|
|
ld [hl+],a
|
|
sub b
|
|
add 3
|
|
and 3
|
|
add b
|
|
pop hl
|
|
add hl,de
|
|
ret
|
|
|
|
.drawLightRowBottom
|
|
push hl
|
|
add 3
|
|
ld [hl+],a
|
|
dec a
|
|
ld [hl+],a
|
|
dec a
|
|
ld [hl+],a
|
|
dec a
|
|
ld [hl+],a
|
|
dec a
|
|
pop hl
|
|
add hl,de
|
|
ret
|
|
|
|
FindFirstLight:
|
|
;leaves hl at mem location of first landing light
|
|
;b = first light index, c=first appx index
|
|
;a = invisible wall index
|
|
;must exist or infinite loop!
|
|
ld bc,classInvisibleWallBG
|
|
call FindClassIndex
|
|
|
|
push af
|
|
ld bc,classLandingLightsBG
|
|
call FindClassIndex
|
|
ld b,a ;index to look for
|
|
|
|
push bc
|
|
ld bc,classAppomattoxBG
|
|
call FindClassIndex
|
|
pop bc
|
|
ld c,a
|
|
|
|
ld a,MAPBANK
|
|
ld [$ff70],a
|
|
|
|
;setup de with first out-of-bounds index
|
|
;ld a,[mapTotalSize]
|
|
;ld e,a
|
|
;ld a,[mapTotalSize+1]
|
|
;ld d,a
|
|
|
|
ld hl,map
|
|
.lookForFirstLight
|
|
ld a,[hl+]
|
|
cp b
|
|
jr nz,.lookForFirstLight
|
|
|
|
dec hl
|
|
pop af
|
|
|
|
ret
|
|
|
|
AddAppomattoxIfPresent::
|
|
push bc
|
|
ld a,[curLevelStateIndex]
|
|
ld b,a
|
|
ld a,[appomattoxMapIndex]
|
|
cp b
|
|
jr nz,.done
|
|
|
|
call AddAppomattox
|
|
|
|
.done
|
|
pop bc
|
|
ret
|
|
|
|
EngineSound:
|
|
DB 4,$00,$d7,$a9,$80
|
|
|
|
AddAppomattox:
|
|
push bc
|
|
push de
|
|
push hl
|
|
|
|
call FindFirstLight ;draw appx
|
|
ld d,0
|
|
ld a,[mapPitch]
|
|
ld e,a
|
|
ld a,c ;appx index
|
|
call .drawAppxMiddle
|
|
call .drawAppxMiddle
|
|
call .drawAppxMiddle
|
|
call .drawAppxRow
|
|
call .drawAppxRow
|
|
call .drawAppxRow
|
|
|
|
;link exit up to $1300 (appomattox interior)
|
|
ld hl,mapExitLinks+EXIT_U*2
|
|
xor a
|
|
ld [hl+],a
|
|
ld [hl],$13
|
|
|
|
pop hl
|
|
pop de
|
|
pop bc
|
|
ret
|
|
|
|
.drawAppxMiddle
|
|
push hl
|
|
inc hl
|
|
inc a
|
|
ld [hl+],a
|
|
inc a
|
|
ld [hl],a
|
|
add 2
|
|
pop hl
|
|
add hl,de
|
|
ret
|
|
|
|
.drawAppxRow
|
|
push hl
|
|
ld [hl+],a
|
|
inc a
|
|
ld [hl+],a
|
|
inc a
|
|
ld [hl+],a
|
|
inc a
|
|
ld [hl],a
|
|
inc a
|
|
pop hl
|
|
add hl,de
|
|
|
|
ret
|
|
|
|
;WARNING not in HOME section
|
|
|
|
;---------------------------------------------------------------------
|
|
; Variables
|
|
;---------------------------------------------------------------------
|
|
SECTION "UserVarsHRAM",HRAM
|
|
vblankFlag: DS 1 ;0=no interrupt, 1=vblank
|
|
vblankTemp: DS 1
|
|
vblankTimer:: DS 1
|
|
vblanksPerUpdate: DS 1
|
|
backBufferReady:: DS 1
|
|
backBufferDestHighByte:: DS 1
|
|
paletteBufferReady:: DS 1
|
|
dmaLoad:: DS 1 ;1=load bank 0, 2=load bank 1
|
|
randomLoc:: DS 1 ;ff92
|
|
jiggleDuration:: DS 1 ;ff93
|
|
temp:: DS 1 ;ff94
|
|
drawMapTemp:: DS 2 ;ff95
|
|
hblankFlag:: DS 1 ;ff97 :0 top/bottom, :1 show, :2 wave
|
|
hblankWinOn:: DS 1 ;ff98
|
|
hblankWinOff:: DS 1 ;ff99
|
|
firstMonster:: DS 1 ;ff9a
|
|
curROMBank:: DS 1 ;ff9b
|
|
updateTimer:: DS 1 ;ff9c
|
|
curObjWidthHeight:: DS 1 ;7:4 width, 3:0 height
|
|
scrollSpeed:: DS 1 ;7:4 fast speed (0,2,4,8) 3:0 slow speed
|
|
mapState:: DS 2 ;ff9f
|
|
transmitACK:: DS 1 ;ffa1
|
|
musicEnabled:: DS 1 ;ffa2
|
|
baseHorizontalOffset: DS 1 ;ffa3
|
|
|
|
samplePlaying: DS 1 ;ffa4
|
|
sampleBank: DS 1 ;ffa5
|
|
sampleAddress: DS 2 ;ffa6
|
|
sampleMask: DS 1 ;ffa8
|
|
jiggleType:: DS 1 ;ffa9
|
|
asyncRandLoc:: DS 1 ;ffaa
|
|
|
|
|
|
IF 0
|
|
;clear memory to ff
|
|
ld hl,$c000
|
|
ld bc,$0f00
|
|
;ld d,$ff
|
|
ld d,0
|
|
xor a
|
|
call MemSet
|
|
|
|
ld hl,$d000
|
|
ld bc,$1000
|
|
ld a,1
|
|
call MemSet
|
|
|
|
ld hl,$d000
|
|
ld bc,$1000
|
|
ld a,2
|
|
call MemSet
|
|
|
|
ld hl,$d000
|
|
ld bc,$1000
|
|
ld a,3
|
|
call MemSet
|
|
|
|
ld hl,$d000
|
|
ld bc,$1000
|
|
ld a,4
|
|
call MemSet
|
|
|
|
ld hl,$d000
|
|
ld bc,$1000
|
|
ld a,5
|
|
call MemSet
|
|
|
|
ld hl,$d000
|
|
ld bc,$1000
|
|
ld a,6
|
|
call MemSet
|
|
|
|
ld hl,$d000
|
|
ld bc,$0020
|
|
ld a,7
|
|
call MemSet
|
|
|
|
ld d,0
|
|
ld hl,$d020
|
|
ld bc,$0fe0
|
|
ld a,7
|
|
call MemSet
|
|
ENDC
|
|
|