mirror of
				https://github.com/MarioSpore/Grinch-AP.git
				synced 2025-10-21 20:21:32 -06:00 
			
		
		
		
	 043ba418ec
			
		
	
	043ba418ec
	
	
	
		
			
			* - SM now displays message when getting an item outside for someone else (fills ROM item table) This is dependant on modifications done to sm_randomizer_rom project * First working MultiWorld SM * some missing things: - player name inject in ROM and get in client - end game get from ROM in client - send self item to server - add player names table in ROM * replaced CollectionState inheritance from SMBoolManager with a composition of an array of it (required to generation more than one SM world, which is still fails but is better) * - reenabled balancing * post rebase fixes * updated SmClient.py * + added VariaRandomizer LICENSE * + added sm_randomizer_rom project (which builds sm.ips) * Moved VariaRandomizer and sm_randomizer_rom projects inside worlds/sm and done some cleaning * properly revert change made to CollectionState and more cleaning * Fixed multiworld support patch not working with VariaRandomizer's * missing file commit * Fixed syntax error in unused code to satisfy Linter * Revert "Fixed multiworld support patch not working with VariaRandomizer's" This reverts commit fb3ca18528bb331995e3d3051648c8f84d04c08b. * many fixes and improovement - fixed seeded generation - fixed broken logic when more than one SM world - added missing rules for inter-area transitions - added basic patch presence for logic - added DoorManager init call to reflect present patches for logic - moved CollectionState addition out of BaseClasses into SM world - added condition to apply progitempool presorting only if SM world is present - set Bosses item id to None to prevent them going into multidata - now use get_game_players * first working (most of the time) progression generation for SM using VariaRandomizer's rules, items, locations and accessPoint (as regions) * first working single-world randomized SM rom patches * - SM now displays message when getting an item outside for someone else (fills ROM item table) This is dependant on modifications done to sm_randomizer_rom project * First working MultiWorld SM * some missing things: - player name inject in ROM and get in client - end game get from ROM in client - send self item to server - add player names table in ROM * replaced CollectionState inheritance from SMBoolManager with a composition of an array of it (required to generation more than one SM world, which is still fails but is better) * - reenabled balancing * post rebase fixes * updated SmClient.py * + added VariaRandomizer LICENSE * + added sm_randomizer_rom project (which builds sm.ips) * Moved VariaRandomizer and sm_randomizer_rom projects inside worlds/sm and done some cleaning * properly revert change made to CollectionState and more cleaning * Fixed multiworld support patch not working with VariaRandomizer's * missing file commit * Fixed syntax error in unused code to satisfy Linter * Revert "Fixed multiworld support patch not working with VariaRandomizer's" This reverts commit fb3ca18528bb331995e3d3051648c8f84d04c08b. * many fixes and improovement - fixed seeded generation - fixed broken logic when more than one SM world - added missing rules for inter-area transitions - added basic patch presence for logic - added DoorManager init call to reflect present patches for logic - moved CollectionState addition out of BaseClasses into SM world - added condition to apply progitempool presorting only if SM world is present - set Bosses item id to None to prevent them going into multidata - now use get_game_players * Fixed multiworld support patch not working with VariaRandomizer's Added stage_fill_hook to set morph first in progitempool Added back VariaRandomizer's standard patches * + added missing files from variaRandomizer project * + added missing variaRandomizer files (custom sprites) + started integrating VariaRandomizer options (WIP) * Some fixes for player and server name display - fixed player name of 16 characters reading too far in SM client - fixed 12 bytes SM player name limit (now 16) - fixed server name not being displayed in SM when using server cheat ( now displays RECEIVED FROM ARCHIPELAGO) - request: temporarly changed default seed names displayed in SM main menu to OWTCH * Fixed Goal completion not triggering in smClient * integrated VariaRandomizer's options into AP (WIP) - startAP is working - door rando is working - skillset is working * - fixed itemsounds.ips crash by always including nofanfare.ips into multiworld.ips (itemsounds is now always applied and "itemsounds" preset must always be "off") * skillset are now instanced per player instead of being a singleton class * RomPatches are now instanced per player instead of being a singleton class * DoorManager is now instanced per player instead of being a singleton class * - fixed the last bugs that prevented generation of >1 SM world * fixed crash when no skillset preset is specified in randoPreset (default to "casual") * maxDifficulty support and itemsounds removal - added support for maxDifficulty - removed itemsounds patch as its always applied from multiworld patch for now * Fixed bad merge * Post merge adaptation * fixed player name length fix that got lost with the merge * fixed generation with other game type than SM * added default randoPreset json for SM in playerSettings.yaml * fixed broken SM client following merge * beautified json skillset presets * Fixed ArchipelagoSmClient not building * Fixed conflict between mutliworld patch and beam_doors_plms patch - doorsColorsRando now working * SM generation now outputs APBP - Fixed paths for patches and presets when frozen * added missing file and fixed multithreading issue * temporarily set data_version = 0 * more work - added support for AP starting items - fixed client crash with gamemode being None - patch.py "compatible_version" is now 3 * commited missing asm files fixed start item reserve breaking game (was using bad write offset when patching) * Nothing item are now handled game-side. the game will now skip displaying a message box for received Nothing item (but the client will still receive it). fixed crash in SMClient when loosing connection to SNI * fixed No Energy Item missing its ID fixed Plando * merge post fixes * fixed start item Grapple, XRay and Reserve HUD, as well as graphic beams (except ice palette color) * fixed freeze in blue brinstar caused by Varia's custom PLM not being filled with proper Multiworld PLM address (altLocsAddresses) * fixed start item x-ray HUD display * Fixed start items being sent by the server (is all handled in ROM) Start items are now not removed from itempool anymore Nothing Item is now local_items so no player will ever pickup Nothing. Doing so reduces contribution of this world to the Multiworld the more Nothing there is though. Fixed crash (and possibly passing but broken) at generation where the static list of IPSPatches used by all SM worlds was being modified * fixed settings that could be applied to any SM players * fixed auth to server only using player name (now does as ALTTP to authenticate) * - fixed End Credits broken text * added non SM item name display * added all supported SM options in playerSettings.yaml * fixed locations needing a list of parent regions (now generate a region for each location with one-way exits to each (previously) parent region did some cleaning (mainly reverts on unnecessary core classes * minor setting fixes and tweaks - merged Area and lightArea settings - made missileQty, superQty and powerBombQty use value from 10 to 90 and divide value by float(10) when generating - fixed inverted layoutPatch setting * added option start_inventory_removes_from_pool fixed option names formatting fixed lint errors small code and repo cleanup * Hopefully fixed ROR2 that could not send any items * - fixed missing required change to ROR2 * fixed 0 hp when respawning without having ever saved (start items were not updating the save checksum) * fixed typo with doors_colors_rando * fixed checksum * added custom sprites for off-world items (progression or not) the original AP sprite was made with PierRoulette's SM Item Sprite Utility by ijwu * - added missing change following upstream merge - changed patch filename extension from apbp to apm3 so patch can be used with the new client * added morph placement options: early means local and sphere 1 * fixed failing unit tests * - fixed broken custom_preset options * - big cleanup to remove unnecessary or unsupported features * - more cleanup * - moved sm_randomizer_rom and all always applied patches into an external project that outputs basepatch.ips - small cleanup * - added comment to refer to project for generating basepatch.ips (https://github.com/lordlou/SMBasepatch) * fixed g4_skip patch that can be not applied if hud is enabled * - fixed off world sprite that can have broken graphics (restricted to use only first 2 palette) * - updated basepatch to reflect g4_skip removal - moved more asm files to SMBasepatch project * - tourian grey doors at baby metroid are now always flashing (allowing to go back if needed) * fixed wrong path if using built as exe * - cleaned exposed maxDifficulty options - removed always enabled Knows * Merged LttPClient and SMClient into SNIClient * added varia_custom Preset Option that fetch a preset (read from a new varia_custom_preset Option) from varia's web service * small doc precision * - added death_link support - fixed broken Goal Completion - post merge fix * - removed now useless presets * - fixed bad internal mapping with maxDiff - increases maxDiff if only Bosses is preventing beating the game * - added support for lowercase custom preset sections (knows, settings and controller) - fixed controller settings not applying to ROM * - fixed death loop when dying with Door rando, bomb or speed booster as starting items - varia's backup save should now be usable (automatically enabled when doing door rando) * -added docstring for generated yaml * fixed bad merge * fixed broken infinity max difficulty * commented debug prints * adjusted credits to mark progression speed and difficulty as Non Available * added support for more than 255 players (will print Archipelago for higher player number) * fixed missing cleanup * added support for 65535 different player names in ROM * fixed generations failing when only bosses are unreachable * - replaced setting maxDiff to infinity with a bool only affecting boss logics if only bosses are left to finish * fixed failling generations when using 'fun' settings Accessibility checks are forced to 'items' if restricted locations are used by VARIA following usage of 'fun' settings * fixed debug logger * removed unsupported "suits_restriction" option * fixed generations failing when only bosses are unreachable (using a less intrusive approach for AP) * - fixed deathlink emptying reserves - added death_link_survive option that lets player survive when receiving a deathlink if the have non-empty reserves * - merged death_link and death_link_survive options * fixed death_link * added a fallback default starting location instead of failing generation if an invalid one was chosen * added Nothing and NoEnergy as hint blacklist added missing NoEnergy as local items and removed it from progression * SM Varia can now generate without ROM * removed stage_assert_generate
		
			
				
	
	
		
			907 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			907 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from __future__ import annotations
 | |
| 
 | |
| import base64
 | |
| import copy
 | |
| import logging
 | |
| import threading
 | |
| import typing
 | |
| from typing import Any, Dict, Iterable, List, Set, TextIO, TypedDict
 | |
| 
 | |
| import settings
 | |
| from BaseClasses import CollectionState, Entrance, Item, ItemClassification, Location, MultiWorld, Region, Tutorial
 | |
| from Options import Accessibility
 | |
| from worlds.AutoWorld import AutoLogicRegister, WebWorld, World
 | |
| from worlds.generic.Rules import add_rule, set_rule
 | |
| 
 | |
| logger = logging.getLogger("Super Metroid")
 | |
| 
 | |
| from .Options import SMOptions
 | |
| from .Client import SMSNIClient
 | |
| from .Rom import SM_ROM_MAX_PLAYERID, SM_ROM_PLAYERDATA_COUNT, SMProcedurePatch, get_sm_symbols
 | |
| import Utils
 | |
| 
 | |
| from .variaRandomizer.logic.smboolmanager import SMBoolManager
 | |
| from .variaRandomizer.graph.vanilla.graph_locations import locationsDict
 | |
| from .variaRandomizer.graph.graph_utils import getAccessPoint
 | |
| from .variaRandomizer.rando.ItemLocContainer import ItemLocation, ItemLocContainer
 | |
| from .variaRandomizer.rando.Items import ItemManager
 | |
| from .variaRandomizer.rando.RandoServices import ComebackCheckType
 | |
| from .variaRandomizer.utils.parameters import *
 | |
| from .variaRandomizer.utils.utils import openFile
 | |
| from .variaRandomizer.logic.logic import Logic
 | |
| from .variaRandomizer.randomizer import VariaRandomizer
 | |
| from .variaRandomizer.utils.doorsmanager import DoorsManager
 | |
| from .variaRandomizer.rom.rom_patches import RomPatches
 | |
| from .variaRandomizer.graph.graph_utils import GraphUtils
 | |
| 
 | |
| 
 | |
| class SMSettings(settings.Group):
 | |
|     class RomFile(settings.SNESRomPath):
 | |
|         """File name of the v1.0 J rom"""
 | |
|         description = "Super Metroid (JU) ROM"
 | |
|         copy_to = "Super Metroid (JU).sfc"
 | |
|         md5s = [SMProcedurePatch.hash]
 | |
| 
 | |
|     rom_file: RomFile = RomFile(RomFile.copy_to)
 | |
| 
 | |
| 
 | |
| class SMCollectionState(metaclass=AutoLogicRegister):
 | |
|     def init_mixin(self, parent: MultiWorld):
 | |
|         
 | |
|         # for unit tests where MultiWorld is instantiated before worlds
 | |
|         if hasattr(parent, "state"):
 | |
|             self.smbm = {player: SMBoolManager(player, parent.state.smbm[player].maxDiff,
 | |
|                                     parent.state.smbm[player].onlyBossLeft) for player in
 | |
|                                         parent.get_game_players("Super Metroid")}
 | |
|             for player, group in parent.groups.items():
 | |
|                 if (group["game"] == "Super Metroid"):
 | |
|                     self.smbm[player] = SMBoolManager(player)
 | |
|                     if player not in parent.state.smbm:
 | |
|                         parent.state.smbm[player] = SMBoolManager(player)
 | |
|         else:
 | |
|             self.smbm = {}
 | |
| 
 | |
|     def copy_mixin(self, ret) -> CollectionState:
 | |
|         ret.smbm = {player: copy.deepcopy(self.smbm[player]) for player in self.smbm}
 | |
|         return ret
 | |
| 
 | |
|     def get_game_players(self, multiword: MultiWorld, game_name: str):
 | |
|         return tuple(player for player in multiword.get_all_ids() if multiword.game[player] == game_name)
 | |
| 
 | |
| 
 | |
| class SMWeb(WebWorld):
 | |
|     tutorials = [Tutorial(
 | |
|         "Multiworld Setup Guide",
 | |
|         "A guide to setting up the Super Metroid Client on your computer. This guide covers single-player, multiworld, and related software.",
 | |
|         "English",
 | |
|         "multiworld_en.md",
 | |
|         "multiworld/en",
 | |
|         ["Farrak Kilhn"]
 | |
|     )]
 | |
| 
 | |
| 
 | |
| class ByteEdit(TypedDict):
 | |
|     sym: Dict[str, Any]
 | |
|     offset: int
 | |
|     values: Iterable[int]
 | |
| 
 | |
| 
 | |
| locations_start_id = 82000
 | |
| items_start_id = 83000
 | |
| 
 | |
| 
 | |
| class SMWorld(World):
 | |
|     """
 | |
|      This is Very Adaptive Randomizer of Items and Areas for Super Metroid (VARIA SM). It supports
 | |
|      a wide range of options to randomize Item locations, required skills and even the connections 
 | |
|      between the main Areas!
 | |
|     """
 | |
|     game: str = "Super Metroid"
 | |
|     topology_present = True
 | |
|     options_dataclass = SMOptions
 | |
|     options: SMOptions
 | |
|       
 | |
|     settings: typing.ClassVar[SMSettings]
 | |
| 
 | |
|     item_name_to_id = {value.Name: items_start_id + value.Id for key, value in ItemManager.Items.items() if value.Id != None}
 | |
|     location_name_to_id = {key: locations_start_id + value.Id for key, value in locationsDict.items() if value.Id != None}
 | |
|     web = SMWeb()
 | |
| 
 | |
|     # changes to client DeathLink handling for 0.2.1
 | |
|     # changes to client Remote Item handling for 0.2.6
 | |
|     required_client_version = (0, 2, 6)
 | |
| 
 | |
|     itemManager: ItemManager
 | |
| 
 | |
|     Logic.factory('vanilla')
 | |
| 
 | |
|     def __init__(self, world: MultiWorld, player: int):
 | |
|         self.rom_name_available_event = threading.Event()
 | |
|         self.locations = {}
 | |
|         super().__init__(world, player)
 | |
| 
 | |
|     def generate_early(self):
 | |
|         Logic.factory('vanilla')
 | |
| 
 | |
|         dummy_rom_file = Utils.user_path(SMSettings.RomFile.copy_to)  # actual rom set in generate_output
 | |
|         self.variaRando = VariaRandomizer(self.options, dummy_rom_file, self.player)
 | |
|         self.multiworld.state.smbm[self.player] = SMBoolManager(self.player, self.variaRando.maxDifficulty)
 | |
| 
 | |
|         # keeps Nothing items local so no player will ever pickup Nothing
 | |
|         # doing so reduces contribution of this world to the Multiworld the more Nothing there is though
 | |
|         self.options.local_items.value.add('Nothing')
 | |
|         self.options.local_items.value.add('No Energy')
 | |
| 
 | |
|         if (self.variaRando.args.morphPlacement == "early"):
 | |
|             self.multiworld.local_early_items[self.player]['Morph Ball'] = 1
 | |
| 
 | |
|         self.remote_items = self.options.remote_items
 | |
| 
 | |
|         if (len(self.variaRando.randoExec.setup.restrictedLocs) > 0):
 | |
|             self.options.accessibility.value = Accessibility.option_minimal
 | |
|             logger.warning(f"accessibility forced to 'minimal' for player {self.multiworld.get_player_name(self.player)} because of 'fun' settings")
 | |
|     
 | |
|     def create_items(self):
 | |
|         itemPool = self.variaRando.container.itemPool
 | |
|         self.startItems = [variaItem for item in self.multiworld.precollected_items[self.player] for variaItem in ItemManager.Items.values() if variaItem.Name == item.name]
 | |
|         if self.options.start_inventory_removes_from_pool:
 | |
|             for item in self.startItems:
 | |
|                 if (item in itemPool):
 | |
|                     itemPool.remove(item)
 | |
| 
 | |
|         missingPool = 109 - len(itemPool)
 | |
|         for i in range(missingPool):
 | |
|             itemPool.append(ItemManager.Items['Nothing'])
 | |
|         
 | |
|         # Generate item pool
 | |
|         pool = []
 | |
|         self.locked_items = {}
 | |
|         self.NothingPool = []
 | |
|         weaponCount = [0, 0, 0]
 | |
|         for item in itemPool:
 | |
|             isAdvancement = True
 | |
|             if item.Type == 'Missile':
 | |
|                 if weaponCount[0] < 3:
 | |
|                     weaponCount[0] += 1
 | |
|                 else:
 | |
|                     isAdvancement = False
 | |
|             elif item.Type == 'Super':
 | |
|                 if weaponCount[1] < 2:
 | |
|                     weaponCount[1] += 1
 | |
|                 else:
 | |
|                     isAdvancement = False
 | |
|             elif item.Type == 'PowerBomb':
 | |
|                 if weaponCount[2] < 3:
 | |
|                     weaponCount[2] += 1
 | |
|                 else:
 | |
|                     isAdvancement = False
 | |
|             elif item.Category == 'Nothing':
 | |
|                 isAdvancement = False
 | |
| 
 | |
|             itemClass = ItemManager.Items[item.Type].Class
 | |
|             smitem = SMItem(item.Name,
 | |
|                             ItemClassification.progression if isAdvancement else ItemClassification.filler,
 | |
|                             item.Type,
 | |
|                             None if itemClass == 'Boss' else self.item_name_to_id[item.Name],
 | |
|                             player=self.player)
 | |
| 
 | |
|             if itemClass == 'Boss':
 | |
|                 self.locked_items[item.Name] = smitem
 | |
|             elif item.Category == 'Nothing':
 | |
|                 self.NothingPool.append(smitem)
 | |
|             else:
 | |
|                 pool.append(smitem)
 | |
| 
 | |
|         self.multiworld.itempool += pool
 | |
| 
 | |
|         for (location, item) in self.locked_items.items():
 | |
|             self.multiworld.get_location(location, self.player).place_locked_item(item)
 | |
|             self.multiworld.get_location(location, self.player).address = None
 | |
| 
 | |
|     def evalSMBool(self, smbool, maxDiff):
 | |
|         return smbool.bool == True and smbool.difficulty <= maxDiff
 | |
| 
 | |
|     def add_entrance_rule(self, entrance, player, func):
 | |
|         add_rule(entrance, lambda state: self.evalSMBool(func(state.smbm[player]), state.smbm[player].maxDiff))
 | |
| 
 | |
|     def set_rules(self):
 | |
|         def add_accessFrom_rule(location, player, accessFrom):
 | |
|             add_rule(location, lambda state: any((state.can_reach(accessName, player=player) and self.evalSMBool(rule(state.smbm[player]), state.smbm[player].maxDiff)) for accessName, rule in accessFrom.items()))
 | |
| 
 | |
|         def add_postAvailable_rule(location, player, func):
 | |
|             add_rule(location, lambda state: self.evalSMBool(func(state.smbm[player]), state.smbm[player].maxDiff))
 | |
| 
 | |
|         def set_available_rule(location, player, func):
 | |
|             set_rule(location, lambda state: self.evalSMBool(func(state.smbm[player]), state.smbm[player].maxDiff))
 | |
| 
 | |
|         def set_entrance_rule(entrance, player, func):
 | |
|             set_rule(entrance, lambda state: self.evalSMBool(func(state.smbm[player]), state.smbm[player].maxDiff))
 | |
| 
 | |
|         self.multiworld.completion_condition[self.player] = lambda state: state.has('Mother Brain', self.player)
 | |
| 
 | |
|         for key, value in locationsDict.items():
 | |
|             location = self.multiworld.get_location(key, self.player)
 | |
|             set_available_rule(location, self.player, value.Available)
 | |
|             if value.AccessFrom is not None:
 | |
|                 add_accessFrom_rule(location, self.player, value.AccessFrom)
 | |
|             if value.PostAvailable is not None:
 | |
|                 add_postAvailable_rule(location, self.player, value.PostAvailable)
 | |
| 
 | |
|         for accessPoint in Logic.accessPoints:
 | |
|             if not accessPoint.Escape:
 | |
|                 for key, value1 in accessPoint.intraTransitions.items():
 | |
|                     set_entrance_rule(self.multiworld.get_entrance(accessPoint.Name + "->" + key, self.player), self.player, value1)
 | |
| 
 | |
|     def create_region(self, world: MultiWorld, player: int, name: str, locations=None, exits=None):
 | |
|         ret = Region(name, player, world)
 | |
|         if locations:
 | |
|             for loc in locations:
 | |
|                 location = self.locations[loc]
 | |
|                 location.parent_region = ret
 | |
|                 ret.locations.append(location)
 | |
|         if exits:
 | |
|             for exit in exits:
 | |
|                 ret.exits.append(Entrance(player, exit, ret))
 | |
|         return ret
 | |
| 
 | |
|     def create_regions(self):
 | |
|         # create locations
 | |
|         for name in locationsDict:
 | |
|             self.locations[name] = SMLocation(self.player, name, self.location_name_to_id.get(name, None))
 | |
| 
 | |
|         # create regions
 | |
|         regions = []
 | |
|         for accessPoint in Logic.accessPoints:
 | |
|             if not accessPoint.Escape:
 | |
|                 regions.append(self.create_region(  self.multiworld,
 | |
|                                                     self.player,
 | |
|                                                     accessPoint.Name,
 | |
|                                                     None,
 | |
|                                                     [accessPoint.Name + "->" + key for key in accessPoint.intraTransitions.keys()]))
 | |
| 
 | |
|         self.multiworld.regions += regions
 | |
| 
 | |
|         # create a region for each location and link each to what the location has access
 | |
|         # we make them one way so that the filler (and spoiler log) doesnt try to use those region as intermediary path
 | |
|         # this is required in AP because a location cant have multiple parent regions
 | |
|         locationRegions = []
 | |
|         for locationName, value in locationsDict.items():
 | |
|             locationRegions.append(self.create_region(  self.multiworld,
 | |
|                                                         self.player,
 | |
|                                                         locationName,
 | |
|                                                         [locationName]))
 | |
|             for key in value.AccessFrom.keys():
 | |
|                 currentRegion = self.multiworld.get_region(key, self.player)
 | |
|                 currentRegion.exits.append(Entrance(self.player, key + "->"+ locationName, currentRegion))
 | |
| 
 | |
|         self.multiworld.regions += locationRegions
 | |
| 
 | |
|         #create entrances
 | |
|         regionConcat = regions + locationRegions
 | |
|         for region in regionConcat:
 | |
|             for exit in region.exits:
 | |
|                 exit.connect(self.multiworld.get_region(exit.name[exit.name.find("->") + 2:], self.player))
 | |
| 
 | |
|         for src, dest in self.variaRando.randoExec.areaGraph.InterAreaTransitions:
 | |
|             src_region = self.multiworld.get_region(src.Name, self.player)
 | |
|             dest_region = self.multiworld.get_region(dest.Name, self.player)
 | |
|             if src.Name + "->" + dest.Name not in self.multiworld.regions.entrance_cache[self.player]:
 | |
|                 src_region.exits.append(Entrance(self.player, src.Name + "->" + dest.Name, src_region))
 | |
|             srcDestEntrance = self.multiworld.get_entrance(src.Name + "->" + dest.Name, self.player)
 | |
|             srcDestEntrance.connect(dest_region)
 | |
|             self.add_entrance_rule(self.multiworld.get_entrance(src.Name + "->" + dest.Name, self.player), self.player, getAccessPoint(src.Name).traverse)
 | |
| 
 | |
|         self.multiworld.regions += [
 | |
|             self.create_region(self.multiworld, self.player, 'Menu', None, ['StartAP'])
 | |
|         ]
 | |
| 
 | |
|         startAP = self.multiworld.get_entrance('StartAP', self.player)
 | |
|         startAP.connect(self.multiworld.get_region(self.variaRando.args.startLocation, self.player))
 | |
| 
 | |
|     def collect(self, state: CollectionState, item: Item) -> bool:
 | |
|         state.smbm[self.player].addItem(item.type)
 | |
|         return super(SMWorld, self).collect(state, item)
 | |
| 
 | |
|     def remove(self, state: CollectionState, item: Item) -> bool:
 | |
|         state.smbm[self.player].removeItem(item.type)
 | |
|         return super(SMWorld, self).remove(state, item)
 | |
| 
 | |
|     def create_item(self, name: str) -> Item:
 | |
|         item = next((x for x in ItemManager.Items.values() if x.Name == name), None)
 | |
|         if item:
 | |
|             return SMItem(item.Name, ItemClassification.progression if item.Class != 'Minor' else ItemClassification.filler, item.Type, self.item_name_to_id[item.Name],
 | |
|                       player=self.player)
 | |
|         raise KeyError(f"Item {name} for {self.player_name} is invalid.")
 | |
| 
 | |
|     def get_filler_item_name(self) -> str:
 | |
|         if self.multiworld.random.randint(0, 100) < self.options.minor_qty.value:
 | |
|             power_bombs = self.options.power_bomb_qty.value
 | |
|             missiles = self.options.missile_qty.value
 | |
|             super_missiles = self.options.super_qty.value
 | |
|             roll = self.multiworld.random.randint(1, power_bombs + missiles + super_missiles)
 | |
|             if roll <= power_bombs:
 | |
|                 return "Power Bomb"
 | |
|             elif roll <= power_bombs + missiles:
 | |
|                 return "Missile"
 | |
|             else:
 | |
|                 return "Super Missile"
 | |
|         else:
 | |
|             return "Nothing"
 | |
| 
 | |
|     def pre_fill(self):
 | |
|         if len(self.NothingPool) > 0:
 | |
|             nonChozoLoc = []
 | |
|             chozoLoc = []
 | |
| 
 | |
|             for loc in self.locations.values():
 | |
|                 if loc.item is None:
 | |
|                     if locationsDict[loc.name].isChozo():
 | |
|                         chozoLoc.append(loc)
 | |
|                     else:
 | |
|                         nonChozoLoc.append(loc)
 | |
| 
 | |
|             self.multiworld.random.shuffle(nonChozoLoc)
 | |
|             self.multiworld.random.shuffle(chozoLoc)
 | |
|             missingCount = len(self.NothingPool) - len(nonChozoLoc)
 | |
|             locations = nonChozoLoc
 | |
|             if (missingCount > 0):
 | |
|                 locations += chozoLoc[:missingCount]
 | |
|             locations = locations[:len(self.NothingPool)]
 | |
|             for item, loc in zip(self.NothingPool, locations):
 | |
|                 loc.place_locked_item(item)
 | |
|                 loc.address = loc.item.code = None
 | |
| 
 | |
|     def post_fill(self):
 | |
|         def get_player_ItemLocation(progression_only: bool):
 | |
|             return [
 | |
|                 ItemLocation(
 | |
|                     copy.copy(
 | |
|                         ItemManager.Items[
 | |
|                             itemLoc.item.type
 | |
|                             if isinstance(itemLoc.item, SMItem) and itemLoc.item.type in ItemManager.Items
 | |
|                             else 'ArchipelagoItem'
 | |
|                         ]
 | |
|                     ),
 | |
|                     copy.copy(
 | |
|                         locationsDict[itemLoc.name]
 | |
|                         if itemLoc.game == self.game
 | |
|                         else locationsDict[first_local_collected_loc.name]
 | |
|                     ),
 | |
|                     itemLoc.item.player,
 | |
|                     True
 | |
|                 )
 | |
|                 for itemLoc in spheres
 | |
|                 if itemLoc.item.player == self.player and (not progression_only or itemLoc.item.advancement)
 | |
|             ]
 | |
| 
 | |
|         # Having a sorted itemLocs from collection order is required for escapeTrigger when Tourian is Disabled.
 | |
|         # We cant use stage_post_fill for this as its called after worlds' post_fill.
 | |
|         # get_spheres could be cached in multiworld?
 | |
|         # Another possible solution would be to have a globally accessible list of items in the order in which the get placed in push_item
 | |
|         # and use the inversed starting from the first progression item.
 | |
|         spheres: List[Location] = getattr(self.multiworld, "_sm_spheres", None)
 | |
|         if spheres is None:
 | |
|             spheres = [itemLoc for sphere in self.multiworld.get_spheres() for itemLoc in sorted(sphere, key=lambda location: location.name)]
 | |
|             setattr(self.multiworld, "_sm_spheres", spheres)
 | |
| 
 | |
|         self.itemLocs = [
 | |
|             ItemLocation(copy.copy(ItemManager.Items[itemLoc.item.type
 | |
|                          if isinstance(itemLoc.item, SMItem) and itemLoc.item.type in ItemManager.Items else
 | |
|                          'ArchipelagoItem']),
 | |
|                          copy.copy(locationsDict[itemLoc.name]), itemLoc.item.player, True)
 | |
|             for itemLoc in self.multiworld.get_locations(self.player)
 | |
|         ]
 | |
| 
 | |
|         escapeTrigger = None
 | |
|         if self.variaRando.randoExec.randoSettings.restrictions["EscapeTrigger"]:
 | |
|             #used to simulate received items
 | |
|             first_local_collected_loc = next(itemLoc for itemLoc in spheres if itemLoc.player == self.player)
 | |
| 
 | |
|             playerItemsItemLocs = get_player_ItemLocation(False)
 | |
|             playerProgItemsItemLocs = get_player_ItemLocation(True)
 | |
| 
 | |
|             for itemLoc in playerItemsItemLocs + playerProgItemsItemLocs:
 | |
|                 if itemLoc.Item.Class == "Boss":
 | |
|                     itemLoc.Item.Class = "Minor"
 | |
| 
 | |
|             escapeTrigger = (playerItemsItemLocs, playerProgItemsItemLocs, 'Full')
 | |
| 
 | |
|         escapeOk = self.variaRando.randoExec.graphBuilder.escapeGraph(self.variaRando.container, self.variaRando.randoExec.areaGraph, self.variaRando.randoExec.randoSettings.maxDiff, escapeTrigger)
 | |
|         if (not escapeOk):
 | |
|             logger.warning(f"Escape Rando forced to 'Off' for player {self.multiworld.get_player_name(self.player)} because could not find a solution for escape")
 | |
| 
 | |
|         # if we couldn't find an area layout then the escape graph is not created either
 | |
|         # and getDoorConnections will crash if random escape is activated.
 | |
|         self.variaRando.doors = GraphUtils.getDoorConnections(self.variaRando.randoExec.areaGraph,
 | |
|                                     self.variaRando.args.area, self.variaRando.args.bosses,
 | |
|                                     self.variaRando.args.escapeRando if escapeOk else False)
 | |
|         
 | |
|         self.variaRando.randoExec.postProcessItemLocs(self.itemLocs, self.variaRando.args.hideItems)
 | |
| 
 | |
|     @classmethod
 | |
|     def stage_post_fill(cls, world):
 | |
|         new_state = CollectionState(world)
 | |
|         progitempool = []
 | |
|         for item in world.itempool:
 | |
|             if item.game == "Super Metroid" and item.advancement:
 | |
|                 progitempool.append(item)
 | |
| 
 | |
|         for item in progitempool:
 | |
|             new_state.collect(item, True)
 | |
| 
 | |
|         bossesLoc = ['Draygon', 'Kraid', 'Ridley', 'Phantoon', 'Mother Brain']
 | |
|         for player in world.get_game_players("Super Metroid"):
 | |
|             for bossLoc in bossesLoc:
 | |
|                 if not world.get_location(bossLoc, player).can_reach(new_state):
 | |
|                     world.state.smbm[player].onlyBossLeft = True
 | |
|                     break
 | |
| 
 | |
|     def getWordArray(self, w: int) -> List[int]:
 | |
|         """ little-endian convert a 16-bit number to an array of numbers <= 255 each """
 | |
|         return [w & 0x00FF, (w & 0xFF00) >> 8]
 | |
| 
 | |
|     # used for remote location Credits Spoiler of local items and Objectives' writeItemsMasks
 | |
|     class DummyLocation:
 | |
|         def __init__(self, name):
 | |
|             self.Name = name
 | |
|             self.restricted = False
 | |
| 
 | |
|         def isBoss(self):
 | |
|             return False
 | |
| 
 | |
|     def convertToROMItemName(self, itemName):
 | |
|         charMap = { "A" : 0x3CE0, 
 | |
|                     "B" : 0x3CE1,
 | |
|                     "C" : 0x3CE2,
 | |
|                     "D" : 0x3CE3,
 | |
|                     "E" : 0x3CE4,
 | |
|                     "F" : 0x3CE5,
 | |
|                     "G" : 0x3CE6,
 | |
|                     "H" : 0x3CE7,
 | |
|                     "I" : 0x3CE8,
 | |
|                     "J" : 0x3CE9,
 | |
|                     "K" : 0x3CEA,
 | |
|                     "L" : 0x3CEB,
 | |
|                     "M" : 0x3CEC,
 | |
|                     "N" : 0x3CED,
 | |
|                     "O" : 0x3CEE,
 | |
|                     "P" : 0x3CEF,
 | |
|                     "Q" : 0x3CF0,
 | |
|                     "R" : 0x3CF1,
 | |
|                     "S" : 0x3CF2,
 | |
|                     "T" : 0x3CF3,
 | |
|                     "U" : 0x3CF4,
 | |
|                     "V" : 0x3CF5,
 | |
|                     "W" : 0x3CF6,
 | |
|                     "X" : 0x3CF7,
 | |
|                     "Y" : 0x3CF8,
 | |
|                     "Z" : 0x3CF9,
 | |
|                     " " : 0x3C4E,
 | |
|                     "!" : 0x3CFF,
 | |
|                     "?" : 0x3CFE,
 | |
|                     "'" : 0x3CFD,
 | |
|                     "," : 0x3CFB,
 | |
|                     "." : 0x3CFA,
 | |
|                     "-" : 0x3CCF,
 | |
|                     "_" : 0x000E,
 | |
|                     "1" : 0x3C00,
 | |
|                     "2" : 0x3C01,
 | |
|                     "3" : 0x3C02,
 | |
|                     "4" : 0x3C03,
 | |
|                     "5" : 0x3C04,
 | |
|                     "6" : 0x3C05,
 | |
|                     "7" : 0x3C06,
 | |
|                     "8" : 0x3C07,
 | |
|                     "9" : 0x3C08,
 | |
|                     "0" : 0x3C09,
 | |
|                     "%" : 0x3C0A}
 | |
|         data = []
 | |
| 
 | |
|         itemName = itemName.upper()[:26]
 | |
|         itemName = itemName.strip()
 | |
|         itemName = itemName.center(26, " ")    
 | |
|         itemName = "___" + itemName + "___"
 | |
| 
 | |
|         for char in itemName:
 | |
|             [w0, w1] = self.getWordArray(charMap.get(char, 0x3C4E))
 | |
|             data.append(w0)
 | |
|             data.append(w1)
 | |
|         return data
 | |
| 
 | |
|     def APPrePatchRom(self, romPatcher):
 | |
|         # first apply the sm multiworld code patch named 'basepatch' (also has empty tables that we'll overwrite),
 | |
|         #  + apply some patches from varia that we want to be always-on.
 | |
|         # basepatch and variapatches are both generated from https://github.com/lordlou/SMBasepatch
 | |
|         romPatcher.applyIPSPatch("/".join((os.path.dirname(self.__file__),
 | |
|                                               "data", "SMBasepatch_prebuilt", "multiworld-basepatch.ips")))
 | |
|         romPatcher.applyIPSPatch("/".join((os.path.dirname(self.__file__),
 | |
|                                               "data", "SMBasepatch_prebuilt", "variapatches.ips")))
 | |
| 
 | |
|     def APPostPatchRom(self, romPatcher):
 | |
|         symbols = get_sm_symbols("/".join((os.path.dirname(self.__file__),
 | |
|                                               "data", "SMBasepatch_prebuilt", "sm-basepatch-symbols.json")))
 | |
| 
 | |
|         # gather all player ids and names relevant to this rom, then write player name and player id data tables
 | |
|         playerIdSet: Set[int] = {0}  # 0 is for "Archipelago" server
 | |
|         for itemLoc in self.multiworld.get_locations():
 | |
|             assert itemLoc.item, f"World of player '{self.multiworld.player_name[itemLoc.player]}' has a loc.item " + \
 | |
|                                  f"that is {itemLoc.item} during generate_output"
 | |
|             # add each playerid who has a location containing an item to send to us *or* to an item_link we're part of
 | |
|             if itemLoc.item.player == self.player or \
 | |
|                     (itemLoc.item.player in self.multiworld.groups and
 | |
|                      self.player in self.multiworld.groups[itemLoc.item.player]['players']):
 | |
|                 playerIdSet |= {itemLoc.player}
 | |
|             # add each playerid, including item link ids, that we'll be sending items to
 | |
|             if itemLoc.player == self.player:
 | |
|                 playerIdSet |= {itemLoc.item.player}
 | |
|         if len(playerIdSet) > SM_ROM_PLAYERDATA_COUNT:
 | |
|             # max 202 entries, but it's possible for item links to add enough replacement items for us, that are placed
 | |
|             # in worlds that otherwise have no relation to us, that the 2*location count limit is exceeded
 | |
|             logger.warning("SM is interacting with too many players to fit in ROM. "
 | |
|                            f"Removing the highest {len(playerIdSet) - SM_ROM_PLAYERDATA_COUNT} ids to fit")
 | |
|             playerIdSet = set(sorted(playerIdSet)[:SM_ROM_PLAYERDATA_COUNT])
 | |
|         otherPlayerIndex: Dict[int, int] = {}  # ap player id -> rom-local player index
 | |
|         playerNameData: List[ByteEdit] = []
 | |
|         playerIdData: List[ByteEdit] = []
 | |
|         # sort all player data by player id so that the game can look up a player's data reasonably quickly when
 | |
|         # the client sends an ap playerid to the game
 | |
|         for i, playerid in enumerate(sorted(playerIdSet)):
 | |
|             playername = self.multiworld.player_name[playerid] if playerid != 0 else "Archipelago"
 | |
|             playerIdForRom = playerid
 | |
|             if playerid > SM_ROM_MAX_PLAYERID:
 | |
|                 # note, playerIdForRom = 0 is not unique so the game cannot look it up.
 | |
|                 # instead it will display the player received-from as "Archipelago"
 | |
|                 playerIdForRom = 0
 | |
|                 if playerid == self.player:
 | |
|                     raise Exception(f"SM rom cannot fit enough bits to represent self player id {playerid}")
 | |
|                 else:
 | |
|                     logger.warning(f"SM rom cannot fit enough bits to represent player id {playerid}, setting to 0 in rom")
 | |
|             otherPlayerIndex[playerid] = i
 | |
|             playerNameData.append({"sym": symbols["rando_player_name_table"],
 | |
|                                    "offset": i * 16,
 | |
|                                    "values": playername[:16].upper().center(16).encode()})
 | |
|             playerIdData.append({"sym": symbols["rando_player_id_table"],
 | |
|                                  "offset": i * 2,
 | |
|                                  "values": self.getWordArray(playerIdForRom)})
 | |
| 
 | |
|         multiWorldLocations: List[ByteEdit] = []
 | |
|         multiWorldItems: List[ByteEdit] = []
 | |
|         idx = 0
 | |
|         vanillaItemTypesCount = 21
 | |
|         for itemLoc in self.multiworld.get_locations(self.player):
 | |
|             if "Boss" not in locationsDict[itemLoc.name].Class:
 | |
|                 SMZ3NameToSMType = {
 | |
|                     "ETank": "ETank", "Missile": "Missile", "Super": "Super", "PowerBomb": "PowerBomb", "Bombs": "Bomb",
 | |
|                     "Charge": "Charge", "Ice": "Ice", "HiJump": "HiJump", "SpeedBooster": "SpeedBooster",
 | |
|                     "Wave": "Wave", "Spazer": "Spazer", "SpringBall": "SpringBall", "Varia": "Varia", "Plasma": "Plasma",
 | |
|                     "Grapple": "Grapple", "Morph": "Morph", "ReserveTank": "Reserve", "Gravity": "Gravity",
 | |
|                     "XRay": "XRayScope", "SpaceJump": "SpaceJump", "ScrewAttack": "ScrewAttack"
 | |
|                 }
 | |
|                 if isinstance(itemLoc.item, SMItem) and itemLoc.item.type in ItemManager.Items:
 | |
|                     itemId = ItemManager.Items[itemLoc.item.type].Id
 | |
|                 elif itemLoc.item.game == "SMZ3" and itemLoc.item.name in SMZ3NameToSMType.keys():
 | |
|                     itemId = ItemManager.Items[SMZ3NameToSMType[itemLoc.item.name]].Id
 | |
|                 else:
 | |
|                     itemId = ItemManager.Items["ArchipelagoItem"].Id + idx
 | |
|                     multiWorldItems.append({"sym": symbols["message_item_names"],
 | |
|                                             "offset": (vanillaItemTypesCount + idx)*64,
 | |
|                                             "values": self.convertToROMItemName(itemLoc.item.name)})
 | |
|                     idx += 1
 | |
| 
 | |
|                 if itemLoc.item.player == self.player:
 | |
|                     itemDestinationType = 0  # dest type 0 means 'regular old SM item' per itemtable.asm
 | |
|                 elif itemLoc.item.player in self.multiworld.groups and \
 | |
|                         self.player in self.multiworld.groups[itemLoc.item.player]['players']:
 | |
|                     # dest type 2 means 'SM item link item that sends to the current player and others'
 | |
|                     # per itemtable.asm (groups are synonymous with item_links, currently)
 | |
|                     itemDestinationType = 2
 | |
|                 else:
 | |
|                     itemDestinationType = 1  # dest type 1 means 'item for entirely someone else' per itemtable.asm
 | |
| 
 | |
|                 [w0, w1] = self.getWordArray(itemDestinationType)
 | |
|                 [w2, w3] = self.getWordArray(itemId)
 | |
|                 [w4, w5] = self.getWordArray(otherPlayerIndex[itemLoc.item.player] if itemLoc.item.player in
 | |
|                                              otherPlayerIndex else 0)
 | |
|                 [w6, w7] = self.getWordArray(0 if itemLoc.item.advancement else 1)
 | |
|                 multiWorldLocations.append({"sym": symbols["rando_item_table"],
 | |
|                                             "offset": locationsDict[itemLoc.name].Id*8,
 | |
|                                             "values": [w0, w1, w2, w3, w4, w5, w6, w7]})
 | |
| 
 | |
|         itemSprites = [{"fileName":          "off_world_prog_item.bin",
 | |
|                         "paletteSymbolName": "prog_item_eight_palette_indices",
 | |
|                         "dataSymbolName":    "offworld_graphics_data_progression_item"},
 | |
| 
 | |
|                        {"fileName":          "off_world_item.bin",
 | |
|                         "paletteSymbolName": "nonprog_item_eight_palette_indices",
 | |
|                         "dataSymbolName":    "offworld_graphics_data_item"}]
 | |
|         idx = 0
 | |
|         offworldSprites: List[ByteEdit] = []
 | |
|         for itemSprite in itemSprites:
 | |
|             with openFile("/".join((os.path.dirname(self.__file__), "data", "custom_sprite", itemSprite["fileName"])), 'rb') as stream:
 | |
|                 buffer = bytearray(stream.read())
 | |
|                 offworldSprites.append({"sym": symbols[itemSprite["paletteSymbolName"]],
 | |
|                                         "offset": 0,
 | |
|                                         "values": buffer[0:8]})
 | |
|                 offworldSprites.append({"sym": symbols[itemSprite["dataSymbolName"]],
 | |
|                                         "offset": 0,
 | |
|                                         "values": buffer[8:264]})
 | |
|                 idx += 1
 | |
| 
 | |
|         deathLink: List[ByteEdit] = [{
 | |
|             "sym": symbols["config_deathlink"],
 | |
|             "offset": 0,
 | |
|             "values": [self.options.death_link.value]
 | |
|         }]
 | |
|         remoteItem: List[ByteEdit] = [{
 | |
|             "sym": symbols["config_remote_items"],
 | |
|             "offset": 0,
 | |
|             "values": self.getWordArray(0b001 + (0b010 if self.remote_items else 0b000))
 | |
|         }]
 | |
|         ownPlayerId: List[ByteEdit] = [{
 | |
|             "sym": symbols["config_player_id"],
 | |
|             "offset": 0,
 | |
|             "values": self.getWordArray(self.player)
 | |
|         }]
 | |
| 
 | |
|         patchDict = {   'MultiWorldLocations': multiWorldLocations,
 | |
|                         'MultiWorldItems': multiWorldItems,
 | |
|                         'offworldSprites': offworldSprites,
 | |
|                         'deathLink': deathLink,
 | |
|                         'remoteItem': remoteItem,
 | |
|                         'ownPlayerId': ownPlayerId,
 | |
|                         'playerNameData':  playerNameData,
 | |
|                         'playerIdData':  playerIdData}
 | |
| 
 | |
|         # convert an array of symbolic byte_edit dicts like {"sym": symobj, "offset": 0, "values": [1, 0]}
 | |
|         # to a single rom patch dict like {0x438c: [1, 0], 0xa4a5: [0, 0, 0]} which varia will understand and apply
 | |
|         def resolve_symbols_to_file_offset_based_dict(byte_edits_arr: List[ByteEdit]) -> Dict[int, Iterable[int]]:
 | |
|             this_patch_as_dict: Dict[int, Iterable[int]] = {}
 | |
|             for byte_edit in byte_edits_arr:
 | |
|                 offset_within_rom_file: int = byte_edit["sym"]["offset_within_rom_file"] + byte_edit["offset"]
 | |
|                 this_patch_as_dict[offset_within_rom_file] = byte_edit["values"]
 | |
|             return this_patch_as_dict
 | |
| 
 | |
|         for patchname, byte_edits_arr in patchDict.items():
 | |
|             patchDict[patchname] = resolve_symbols_to_file_offset_based_dict(byte_edits_arr)
 | |
| 
 | |
|         romPatcher.applyIPSPatchDict(patchDict)
 | |
| 
 | |
|         openTourianGreyDoors = {0x07C823 + 5: [0x0C], 0x07C831 + 5: [0x0C]}
 | |
|         romPatcher.applyIPSPatchDict({'openTourianGreyDoors': openTourianGreyDoors})
 | |
| 
 | |
| 
 | |
|         # set rom name
 | |
|         # 21 bytes
 | |
|         from Utils import __version__
 | |
|         self.romName = bytearray(f'SM{__version__.replace(".", "")[0:3]}_{self.player}_{self.multiworld.seed:11}', 'utf8')[:21]
 | |
|         self.romName.extend([0] * (21 - len(self.romName)))
 | |
|         # clients should read from 0x7FC0, the location of the rom title in the SNES header.
 | |
|         # duplicative ROM name at 0x1C4F00 is still written here for now, since people with archipelago pre-0.3.0 client installed will still be depending on this location for connecting to SM
 | |
|         romPatcher.applyIPSPatch('ROMName', { 'ROMName':  {0x1C4F00 : self.romName, 0x007FC0 : self.romName} })
 | |
| 
 | |
| 
 | |
|         startItemROMAddressBase = symbols["start_item_data_major"]["offset_within_rom_file"]
 | |
| 
 | |
|         # array for each item:
 | |
|         #  offset within ROM table "start_item_data_major" of this item"s info (starting status)
 | |
|         #  item bitmask or amount per pickup (BVOB = base value or bitmask),
 | |
|         #  offset within ROM table "start_item_data_major" of this item"s info (starting maximum/starting collected items)
 | |
|         #                                 current  BVOB   max
 | |
|         #                                 -------  ----   ---
 | |
|         startItemROMDict = {"ETank":        [ 0x8, 0x64,  0xA],
 | |
|                             "Missile":      [ 0xC,  0x5,  0xE],
 | |
|                             "Super":        [0x10,  0x5, 0x12],
 | |
|                             "PowerBomb":    [0x14,  0x5, 0x16],
 | |
|                             "Reserve":      [0x1A, 0x64, 0x18],
 | |
|                             "Morph":        [ 0x2,  0x4,  0x0],
 | |
|                             "Bomb":         [ 0x3, 0x10,  0x1],
 | |
|                             "SpringBall":   [ 0x2,  0x2,  0x0],
 | |
|                             "HiJump":       [ 0x3,  0x1,  0x1],
 | |
|                             "Varia":        [ 0x2,  0x1,  0x0],
 | |
|                             "Gravity":      [ 0x2, 0x20,  0x0],
 | |
|                             "SpeedBooster": [ 0x3, 0x20,  0x1],
 | |
|                             "SpaceJump":    [ 0x3,  0x2,  0x1],
 | |
|                             "ScrewAttack":  [ 0x2,  0x8,  0x0],
 | |
|                             "Charge":       [ 0x7, 0x10,  0x5],
 | |
|                             "Ice":          [ 0x6,  0x2,  0x4],
 | |
|                             "Wave":         [ 0x6,  0x1,  0x4],
 | |
|                             "Spazer":       [ 0x6,  0x4,  0x4],
 | |
|                             "Plasma":       [ 0x6,  0x8,  0x4],
 | |
|                             "Grapple":      [ 0x3, 0x40,  0x1],
 | |
|                             "XRayScope":    [ 0x3, 0x80,  0x1]
 | |
| 
 | |
|         # BVOB = base value or bitmask
 | |
|                             }
 | |
|         mergedData = {}
 | |
|         hasETank = False
 | |
|         hasSpazer = False
 | |
|         hasPlasma = False
 | |
|         for startItem in self.startItems:
 | |
|             item = startItem.Type
 | |
|             if item == "ETank": hasETank = True
 | |
|             if item == "Spazer": hasSpazer = True
 | |
|             if item == "Plasma": hasPlasma = True
 | |
|             if (item in ["ETank", "Missile", "Super", "PowerBomb", "Reserve"]):
 | |
|                 (currentValue, amountPerItem, maxValue) = startItemROMDict[item]
 | |
|                 if (startItemROMAddressBase + currentValue) in mergedData:
 | |
|                     mergedData[startItemROMAddressBase + currentValue] += amountPerItem
 | |
|                     mergedData[startItemROMAddressBase + maxValue] += amountPerItem
 | |
|                 else:
 | |
|                     mergedData[startItemROMAddressBase + currentValue] = amountPerItem
 | |
|                     mergedData[startItemROMAddressBase + maxValue] = amountPerItem
 | |
|             else:
 | |
|                 (collected, bitmask, equipped) = startItemROMDict[item]
 | |
|                 if (startItemROMAddressBase + collected) in mergedData:
 | |
|                     mergedData[startItemROMAddressBase + collected] |= bitmask
 | |
|                     mergedData[startItemROMAddressBase + equipped] |= bitmask
 | |
|                 else:
 | |
|                     mergedData[startItemROMAddressBase + collected] = bitmask
 | |
|                     mergedData[startItemROMAddressBase + equipped] = bitmask
 | |
| 
 | |
|         if hasETank:
 | |
|             # we are overwriting the starting energy, so add up the E from 99 (normal starting energy) rather than from 0
 | |
|             mergedData[startItemROMAddressBase + 0x8] += 99
 | |
|             mergedData[startItemROMAddressBase + 0xA] += 99
 | |
| 
 | |
|         if hasSpazer and hasPlasma:
 | |
|             # de-equip spazer.
 | |
|             # otherwise, firing the unintended spazer+plasma combo would cause massive game glitches and crashes
 | |
|             mergedData[startItemROMAddressBase + 0x4] &= ~0x4
 | |
| 
 | |
|         for key, value in mergedData.items():
 | |
|             if (key - startItemROMAddressBase > 7):
 | |
|                 [w0, w1] = self.getWordArray(value)
 | |
|                 mergedData[key] = [w0, w1]
 | |
|             else:
 | |
|                 mergedData[key] = [value]
 | |
| 
 | |
| 
 | |
|         startItemPatch = { "startItemPatch":  mergedData }
 | |
|         romPatcher.applyIPSPatch("startItemPatch", startItemPatch)
 | |
| 
 | |
|         # commit all the changes we've made here to the ROM
 | |
|         romPatcher.commitIPS()
 | |
| 
 | |
|         itemLocs = [ItemLocation(ItemManager.Items[itemLoc.item.type], locationsDict[itemLoc.name] if itemLoc.name in locationsDict and itemLoc.player == self.player else self.DummyLocation(self.multiworld.get_player_name(itemLoc.player) + " " + itemLoc.name), True) for itemLoc in self.multiworld.get_locations() if itemLoc.item.player == self.player]
 | |
|         progItemLocs = [ItemLocation(ItemManager.Items[itemLoc.item.type], locationsDict[itemLoc.name] if itemLoc.name in locationsDict and itemLoc.player == self.player else self.DummyLocation(self.multiworld.get_player_name(itemLoc.player) + " " + itemLoc.name), True) for itemLoc in self.multiworld.get_locations() if itemLoc.item.player == self.player and itemLoc.item.advancement == True]
 | |
| 
 | |
| 
 | |
|         romPatcher.writeObjectives(itemLocs, romPatcher.settings["tourian"])
 | |
|         romPatcher.writeItemsLocs(self.itemLocs)
 | |
| 
 | |
|         romPatcher.writeSplitLocs(self.variaRando.args.majorsSplit, self.itemLocs, None) #progItemLocs)
 | |
|         romPatcher.writeItemsNumber()
 | |
|         if not romPatcher.settings["isPlando"]:
 | |
|             romPatcher.writeSeed(romPatcher.settings["seed"]) # lol if race mode
 | |
|         romPatcher.writeSpoiler(itemLocs, progItemLocs)
 | |
|         romPatcher.writeRandoSettings(self.variaRando.randoExec.randoSettings, itemLocs)
 | |
|         romPatcher.writeDoorConnections(romPatcher.settings["doors"])
 | |
|         romPatcher.writeVersion(romPatcher.settings["displayedVersion"])
 | |
|         if romPatcher.settings["ctrlDict"] is not None:
 | |
|             romPatcher.writeControls(romPatcher.settings["ctrlDict"])
 | |
|         if romPatcher.settings["moonWalk"] == True:
 | |
|             romPatcher.enableMoonWalk()
 | |
| 
 | |
|         romPatcher.writeMagic()
 | |
|         romPatcher.writeMajorsSplit(romPatcher.settings["majorsSplit"])
 | |
| 
 | |
|         #if self.settings["isPlando"] and self.race is None:
 | |
|         #    doorsPtrs = GraphUtils.getAps2DoorsPtrs()
 | |
|         #    self.writePlandoTransitions(self.settings["plando"]["graphTrans"], doorsPtrs,
 | |
|         #                                self.settings["plando"]["maxTransitions"])
 | |
|         #    self.writePlandoAddresses(self.settings["plando"]["visitedLocations"])
 | |
|         #if self.settings["isPlando"] and self.settings["plando"]["additionalETanks"] != 0:
 | |
|         #    self.writeAdditionalETanks(self.settings["plando"]["additionalETanks"])
 | |
| 
 | |
|         romPatcher.end()
 | |
| 
 | |
|     def generate_output(self, output_directory: str):
 | |
|         try:
 | |
|             patcher = self.variaRando.PatchRom(self.APPrePatchRom, self.APPostPatchRom)
 | |
|             self.rom_name = self.romName
 | |
| 
 | |
|             patch = SMProcedurePatch(player=self.player, player_name=self.multiworld.player_name[self.player])
 | |
|             patch.write_tokens(patcher.romFile.getPatchDict())
 | |
|             rom_path = os.path.join(output_directory, f"{self.multiworld.get_out_file_name_base(self.player)}"
 | |
|                                                       f"{patch.patch_file_ending}")
 | |
|             patch.write(rom_path)
 | |
|             
 | |
|         except:
 | |
|             raise
 | |
|         finally:
 | |
|             self.rom_name_available_event.set()  # make sure threading continues and errors are collected
 | |
| 
 | |
|     def checksum_mirror_sum(self, start, length, mask = 0x800000):
 | |
|         while not(length & mask) and mask:
 | |
|             mask >>= 1
 | |
| 
 | |
|         part1 = sum(start[:mask]) & 0xFFFF
 | |
|         part2 = 0
 | |
| 
 | |
|         next_length = length - mask
 | |
|         if next_length:
 | |
|             part2 = self.checksum_mirror_sum(start[mask:], next_length, mask >> 1)
 | |
| 
 | |
|             while (next_length < mask):
 | |
|                 next_length += next_length
 | |
|                 part2 += part2
 | |
| 
 | |
|         return (part1 + part2) & 0xFFFF
 | |
| 
 | |
|     def write_bytes(self, buffer, startaddress: int, values):
 | |
|         buffer[startaddress:startaddress + len(values)] = values
 | |
| 
 | |
|     def write_crc(self, romName):
 | |
|         with open(romName, 'rb') as stream:
 | |
|             buffer = bytearray(stream.read())
 | |
|             crc = self.checksum_mirror_sum(buffer, len(buffer))
 | |
|             inv = crc ^ 0xFFFF
 | |
|             self.write_bytes(buffer, 0x7FDC, [inv & 0xFF, (inv >> 8) & 0xFF, crc & 0xFF, (crc >> 8) & 0xFF])
 | |
|         with open(romName, 'wb') as outfile:
 | |
|             outfile.write(buffer)
 | |
| 
 | |
|     def modify_multidata(self, multidata: dict):
 | |
|         # wait for self.rom_name to be available.
 | |
|         self.rom_name_available_event.wait()
 | |
|         rom_name = getattr(self, "rom_name", None)
 | |
|         # we skip in case of error, so that the original error in the output thread is the one that gets raised
 | |
|         if rom_name:
 | |
|             new_name = base64.b64encode(bytes(self.rom_name)).decode()
 | |
|             multidata["connect_names"][new_name] = multidata["connect_names"][self.multiworld.player_name[self.player]]
 | |
| 
 | |
|     def fill_slot_data(self): 
 | |
|         slot_data = {}
 | |
|         if not self.multiworld.is_race:
 | |
|             slot_data = self.options.as_dict(*self.options_dataclass.type_hints)
 | |
|             slot_data["Preset"] = { "Knows": {},
 | |
|                                     "Settings": {"hardRooms": Settings.SettingsDict[self.player].hardRooms,
 | |
|                                                  "bossesDifficulty": Settings.SettingsDict[self.player].bossesDifficulty,
 | |
|                                                  "hellRuns": Settings.SettingsDict[self.player].hellRuns},
 | |
|                                     "Controller": Controller.ControllerDict[self.player].__dict__}
 | |
| 
 | |
|             for knows in Knows.__dict__:
 | |
|                 if isKnows(knows):
 | |
|                     slot_data["Preset"]["Knows"][knows] = [ getattr(Knows.knowsDict[self.player], knows).bool, 
 | |
|                                                             getattr(Knows.knowsDict[self.player], knows).difficulty]
 | |
| 
 | |
|             slot_data["InterAreaTransitions"] = {}
 | |
|             for src, dest in self.variaRando.randoExec.areaGraph.InterAreaTransitions:
 | |
|                 slot_data["InterAreaTransitions"][src.Name] = dest.Name
 | |
|                 
 | |
|             slot_data["Doors"] = {}
 | |
|             for door in DoorsManager.doorsDict[self.player].values():
 | |
|                 slot_data["Doors"][door.name] = door.getColor()
 | |
| 
 | |
|             slot_data["RomPatches"] = RomPatches.ActivePatches[self.player]
 | |
|                 
 | |
|         return slot_data
 | |
| 
 | |
|     def write_spoiler(self, spoiler_handle: TextIO):
 | |
|         if self.options.area_randomization.value != 0:
 | |
|             spoiler_handle.write('\n\nArea Transitions:\n\n')
 | |
|             spoiler_handle.write('\n'.join(['%s%s %s %s' % (f'{self.multiworld.get_player_name(self.player)}: '
 | |
|                                                             if self.multiworld.players > 1 else '', src.Name,
 | |
|                                                             '<=>',
 | |
|                                                             dest.Name) for src, dest in self.variaRando.randoExec.areaGraph.InterAreaTransitions if not src.Boss]))
 | |
| 
 | |
|         if self.options.boss_randomization.value != 0:
 | |
|             spoiler_handle.write('\n\nBoss Transitions:\n\n')
 | |
|             spoiler_handle.write('\n'.join(['%s%s %s %s' % (f'{self.multiworld.get_player_name(self.player)}: '
 | |
|                                                             if self.multiworld.players > 1 else '', src.Name,
 | |
|                                                             '<=>',
 | |
|                                                             dest.Name) for src, dest in self.variaRando.randoExec.areaGraph.InterAreaTransitions if src.Boss]))
 | |
| 
 | |
| class SMLocation(Location):
 | |
|     game: str = "Super Metroid"
 | |
| 
 | |
|     def __init__(self, player: int, name: str, address=None, parent=None):
 | |
|         super(SMLocation, self).__init__(player, name, address, parent)
 | |
| 
 | |
| class SMItem(Item):
 | |
|     game = "Super Metroid"
 | |
|     type: str
 | |
| 
 | |
|     def __init__(self, name, classification, type: str, code, player: int):
 | |
|         super(SMItem, self).__init__(name, classification, code, player)
 | |
|         self.type = type
 |