 81a239325d
			
		
	
	81a239325d
	
	
	
		
			
			Adds Link's Awakening: DX. Fully imports and forks LADXR, with permission - https://github.com/daid/LADXR
		
			
				
	
	
		
			208 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			208 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from ..assembler import ASM
 | |
| from ..roomEditor import RoomEditor
 | |
| 
 | |
| 
 | |
| def fixBowwow(rom, everywhere=False):
 | |
|     ### BowWow patches
 | |
|     rom.patch(0x03, 0x1E0E, ASM("ld [$DB56], a"), "", fill_nop=True)  # Do not mark BowWow as kidnapped after we complete dungeon 1.
 | |
|     rom.patch(0x15, 0x06B6, ASM("ld a, [$DB56]\ncp $80"), ASM("xor a"), fill_nop=True)  # always load the moblin boss
 | |
|     rom.patch(0x03, 0x182D, ASM("ld a, [$DB56]\ncp $80"), ASM("ld a, [$DAE2]\nand $10"))  # load the cave moblins if the chest is not opened
 | |
|     rom.patch(0x07, 0x3947, ASM("ld a, [$DB56]\ncp $80"), ASM("ld a, [$DAE2]\nand $10"))  # load the cave moblin with sword if the chest is not opened
 | |
| 
 | |
|     # Modify the moblin cave to contain a chest at the end, which contains bowwow
 | |
|     re = RoomEditor(rom, 0x2E2)
 | |
|     re.removeEntities(0x6D)
 | |
|     re.changeObject(8, 3, 0xA0)
 | |
|     re.store(rom)
 | |
|     # Place bowwow in the chest table
 | |
|     rom.banks[0x14][0x560 + 0x2E2] = 0x81
 | |
| 
 | |
|     # Patch bowwow follower sprite to be used from 2nd vram bank
 | |
|     rom.patch(0x05, 0x001C,
 | |
|         b"40034023"
 | |
|         b"42034223"
 | |
|         b"44034603"
 | |
|         b"48034A03"
 | |
|         b"46234423"
 | |
|         b"4A234823"
 | |
|         b"4C034C23",
 | |
|         b"500B502B"
 | |
|         b"520B522B"
 | |
|         b"540B560B"
 | |
|         b"580B5A0B"
 | |
|         b"562B542B"
 | |
|         b"5A2B582B"
 | |
|         b"5C0B5C2B")
 | |
|     # Patch to use the chain sprite from second vram bank (however, the chain bugs out various things)
 | |
|     rom.patch(0x05, 0x0282,
 | |
|         ASM("ld a, $4E\njr nz, $02\nld a, $7E\nld [de], a\ninc de\nld a, $00"),
 | |
|         ASM("ld a, $5E\nld [de], a\ninc de\nld a, $08"), fill_nop=True)
 | |
|     # Never load the bowwow tiles in the first VRAM bank, as we do not need them.
 | |
|     rom.patch(0x00, 0x2EB0, ASM("ld a, [$DB56]\ncp $01\nld a, $A4\njr z, $18"), "", fill_nop=True)
 | |
| 
 | |
|     # Patch the location where bowwow stores chain X/Y positions so it does not conflict with a lot of other things
 | |
|     rom.patch(0x05, 0x00BE, ASM("ld hl, $D100"), ASM("ld hl, $D180"))
 | |
|     rom.patch(0x05, 0x0275, ASM("ld hl, $D100"), ASM("ld hl, $D180"))
 | |
|     rom.patch(0x05, 0x03AD, ASM("ld [$D100], a"), ASM("ld [$D180], a"))
 | |
|     rom.patch(0x05, 0x03BD, ASM("ld de, $D100"), ASM("ld de, $D180"))
 | |
|     rom.patch(0x05, 0x049F, ASM("ld hl, $D100"), ASM("ld hl, $D180"))
 | |
|     rom.patch(0x05, 0x04C2, ASM("ld a, [$D100]"), ASM("ld a, [$D180]"))
 | |
|     rom.patch(0x05, 0x03C0, ASM("ld hl, $D101"), ASM("ld hl, $D181"))
 | |
|     rom.patch(0x05, 0x0418, ASM("ld [$D106], a"), ASM("ld [$D186], a"))
 | |
|     rom.patch(0x05, 0x0423, ASM("ld de, $D106"), ASM("ld de, $D186"))
 | |
|     rom.patch(0x05, 0x0426, ASM("ld hl, $D105"), ASM("ld hl, $D185"))
 | |
| 
 | |
|     rom.patch(0x19, 0x3A4E, ASM("ld hl, $D100"), ASM("ld hl, $D180"))
 | |
|     rom.patch(0x19, 0x3A5A, ASM("ld hl, $D110"), ASM("ld hl, $D190"))
 | |
| 
 | |
|     rom.patch(0x05, 0x00D9, ASM("ld hl, $D110"), ASM("ld hl, $D190"))
 | |
|     rom.patch(0x05, 0x026E, ASM("ld hl, $D110"), ASM("ld hl, $D190"))
 | |
|     rom.patch(0x05, 0x03BA, ASM("ld [$D110], a"), ASM("ld [$D190], a"))
 | |
|     rom.patch(0x05, 0x03DD, ASM("ld de, $D110"), ASM("ld de, $D190"))
 | |
|     rom.patch(0x05, 0x0480, ASM("ld hl, $D110"), ASM("ld hl, $D190"))
 | |
|     rom.patch(0x05, 0x04B5, ASM("ld a, [$D110]"), ASM("ld a, [$D190]"))
 | |
|     rom.patch(0x05, 0x03E0, ASM("ld hl, $D111"), ASM("ld hl, $D191"))
 | |
|     rom.patch(0x05, 0x0420, ASM("ld [$D116], a"), ASM("ld [$D196], a"))
 | |
|     rom.patch(0x05, 0x044d, ASM("ld de, $D116"), ASM("ld de, $D196"))
 | |
|     rom.patch(0x05, 0x0450, ASM("ld hl, $D115"), ASM("ld hl, $D195"))
 | |
| 
 | |
|     rom.patch(0x05, 0x0039, ASM("ld [$D154], a"), "", fill_nop=True)  # normally this stores the index to bowwow, for the kiki fight
 | |
|     rom.patch(0x05, 0x013C, ASM("ld [$D150], a"), ASM("ld [$D197], a"))
 | |
|     rom.patch(0x05, 0x0144, ASM("ld [$D151], a"), ASM("ld [$D198], a"))
 | |
|     rom.patch(0x05, 0x02F9, ASM("ld [$D152], a"), ASM("ld [$D199], a"))
 | |
|     rom.patch(0x05, 0x0335, ASM("ld a, [$D152]"), ASM("ld a, [$D199]"))
 | |
|     rom.patch(0x05, 0x0485, ASM("ld a, [$D151]"), ASM("ld a, [$D198]"))
 | |
|     rom.patch(0x05, 0x04A4, ASM("ld a, [$D150]"), ASM("ld a, [$D197]"))
 | |
| 
 | |
|     # Patch the bowwow create code to call our custom check of we are in swamp function.
 | |
|     if everywhere:
 | |
|         # Load followers in dungeons, caves, etc
 | |
|         rom.patch(0x01, 0x1FC1, ASM("ret z"), "", fill_nop=True)
 | |
|         rom.patch(0x01, 0x1FC4, ASM("ret z"), "", fill_nop=True)
 | |
|         rom.patch(0x01, 0x1FC7, ASM("ret z"), "", fill_nop=True)
 | |
|         rom.patch(0x01, 0x1FCA, ASM("ret c"), "", fill_nop=True)  # dungeon
 | |
|         # rom.patch(0x01, 0x1FBC, ASM("ret nz"), "", fill_nop=True)  # sidescroller: TOFIX this breaks fishing minigame reward
 | |
|     else:
 | |
|         # Patch the bowwow create code to call our custom check of we are in swamp function.
 | |
|         rom.patch(0x01, 0x211F, ASM("ldh a, [$F6]\ncp $A7\nret z\nld a, [$DB56]\ncp $01\njr nz, $36"), ASM("""
 | |
|             ld a, $07
 | |
|             rst 8
 | |
|             ld  a, e
 | |
|             and a
 | |
|             ret z
 | |
|         """), fill_nop=True)
 | |
|         # Patch bowwow to not stay around when we move from map to map
 | |
|         rom.patch(0x05, 0x0049, 0x0054, ASM("""
 | |
|             cp   [hl]
 | |
|             jr   z, Continue
 | |
|             ld   hl, $C280
 | |
|             add  hl, bc
 | |
|             ld   [hl], b
 | |
|             ret
 | |
| Continue:
 | |
|         """), fill_nop=True)
 | |
| 
 | |
|     # Patch madam meow meow to not take bowwow
 | |
|     rom.patch(0x06, 0x1BD7, ASM("ld a, [$DB66]\nand $02"), ASM("ld a, $00\nand $02"), fill_nop=True)
 | |
| 
 | |
|     # Patch kiki not to react to bowwow, as bowwow is not with link at this map
 | |
|     rom.patch(0x07, 0x18A8, ASM("ld a, [$DB56]\ncp $01"), ASM("ld a, $00\ncp $01"), fill_nop=True)
 | |
| 
 | |
|     # Patch the color dungeon entrance not to check for bowwow
 | |
|     rom.patch(0x02, 0x340D, ASM("ld hl, $DB56\nor [hl]"), "", fill_nop=True)
 | |
| 
 | |
|     # Patch richard to ignore bowwow
 | |
|     rom.patch(0x06, 0x006C, ASM("ld a, [$DB56]"), ASM("xor a"), fill_nop=True)
 | |
| 
 | |
|     # Patch to modify how bowwow eats enemies, normally it just unloads them, but we call our handler in bank 3E
 | |
|     rom.patch(0x05, 0x03A0, 0x03A8, ASM("""
 | |
|         push bc
 | |
|         ld   b, d
 | |
|         ld   c, e
 | |
|         ld   a, $08
 | |
|         rst  8
 | |
|         pop  bc
 | |
|         ret
 | |
|     """), fill_nop=True)
 | |
|     rom.patch(0x05, 0x0387, ASM("ld a, $03\nldh [$F2], a"), "", fill_nop=True)  # remove the default chomp sfx
 | |
| 
 | |
|     # Various enemies
 | |
|     rom.banks[0x14][0x1218 + 0xC5] = 0x01  # Urchin
 | |
|     rom.banks[0x14][0x1218 + 0x93] = 0x01  # MadBomber
 | |
|     rom.banks[0x14][0x1218 + 0x51] = 0x01  # Swinging ball&chain golden leaf enemy
 | |
|     rom.banks[0x14][0x1218 + 0xF2] = 0x01  # Color dungeon flying hopper
 | |
|     rom.banks[0x14][0x1218 + 0xF3] = 0x01  # Color dungeon hopper
 | |
|     rom.banks[0x14][0x1218 + 0xE9] = 0x01  # Color dungeon shell
 | |
|     rom.banks[0x14][0x1218 + 0xEA] = 0x01  # Color dungeon shell
 | |
|     rom.banks[0x14][0x1218 + 0xEB] = 0x01  # Color dungeon shell
 | |
|     rom.banks[0x14][0x1218 + 0xEC] = 0x01  # Color dungeon thing
 | |
|     rom.banks[0x14][0x1218 + 0xED] = 0x01  # Color dungeon thing
 | |
|     rom.banks[0x14][0x1218 + 0xEE] = 0x01  # Color dungeon thing
 | |
|     rom.banks[0x14][0x1218 + 0x87] = 0x01  # Lanmola (for D4 key)
 | |
|     rom.banks[0x14][0x1218 + 0x88] = 0x01  # Armos knight (for D6 key)
 | |
|     rom.banks[0x14][0x1218 + 0x16] = 0x01  # Spark
 | |
|     rom.banks[0x14][0x1218 + 0x17] = 0x01  # Spark
 | |
|     rom.banks[0x14][0x1218 + 0x2C] = 0x01  # Spiked beetle
 | |
|     rom.banks[0x14][0x1218 + 0x90] = 0x01  # Three of a kind (screw these guys)
 | |
|     rom.banks[0x14][0x1218 + 0x18] = 0x01  # Pols voice
 | |
|     rom.banks[0x14][0x1218 + 0x50] = 0x01  # Boo buddy
 | |
|     rom.banks[0x14][0x1218 + 0xA2] = 0x01  # Pirana plant
 | |
|     rom.banks[0x14][0x1218 + 0x52] = 0x01  # Tractor device
 | |
|     rom.banks[0x14][0x1218 + 0x53] = 0x01  # Tractor device (D3)
 | |
|     rom.banks[0x14][0x1218 + 0x55] = 0x01  # Bounding bombite
 | |
|     rom.banks[0x14][0x1218 + 0x56] = 0x01  # Timer bombite
 | |
|     rom.banks[0x14][0x1218 + 0x57] = 0x01  # Pairod
 | |
|     rom.banks[0x14][0x1218 + 0x15] = 0x01  # Antifairy
 | |
|     rom.banks[0x14][0x1218 + 0xA0] = 0x01  # Peahat
 | |
|     rom.banks[0x14][0x1218 + 0x9C] = 0x01  # Star
 | |
|     rom.banks[0x14][0x1218 + 0xA1] = 0x01  # Snake
 | |
|     rom.banks[0x14][0x1218 + 0xBD] = 0x01  # Vire
 | |
|     rom.banks[0x14][0x1218 + 0xE4] = 0x01  # Moblin boss
 | |
| 
 | |
|     # Bosses
 | |
|     rom.banks[0x14][0x1218 + 0x59] = 0x01  # Moldorm
 | |
|     rom.banks[0x14][0x1218 + 0x5C] = 0x01  # Genie
 | |
|     rom.banks[0x14][0x1218 + 0x5B] = 0x01  # Slime Eye
 | |
|     rom.patch(0x04, 0x0AC4, ASM("ld [hl], $28"), ASM("ld [hl], $FF"))  # give more time before slimeeye unsplits
 | |
|     rom.patch(0x04, 0x0B05, ASM("ld [hl], $50"), ASM("ld [hl], $FF"))  # give more time before slimeeye unsplits
 | |
|     rom.banks[0x14][0x1218 + 0x65] = 0x01  # Angler fish
 | |
|     rom.banks[0x14][0x1218 + 0x5D] = 0x01  # Slime eel
 | |
|     rom.banks[0x14][0x1218 + 0x5A] = 0x01  # Facade
 | |
|     rom.banks[0x14][0x1218 + 0x63] = 0x01  # Eagle
 | |
|     rom.banks[0x14][0x1218 + 0x62] = 0x01  # Hot head
 | |
|     rom.banks[0x14][0x1218 + 0xF9] = 0x01  # Hardhit beetle
 | |
|     rom.banks[0x14][0x1218 + 0xE6] = 0x01  # Nightmare
 | |
| 
 | |
|     # Minibosses
 | |
|     rom.banks[0x14][0x1218 + 0x81] = 0x01  # Rolling bones
 | |
|     rom.banks[0x14][0x1218 + 0x89] = 0x01  # Hinox
 | |
|     rom.banks[0x14][0x1218 + 0x8E] = 0x01  # Cue ball
 | |
|     rom.banks[0x14][0x1218 + 0x5E] = 0x01  # Gnoma
 | |
|     rom.banks[0x14][0x1218 + 0x5F] = 0x01  # Master stalfos
 | |
|     rom.banks[0x14][0x1218 + 0x92] = 0x01  # Smasher
 | |
|     rom.banks[0x14][0x1218 + 0xBC] = 0x01  # Grim creeper
 | |
|     rom.banks[0x14][0x1218 + 0xBE] = 0x01  # Blaino
 | |
|     rom.banks[0x14][0x1218 + 0xF8] = 0x01  # Giant buzz blob
 | |
|     rom.banks[0x14][0x1218 + 0xF4] = 0x01  # Avalaunch
 | |
| 
 | |
|     # NPCs
 | |
|     rom.banks[0x14][0x1218 + 0x6F] = 0x01  # Dog
 | |
|     rom.banks[0x14][0x1218 + 0x6E] = 0x01  # Butterfly
 | |
|     rom.banks[0x14][0x1218 + 0x6C] = 0x01  # Cucco
 | |
|     rom.banks[0x14][0x1218 + 0x70] = 0x01  # Kid
 | |
|     rom.banks[0x14][0x1218 + 0x71] = 0x01  # Kid
 | |
|     rom.banks[0x14][0x1218 + 0x72] = 0x01  # Kid
 | |
|     rom.banks[0x14][0x1218 + 0x73] = 0x01  # Kid
 | |
|     rom.banks[0x14][0x1218 + 0xD0] = 0x01  # Animal
 | |
|     rom.banks[0x14][0x1218 + 0xD1] = 0x01  # Animal
 | |
|     rom.banks[0x14][0x1218 + 0xD2] = 0x01  # Animal
 | |
|     rom.banks[0x14][0x1218 + 0xD3] = 0x01  # Animal
 | |
| 
 | |
| 
 | |
| def bowwowMapPatches(rom):
 | |
|     # Remove all the cystal things that can only be destroyed with a sword.
 | |
|     for n in range(0x100, 0x2FF):
 | |
|         re = RoomEditor(rom, n)
 | |
|         re.objects = list(filter(lambda obj: obj.type_id != 0xDD, re.objects))
 | |
|         re.store(rom)
 |