mirror of
https://gitlab.com/xCrystal/pokecrystal-board.git
synced 2025-04-09 05:44:44 -07:00
Conflicts: constants.asm extras/crystal.py main.asm
This commit is contained in:
commit
7df002c3e2
21
.gitignore
vendored
21
.gitignore
vendored
@ -1,30 +1,31 @@
|
||||
#precompiled python
|
||||
# precompiled python
|
||||
*.pyc
|
||||
|
||||
#compiled object file
|
||||
# compiled object file
|
||||
*.o
|
||||
|
||||
#no binaries
|
||||
# no binaries
|
||||
*.exe
|
||||
|
||||
#roms
|
||||
# roms
|
||||
*.gbc
|
||||
*.gb
|
||||
|
||||
#generated
|
||||
# generated
|
||||
*.tx
|
||||
|
||||
#swap files for vim
|
||||
# swap files for vim
|
||||
.*.swp
|
||||
|
||||
#no data from extras/
|
||||
# no data from extras/
|
||||
extras/*.json
|
||||
|
||||
#for any of the poor souls with save game files in their working directory
|
||||
# for any of the poor souls with save game files in their working directory
|
||||
baserom.sgm
|
||||
baserom.sav
|
||||
pokered.sgm
|
||||
pokered.sav
|
||||
|
||||
#for vim configuration
|
||||
#url: http://www.vim.org/scripts/script.php?script_id=441
|
||||
# for vim configuration
|
||||
# url: http://www.vim.org/scripts/script.php?script_id=441
|
||||
.lvimrc
|
||||
|
9
Makefile
9
Makefile
@ -1,10 +1,15 @@
|
||||
.SUFFIXES: .asm .tx .o .gbc
|
||||
|
||||
TEXTFILES = text/sweethoney.tx
|
||||
TEXTFILES = text/sweethoney.tx \
|
||||
text/phone/bill.tx \
|
||||
text/phone/elm.tx \
|
||||
text/phone/mom.tx \
|
||||
text/phone/trainers1.tx \
|
||||
main.tx
|
||||
|
||||
all: pokecrystal.gbc
|
||||
|
||||
pokecrystal.o: pokecrystal.asm main.tx constants.asm wram.asm ${TEXTFILES}
|
||||
pokecrystal.o: pokecrystal.asm constants.asm wram.asm ${TEXTFILES}
|
||||
rgbasm -o pokecrystal.o pokecrystal.asm
|
||||
|
||||
.asm.tx:
|
||||
|
@ -1,5 +1,8 @@
|
||||
_CRYSTAL EQU 1
|
||||
|
||||
FarCall EQU $08
|
||||
Bankswitch EQU $10
|
||||
|
||||
dwb: MACRO
|
||||
dw \1
|
||||
db \2
|
||||
@ -21,13 +24,18 @@ bigdw: MACRO
|
||||
callab: MACRO
|
||||
ld hl, \1
|
||||
ld a, BANK(\1)
|
||||
rst $08
|
||||
rst FarCall
|
||||
ENDM
|
||||
|
||||
callba: MACRO
|
||||
ld a, BANK(\1)
|
||||
ld hl, \1
|
||||
rst $08
|
||||
rst FarCall
|
||||
ENDM
|
||||
|
||||
TX_RAM: MACRO
|
||||
db 1
|
||||
dw \1
|
||||
ENDM
|
||||
|
||||
TX_FAR: MACRO
|
||||
@ -36,6 +44,10 @@ TX_FAR: MACRO
|
||||
db BANK(\1)
|
||||
ENDM
|
||||
|
||||
RGB: MACRO
|
||||
dw ((\3 << 10) | (\2 << 5) | (\1))
|
||||
ENDM
|
||||
|
||||
; eventually replace with python macro
|
||||
note: MACRO
|
||||
db \1
|
||||
@ -3360,3 +3372,20 @@ Unkn1Pals EQU $d000 ; 8 4-color palettes little endian)
|
||||
Unkn2Pals EQU $d040 ; 8 4-color palettes little endian)
|
||||
BGPals EQU $d080 ; 8 4-color palettes little endian)
|
||||
OBPals EQU $d0c0 ; 8 4-color palettes little endian)
|
||||
|
||||
; oh my god this is hacky stop being so hacky
|
||||
frame: MACRO
|
||||
db \1
|
||||
db \2
|
||||
ENDM
|
||||
setrepeat: MACRO
|
||||
db $fe
|
||||
db \1
|
||||
ENDM
|
||||
dorepeat: MACRO
|
||||
db $fd
|
||||
db \1
|
||||
ENDM
|
||||
endanim: MACRO
|
||||
db $ff
|
||||
ENDM
|
||||
|
@ -30,9 +30,8 @@ After running those lines, `cp extras/output.txt main.asm` and run `git diff mai
|
||||
|
||||
Unit tests cover most of the classes.
|
||||
|
||||
```python
|
||||
import crystal
|
||||
crystal.run_tests()
|
||||
```bash
|
||||
python tests.py
|
||||
```
|
||||
|
||||
#### Parsing a script at a known address
|
||||
|
@ -1,34 +1,30 @@
|
||||
#!/usr/bin/python
|
||||
# -*- coding: utf-8 -*-
|
||||
# author: Bryan Bishop <kanzure@gmail.com>
|
||||
# date: 2012-05-29
|
||||
# purpose: find shared functions between red/crystal
|
||||
"""
|
||||
Finds shared functions between red/crystal.
|
||||
"""
|
||||
|
||||
from crystal import get_label_from_line, \
|
||||
get_address_from_line_comment, \
|
||||
AsmSection
|
||||
from crystal import (
|
||||
get_label_from_line,
|
||||
get_address_from_line_comment,
|
||||
AsmSection,
|
||||
direct_load_rom,
|
||||
direct_load_asm,
|
||||
)
|
||||
|
||||
from romstr import RomStr, AsmList
|
||||
from romstr import (
|
||||
RomStr,
|
||||
AsmList,
|
||||
)
|
||||
|
||||
def load_rom(path):
|
||||
""" Loads a ROM file into an abbreviated RomStr object.
|
||||
"""
|
||||
|
||||
fh = open(path, "r")
|
||||
x = RomStr(fh.read())
|
||||
fh.close()
|
||||
|
||||
return x
|
||||
return direct_load_rom(filename=path)
|
||||
|
||||
def load_asm(path):
|
||||
""" Loads source ASM into an abbreviated AsmList object.
|
||||
"""
|
||||
|
||||
fh = open(path, "r")
|
||||
x = AsmList(fh.read().split("\n"))
|
||||
fh.close()
|
||||
|
||||
return x
|
||||
return direct_load_asm(filename=path)
|
||||
|
||||
def findall_iter(sub, string):
|
||||
# url: http://stackoverflow.com/a/3874760/687783
|
||||
|
1412
extras/crystal.py
1412
extras/crystal.py
File diff suppressed because it is too large
Load Diff
14
extras/dump_sections
Executable file
14
extras/dump_sections
Executable file
@ -0,0 +1,14 @@
|
||||
#!/bin/bash
|
||||
# This wraps dump_sections.py so that other python scripts can import the
|
||||
# functions. If dump_sections.py was instead called dump_sections, then other
|
||||
# python source code would be unable to use the functions via import
|
||||
# statements.
|
||||
|
||||
# figure out the path to this script
|
||||
cwd="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )"
|
||||
|
||||
# construct the path to dump_sections.py
|
||||
secpath=$cwd/dump_sections.py
|
||||
|
||||
# run dump_sections.py
|
||||
$secpath $1
|
130
extras/dump_sections.py
Executable file
130
extras/dump_sections.py
Executable file
@ -0,0 +1,130 @@
|
||||
#!/usr/bin/python
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
Use this tool to dump an asm file for a new source code or disassembly project.
|
||||
|
||||
usage:
|
||||
|
||||
from dump_sections import dump_sections
|
||||
|
||||
output = dump_sections("../../butt.gbc")
|
||||
|
||||
file_handler = open("main.asm", "w")
|
||||
file_handler.write(output)
|
||||
file_handler.close()
|
||||
|
||||
You can also use this script from the shell, where it will look for
|
||||
"baserom.gbc" in the current working path or whatever file path you pass in the
|
||||
first positional argument.
|
||||
"""
|
||||
|
||||
import os
|
||||
import sys
|
||||
import argparse
|
||||
|
||||
def upper_hex(input):
|
||||
"""
|
||||
Converts the input to an uppercase hex string.
|
||||
"""
|
||||
if input in [0, "0"]:
|
||||
return "0"
|
||||
elif input <= 0xF:
|
||||
return ("%.x" % (input)).upper()
|
||||
else:
|
||||
return ("%.2x" % (input)).upper()
|
||||
|
||||
def format_bank_number(address, bank_size=0x4000):
|
||||
"""
|
||||
Returns a str of the hex number of the bank based on the address.
|
||||
"""
|
||||
return upper_hex(address / bank_size)
|
||||
|
||||
def calculate_bank_quantity(path, bank_size=0x4000):
|
||||
"""
|
||||
Returns the number of 0x4000 banks in the file at path.
|
||||
"""
|
||||
return float(os.lstat(path).st_size) / bank_size
|
||||
|
||||
def dump_section(bank_number, separator="\n\n"):
|
||||
"""
|
||||
Returns a str of a section header for the asm file.
|
||||
"""
|
||||
output = "SECTION \""
|
||||
if bank_number in [0, "0"]:
|
||||
output += "bank0\",HOME"
|
||||
else:
|
||||
output += "bank"
|
||||
output += bank_number
|
||||
output += "\",DATA,BANK[$"
|
||||
output += bank_number
|
||||
output += "]"
|
||||
output += separator
|
||||
return output
|
||||
|
||||
def dump_incbin_for_section(address, bank_size=0x4000, baserom="baserom.gbc", separator="\n\n"):
|
||||
"""
|
||||
Returns a str for an INCBIN line for an entire section.
|
||||
"""
|
||||
output = "INCBIN \""
|
||||
output += baserom
|
||||
output += "\",$"
|
||||
output += upper_hex(address)
|
||||
output += ",$"
|
||||
output += upper_hex(bank_size)
|
||||
output += separator
|
||||
return output
|
||||
|
||||
def dump_sections(path, bank_size=0x4000, initial_bank=0, last_bank=None, separator="\n\n"):
|
||||
"""
|
||||
Returns a str of assembly source code. The source code delineates each
|
||||
SECTION and includes bytes from the file specified by baserom.
|
||||
"""
|
||||
if not last_bank:
|
||||
last_bank = calculate_bank_quantity(path, bank_size=bank_size)
|
||||
|
||||
if last_bank < initial_bank:
|
||||
raise Exception("last_bank must be greater than or equal to initial_bank")
|
||||
|
||||
baserom_name = os.path.basename(path)
|
||||
|
||||
output = ""
|
||||
|
||||
banks = range(initial_bank, last_bank)
|
||||
|
||||
for bank_number in banks:
|
||||
address = bank_number * bank_size
|
||||
|
||||
# get a formatted hex number of the bank based on the address
|
||||
formatted_bank_number = format_bank_number(address, bank_size=bank_size)
|
||||
|
||||
# SECTION
|
||||
output += dump_section(formatted_bank_number, separator=separator)
|
||||
|
||||
# INCBIN a range of bytes from the ROM
|
||||
output += dump_incbin_for_section(address, bank_size=bank_size, baserom=baserom_name, separator=separator)
|
||||
|
||||
# clean up newlines at the end of the output
|
||||
if output[-2:] == "\n\n":
|
||||
output = output[:-2]
|
||||
output += "\n"
|
||||
|
||||
return output
|
||||
|
||||
if __name__ == "__main__":
|
||||
parser = argparse.ArgumentParser()
|
||||
parser.add_argument("rompath", nargs="?", metavar="rompath", type=str)
|
||||
args = parser.parse_args()
|
||||
|
||||
# default to "baserom.gbc" in the current working directory
|
||||
baserom = "baserom.gbc"
|
||||
|
||||
# but let the user override the default
|
||||
if args.rompath:
|
||||
baserom = args.rompath
|
||||
|
||||
# generate some asm
|
||||
output = dump_sections(baserom)
|
||||
|
||||
# dump everything to stdout
|
||||
sys.stdout.write(output)
|
||||
|
@ -1,26 +1,28 @@
|
||||
#author: Bryan Bishop <kanzure@gmail.com>
|
||||
#date: 2012-01-09
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
import os
|
||||
import sys
|
||||
from copy import copy, deepcopy
|
||||
from ctypes import c_int8
|
||||
import json
|
||||
import random
|
||||
import json
|
||||
|
||||
spacing = "\t"
|
||||
# New versions of json don't have read anymore.
|
||||
if not hasattr(json, "read"):
|
||||
json.read = json.loads
|
||||
|
||||
class XRomStr(str):
|
||||
def __repr__(self):
|
||||
return "RomStr(too long)"
|
||||
from romstr import RomStr
|
||||
|
||||
def load_rom(filename="../baserom.gbc"):
|
||||
"""loads bytes into memory"""
|
||||
global rom
|
||||
file_handler = open(filename, "rb")
|
||||
rom = XRomStr(file_handler.read())
|
||||
file_handler = open(filename, "rb")
|
||||
rom = RomStr(file_handler.read())
|
||||
file_handler.close()
|
||||
return rom
|
||||
|
||||
spacing = "\t"
|
||||
|
||||
temp_opt_table = [
|
||||
[ "ADC A", 0x8f, 0 ],
|
||||
[ "ADC B", 0x88, 0 ],
|
||||
@ -550,7 +552,7 @@ end_08_scripts_with = [
|
||||
0xc9, #ret
|
||||
###0xda, 0xe9, 0xd2, 0xc2, 0xca, 0xc3, 0x38, 0x30, 0x20, 0x28, 0x18, 0xd8, 0xd0, 0xc0, 0xc8, 0xc9
|
||||
]
|
||||
relative_jumps = [0x38, 0x30, 0x20, 0x28, 0x18, 0xc3, 0xda, 0xc2]
|
||||
relative_jumps = [0x38, 0x30, 0x20, 0x28, 0x18, 0xc3, 0xda, 0xc2]
|
||||
relative_unconditional_jumps = [0xc3, 0x18]
|
||||
|
||||
call_commands = [0xdc, 0xd4, 0xc4, 0xcc, 0xcd]
|
||||
@ -559,7 +561,7 @@ all_labels = {}
|
||||
def load_labels(filename="labels.json"):
|
||||
global all_labels
|
||||
if os.path.exists(filename):
|
||||
all_labels = json.loads(open(filename, "r").read())
|
||||
all_labels = json.read(open(filename, "r").read())
|
||||
else:
|
||||
print "You must run crystal.scan_for_predefined_labels() to create \"labels.json\". Trying..."
|
||||
import crystal
|
||||
@ -601,10 +603,10 @@ def output_bank_opcodes(original_offset, max_byte_count=0x4000, debug = False):
|
||||
#i = offset
|
||||
#ad = end_address
|
||||
#a, oa = current_byte_number
|
||||
|
||||
|
||||
load_labels()
|
||||
load_rom()
|
||||
|
||||
|
||||
bank_id = 0
|
||||
if original_offset > 0x8000:
|
||||
bank_id = original_offset / 0x4000
|
||||
|
@ -1043,14 +1043,16 @@ def decompress_monsters(type = front):
|
||||
# decompress
|
||||
monster = decompress_monster_by_id(id, type)
|
||||
if monster != None: # no unowns here
|
||||
filename = str(id+1).zfill(3) + '.2bpp' # 001.2bpp
|
||||
if not type: # front
|
||||
folder = '../gfx/frontpics/'
|
||||
filename = 'front.2bpp'
|
||||
folder = '../gfx/pics/' + str(id+1).zfill(3) + '/'
|
||||
to_file(folder+filename, monster.pic)
|
||||
folder = '../gfx/anim/'
|
||||
filename = 'tiles.2bpp'
|
||||
folder = '../gfx/pics/' + str(id+1).zfill(3) + '/'
|
||||
to_file(folder+filename, monster.animtiles)
|
||||
else: # back
|
||||
folder = '../gfx/backpics/'
|
||||
filename = 'back.2bpp'
|
||||
folder = '../gfx/pics/' + str(id+1).zfill(3) + '/'
|
||||
to_file(folder+filename, monster.pic)
|
||||
|
||||
|
||||
@ -1073,14 +1075,16 @@ def decompress_unowns(type = front):
|
||||
# decompress
|
||||
unown = decompress_unown_by_id(letter, type)
|
||||
|
||||
filename = str(unown_dex).zfill(3) + chr(ord('a') + letter) + '.2bpp' # 201a.2bpp
|
||||
if not type: # front
|
||||
folder = '../gfx/frontpics/'
|
||||
filename = 'front.2bpp'
|
||||
folder = '../gfx/pics/' + str(unown_dex).zfill(3) + chr(ord('a') + letter) + '/'
|
||||
to_file(folder+filename, unown.pic)
|
||||
filename = 'tiles.2bpp'
|
||||
folder = '../gfx/anim/'
|
||||
to_file(folder+filename, unown.animtiles)
|
||||
else: # back
|
||||
folder = '../gfx/backpics/'
|
||||
filename = 'back.2bpp'
|
||||
folder = '../gfx/pics/' + str(unown_dex).zfill(3) + chr(ord('a') + letter) + '/'
|
||||
to_file(folder+filename, unown.pic)
|
||||
|
||||
|
||||
@ -1255,8 +1259,8 @@ def compress_file(filein, fileout, mode = 'horiz'):
|
||||
def compress_monster_frontpic(id, fileout):
|
||||
mode = 'vert'
|
||||
|
||||
fpic = '../gfx/frontpics/' + str(id).zfill(3) + '.2bpp'
|
||||
fanim = '../gfx/anim/' + str(id).zfill(3) + '.2bpp'
|
||||
fpic = '../gfx/pics/' + str(id).zfill(3) + '/front.2bpp'
|
||||
fanim = '../gfx/pics/' + str(id).zfill(3) + '/tiles.2bpp'
|
||||
|
||||
pic = open(fpic, 'rb').read()
|
||||
anim = open(fanim, 'rb').read()
|
||||
@ -1264,7 +1268,7 @@ def compress_monster_frontpic(id, fileout):
|
||||
|
||||
lz = Compressed(image, mode, 5)
|
||||
|
||||
out = '../gfx/frontpics/lz/' + str(id).zfill(3) + '.lz'
|
||||
out = '../gfx/pics/' + str(id).zfill(3) + '/front.lz'
|
||||
|
||||
to_file(out, lz.output)
|
||||
|
||||
@ -1283,6 +1287,28 @@ def get_uncompressed_gfx(start, num_tiles, filename):
|
||||
|
||||
|
||||
|
||||
def hex_to_rgb(word):
|
||||
red = word & 0b11111
|
||||
word >>= 5
|
||||
green = word & 0b11111
|
||||
word >>= 5
|
||||
blue = word & 0b11111
|
||||
return (red, green, blue)
|
||||
|
||||
def grab_palettes(address, length = 0x80):
|
||||
output = ''
|
||||
for word in range(length/2):
|
||||
color = ord(rom[address+1])*0x100 + ord(rom[address])
|
||||
address += 2
|
||||
color = hex_to_rgb(color)
|
||||
red = str(color[0]).zfill(2)
|
||||
green = str(color[1]).zfill(2)
|
||||
blue = str(color[2]).zfill(2)
|
||||
output += '\tRGB '+red+', '+green+', '+blue
|
||||
output += '\n'
|
||||
return output
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
parser = argparse.ArgumentParser()
|
||||
parser.add_argument('cmd', nargs='?', metavar='cmd', type=str)
|
||||
@ -1317,7 +1343,11 @@ if __name__ == "__main__":
|
||||
# python gfx.py un [address] [num_tiles] [filename]
|
||||
get_uncompressed_gfx(int(args.arg1,16), int(args.arg2), args.arg3)
|
||||
|
||||
else:
|
||||
# python gfx.py
|
||||
decompress_all()
|
||||
if debug: print 'decompressed known gfx to ../gfx/!'
|
||||
elif args.cmd == 'pal':
|
||||
# python gfx.py pal [address] [length]
|
||||
print grab_palettes(int(args.arg1,16), int(args.arg2))
|
||||
|
||||
#else:
|
||||
## python gfx.py
|
||||
#decompress_all()
|
||||
#if debug: print 'decompressed known gfx to ../gfx/!'
|
||||
|
@ -1,12 +1,13 @@
|
||||
#!/usr/bin/python
|
||||
# author: Bryan Bishop <kanzure@gmail.com>
|
||||
# date: 2012-06-20
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
import networkx as nx
|
||||
|
||||
from romstr import RomStr, DisAsm, \
|
||||
relative_jumps, call_commands, \
|
||||
relative_unconditional_jumps
|
||||
from romstr import (
|
||||
RomStr,
|
||||
relative_jumps,
|
||||
call_commands,
|
||||
relative_unconditional_jumps,
|
||||
)
|
||||
|
||||
class RomGraph(nx.DiGraph):
|
||||
""" Graphs various functions pointing to each other.
|
||||
|
104
extras/interval_map.py
Normal file
104
extras/interval_map.py
Normal file
@ -0,0 +1,104 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
from bisect import bisect_left, bisect_right
|
||||
from itertools import izip
|
||||
|
||||
class IntervalMap(object):
|
||||
"""
|
||||
This class maps a set of intervals to a set of values.
|
||||
|
||||
>>> i = IntervalMap()
|
||||
>>> i[0:5] = "hello world"
|
||||
>>> i[6:10] = "hello cruel world"
|
||||
>>> print i[4]
|
||||
"hello world"
|
||||
"""
|
||||
|
||||
def __init__(self):
|
||||
"""initializes an empty IntervalMap"""
|
||||
self._bounds = []
|
||||
self._items = []
|
||||
self._upperitem = None
|
||||
|
||||
def __setitem__(self, _slice, _value):
|
||||
"""sets an interval mapping"""
|
||||
assert isinstance(_slice, slice), 'The key must be a slice object'
|
||||
|
||||
if _slice.start is None:
|
||||
start_point = -1
|
||||
else:
|
||||
start_point = bisect_left(self._bounds, _slice.start)
|
||||
|
||||
if _slice.stop is None:
|
||||
end_point = -1
|
||||
else:
|
||||
end_point = bisect_left(self._bounds, _slice.stop)
|
||||
|
||||
if start_point>=0:
|
||||
if start_point < len(self._bounds) and self._bounds[start_point]<_slice.start:
|
||||
start_point += 1
|
||||
|
||||
if end_point>=0:
|
||||
self._bounds[start_point:end_point] = [_slice.start, _slice.stop]
|
||||
if start_point < len(self._items):
|
||||
self._items[start_point:end_point] = [self._items[start_point], _value]
|
||||
else:
|
||||
self._items[start_point:end_point] = [self._upperitem, _value]
|
||||
else:
|
||||
self._bounds[start_point:] = [_slice.start]
|
||||
if start_point < len(self._items):
|
||||
self._items[start_point:] = [self._items[start_point], _value]
|
||||
else:
|
||||
self._items[start_point:] = [self._upperitem]
|
||||
self._upperitem = _value
|
||||
else:
|
||||
if end_point>=0:
|
||||
self._bounds[:end_point] = [_slice.stop]
|
||||
self._items[:end_point] = [_value]
|
||||
else:
|
||||
self._bounds[:] = []
|
||||
self._items[:] = []
|
||||
self._upperitem = _value
|
||||
|
||||
def __getitem__(self,_point):
|
||||
"""gets a value from the mapping"""
|
||||
assert not isinstance(_point, slice), 'The key cannot be a slice object'
|
||||
|
||||
index = bisect_right(self._bounds, _point)
|
||||
if index < len(self._bounds):
|
||||
return self._items[index]
|
||||
else:
|
||||
return self._upperitem
|
||||
|
||||
def items(self):
|
||||
"""returns an iterator with each item being
|
||||
((low_bound, high_bound), value)
|
||||
these items are returned in order"""
|
||||
previous_bound = None
|
||||
for (b, v) in izip(self._bounds, self._items):
|
||||
if v is not None:
|
||||
yield (previous_bound, b), v
|
||||
previous_bound = b
|
||||
if self._upperitem is not None:
|
||||
yield (previous_bound, None), self._upperitem
|
||||
|
||||
def values(self):
|
||||
"""returns an iterator with each item being a stored value
|
||||
the items are returned in order"""
|
||||
for v in self._items:
|
||||
if v is not None:
|
||||
yield v
|
||||
if self._upperitem is not None:
|
||||
yield self._upperitem
|
||||
|
||||
def __repr__(self):
|
||||
s = []
|
||||
for b,v in self.items():
|
||||
if v is not None:
|
||||
s.append('[%r, %r] => %r'%(
|
||||
b[0],
|
||||
b[1],
|
||||
v
|
||||
))
|
||||
return '{'+', '.join(s)+'}'
|
||||
|
@ -1,4 +1,7 @@
|
||||
item_constants = {1: 'MASTER_BALL',
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
item_constants = {
|
||||
1: 'MASTER_BALL',
|
||||
2: 'ULTRA_BALL',
|
||||
3: 'BRIGHTPOWDER',
|
||||
4: 'GREAT_BALL',
|
||||
@ -219,4 +222,20 @@ item_constants = {1: 'MASTER_BALL',
|
||||
246: 'HM_04',
|
||||
247: 'HM_05',
|
||||
248: 'HM_06',
|
||||
249: 'HM_07'}
|
||||
249: 'HM_07',
|
||||
}
|
||||
|
||||
def find_item_label_by_id(id):
|
||||
if id in item_constants.keys():
|
||||
return item_constants[id]
|
||||
else: return None
|
||||
|
||||
def generate_item_constants():
|
||||
"""make a list of items to put in constants.asm"""
|
||||
output = ""
|
||||
for (id, item) in item_constants.items():
|
||||
val = ("$%.2x"%id).upper()
|
||||
while len(item)<13: item+= " "
|
||||
output += item + " EQU " + val + "\n"
|
||||
return output
|
||||
|
||||
|
@ -1,7 +1,12 @@
|
||||
""" Various label/line-related functions.
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
Various label/line-related functions.
|
||||
"""
|
||||
|
||||
from pointers import calculate_pointer, calculate_bank
|
||||
from pointers import (
|
||||
calculate_pointer,
|
||||
calculate_bank,
|
||||
)
|
||||
|
||||
def remove_quoted_text(line):
|
||||
"""get rid of content inside quotes
|
||||
|
@ -1,3 +1,5 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
moves = {
|
||||
0x01: "POUND",
|
||||
0x02: "KARATE_CHOP",
|
||||
|
@ -1,3 +1,4 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
pksv_gs = {
|
||||
0x00: "2call",
|
||||
@ -34,7 +35,7 @@ pksv_gs = {
|
||||
0x21: "checkitem",
|
||||
0x22: "givemoney",
|
||||
0x23: "takemoney",
|
||||
0x24: "checkmonkey",
|
||||
0x24: "checkmoney",
|
||||
0x25: "givecoins",
|
||||
0x26: "takecoins",
|
||||
0x27: "checkcoins",
|
||||
@ -141,8 +142,8 @@ pksv_gs = {
|
||||
0xA3: "displaylocation",
|
||||
}
|
||||
|
||||
#see http://www.pokecommunity.com/showpost.php?p=4347261
|
||||
#NOTE: this has some updates that need to be back-ported to gold
|
||||
# see http://www.pokecommunity.com/showpost.php?p=4347261
|
||||
# NOTE: this has some updates that need to be back-ported to gold
|
||||
pksv_crystal = {
|
||||
0x00: "2call",
|
||||
0x01: "3call",
|
||||
@ -179,7 +180,7 @@ pksv_crystal = {
|
||||
0x21: "checkitem",
|
||||
0x22: "givemoney",
|
||||
0x23: "takemoney",
|
||||
0x24: "checkmonkey",
|
||||
0x24: "checkmoney",
|
||||
0x25: "givecoins",
|
||||
0x26: "takecoins",
|
||||
0x27: "checkcoins",
|
||||
@ -292,13 +293,14 @@ pksv_crystal = {
|
||||
}
|
||||
|
||||
#these cause the script to end; used in create_command_classes
|
||||
pksv_crystal_more_enders = [0x03, 0x04, 0x05, 0x0C, 0x51, 0x53,
|
||||
0x8D, 0x8F, 0x90, 0x91, 0x92, 0x9B,
|
||||
pksv_crystal_more_enders = [0x03, 0x04, 0x05, 0x0C, 0x51, 0x52,
|
||||
0x53, 0x8D, 0x8F, 0x90, 0x91, 0x92,
|
||||
0x9B,
|
||||
0xB2, #maybe?
|
||||
0xCC, #maybe?
|
||||
]
|
||||
|
||||
#these have no pksv names as of pksv 2.1.1
|
||||
# these have no pksv names as of pksv 2.1.1
|
||||
pksv_crystal_unknowns = [
|
||||
0x9F,
|
||||
0xA6, 0xA7, 0xA8, 0xA9, 0xAA, 0xAB, 0xAC, 0xAD, 0xAE, 0xAF,
|
||||
|
@ -1,10 +1,12 @@
|
||||
""" Various functions related to pointer and address math. Mostly to avoid
|
||||
depedency loops.
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
Various functions related to pointer and address math. Mostly to avoid
|
||||
depedency loops.
|
||||
"""
|
||||
|
||||
def calculate_bank(address):
|
||||
"""you are too lazy to divide on your own?"""
|
||||
if type(address) == str:
|
||||
if type(address) == str:
|
||||
address = int(address, 16)
|
||||
#if 0x4000 <= address <= 0x7FFF:
|
||||
# raise Exception, "bank 1 does not exist"
|
||||
|
@ -1,3 +1,5 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
pokemon_constants = {
|
||||
1: "BULBASAUR",
|
||||
2: "IVYSAUR",
|
||||
|
354
extras/romstr.py
354
extras/romstr.py
@ -1,8 +1,21 @@
|
||||
import sys, os, time, datetime, json
|
||||
from gbz80disasm import opt_table
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
import sys
|
||||
import os
|
||||
import time
|
||||
import datetime
|
||||
from ctypes import c_int8
|
||||
from copy import copy, deepcopy
|
||||
from labels import get_label_from_line, get_address_from_line_comment
|
||||
from copy import copy
|
||||
import json
|
||||
|
||||
# New versions of json don't have read anymore.
|
||||
if not hasattr(json, "read"):
|
||||
json.read = json.loads
|
||||
|
||||
from labels import (
|
||||
get_label_from_line,
|
||||
get_address_from_line_comment,
|
||||
)
|
||||
|
||||
relative_jumps = [0x38, 0x30, 0x20, 0x28, 0x18, 0xc3, 0xda, 0xc2, 0x32]
|
||||
relative_unconditional_jumps = [0xc3, 0x18]
|
||||
@ -91,7 +104,7 @@ class RomStr(str):
|
||||
file_handler.close()
|
||||
|
||||
# load the labels from the file
|
||||
self.labels = json.loads(open(filename, "r").read())
|
||||
self.labels = json.read(open(filename, "r").read())
|
||||
|
||||
def get_address_for(self, label):
|
||||
""" Returns the address of a label. This is slow and could be improved
|
||||
@ -137,7 +150,7 @@ class RomStr(str):
|
||||
that will be parsed, so that large patches of data aren't parsed as
|
||||
code.
|
||||
"""
|
||||
if type(address) == str and "0x" in address:
|
||||
if type(address) in [str, unicode] and "0x" in address:
|
||||
address = int(address, 16)
|
||||
|
||||
start_address = address
|
||||
@ -166,333 +179,8 @@ class RomStr(str):
|
||||
elif end_address != None and size == None:
|
||||
size = end_address - start_address
|
||||
|
||||
return DisAsm(start_address=start_address, end_address=end_address, size=size, max_size=max_size, debug=debug, rom=self)
|
||||
|
||||
class DisAsm:
|
||||
""" z80 disassembler
|
||||
"""
|
||||
|
||||
def __init__(self, start_address=None, end_address=None, size=None, max_size=0x4000, debug=True, rom=None):
|
||||
assert start_address != None, "start_address must be given"
|
||||
|
||||
if rom == None:
|
||||
file_handler = open("../baserom.gbc", "r")
|
||||
bytes = file_handler.read()
|
||||
file_handler.close()
|
||||
rom = RomStr(bytes)
|
||||
|
||||
if debug not in [None, True, False]:
|
||||
raise Exception, "debug param is invalid"
|
||||
if debug == None:
|
||||
debug = False
|
||||
|
||||
# get end_address and size in sync with each other
|
||||
if end_address == None and size != None:
|
||||
end_address = start_address + size
|
||||
elif end_address != None and size == None:
|
||||
size = end_address - start_address
|
||||
elif end_address != None and size != None:
|
||||
size = max(end_address - start_address, size)
|
||||
end_address = start_address + size
|
||||
|
||||
# check that the bounds make sense
|
||||
if end_address != None:
|
||||
if end_address <= start_address:
|
||||
raise Exception, "end_address is out of bounds"
|
||||
elif (end_address - start_address) > max_size:
|
||||
raise Exception, "end_address goes beyond max_size"
|
||||
|
||||
# check more edge cases
|
||||
if not start_address >= 0:
|
||||
raise Exception, "start_address must be at least 0"
|
||||
elif end_address != None and not end_address >= 0:
|
||||
raise Exception, "end_address must be at least 0"
|
||||
|
||||
self.rom = rom
|
||||
self.start_address = start_address
|
||||
self.end_address = end_address
|
||||
self.size = size
|
||||
self.max_size = max_size
|
||||
self.debug = debug
|
||||
|
||||
self.parse()
|
||||
|
||||
def parse(self):
|
||||
""" Disassembles stuff and things.
|
||||
"""
|
||||
|
||||
rom = self.rom
|
||||
start_address = self.start_address
|
||||
end_address = self.end_address
|
||||
max_size = self.max_size
|
||||
debug = self.debug
|
||||
|
||||
bank_id = start_address / 0x4000
|
||||
|
||||
# [{"command": 0x20, "bytes": [0x20, 0x40, 0x50],
|
||||
# "asm": "jp $5040", "label": "Unknown5040"}]
|
||||
asm_commands = {}
|
||||
|
||||
offset = start_address
|
||||
|
||||
last_hl_address = None
|
||||
last_a_address = None
|
||||
used_3d97 = False
|
||||
|
||||
keep_reading = True
|
||||
|
||||
while (end_address != 0 and offset <= end_address) or keep_reading:
|
||||
# read the current opcode byte
|
||||
current_byte = ord(rom[offset])
|
||||
current_byte_number = len(asm_commands.keys())
|
||||
|
||||
# setup this next/upcoming command
|
||||
if offset in asm_commands.keys():
|
||||
asm_command = asm_commands[offset]
|
||||
else:
|
||||
asm_command = {}
|
||||
|
||||
asm_command["address"] = offset
|
||||
|
||||
if not "references" in asm_command.keys():
|
||||
# This counts how many times relative jumps reference this
|
||||
# byte. This is used to determine whether or not to print out a
|
||||
# label later.
|
||||
asm_command["references"] = 0
|
||||
|
||||
# some commands have two opcodes
|
||||
next_byte = ord(rom[offset+1])
|
||||
|
||||
if self.debug:
|
||||
print "offset: \t\t" + hex(offset)
|
||||
print "current_byte: \t\t" + hex(current_byte)
|
||||
print "next_byte: \t\t" + hex(next_byte)
|
||||
|
||||
# all two-byte opcodes also have their first byte in there somewhere
|
||||
if (current_byte in opt_table.keys()) or ((current_byte + (next_byte << 8)) in opt_table.keys()):
|
||||
# this might be a two-byte opcode
|
||||
possible_opcode = current_byte + (next_byte << 8)
|
||||
|
||||
# check if this is a two-byte opcode
|
||||
if possible_opcode in opt_table.keys():
|
||||
op_code = possible_opcode
|
||||
else:
|
||||
op_code = current_byte
|
||||
|
||||
op = opt_table[op_code]
|
||||
|
||||
opstr = op[0].lower()
|
||||
optype = op[1]
|
||||
|
||||
if self.debug:
|
||||
print "opstr: " + opstr
|
||||
|
||||
asm_command["type"] = "op"
|
||||
asm_command["id"] = op_code
|
||||
asm_command["format"] = opstr
|
||||
asm_command["opnumberthing"] = optype
|
||||
|
||||
opstr2 = None
|
||||
base_opstr = copy(opstr)
|
||||
|
||||
if "x" in opstr:
|
||||
for x in range(0, opstr.count("x")):
|
||||
insertion = ord(rom[offset + 1])
|
||||
|
||||
# Certain opcodes will have a local relative jump label
|
||||
# here instead of a raw hex value, but this is
|
||||
# controlled through asm output.
|
||||
insertion = "$" + hex(insertion)[2:]
|
||||
|
||||
opstr = opstr[:opstr.find("x")].lower() + insertion + opstr[opstr.find("x")+1:].lower()
|
||||
|
||||
if op_code in relative_jumps:
|
||||
target_address = offset + 2 + c_int8(ord(rom[offset + 1])).value
|
||||
insertion = "asm_" + hex(target_address)
|
||||
|
||||
if str(target_address) in self.rom.labels.keys():
|
||||
insertion = self.rom.labels[str(target_address)]
|
||||
|
||||
opstr2 = base_opstr[:base_opstr.find("x")].lower() + insertion + base_opstr[base_opstr.find("x")+1:].lower()
|
||||
asm_command["formatted_with_labels"] = opstr2
|
||||
asm_command["target_address"] = target_address
|
||||
|
||||
current_byte_number += 1
|
||||
offset += 1
|
||||
|
||||
if "?" in opstr:
|
||||
for y in range(0, opstr.count("?")):
|
||||
byte1 = ord(rom[offset + 1])
|
||||
byte2 = ord(rom[offset + 2])
|
||||
|
||||
number = byte1
|
||||
number += byte2 << 8;
|
||||
|
||||
# In most cases, you can use a label here. Labels will
|
||||
# be shown during asm output.
|
||||
insertion = "$%.4x" % (number)
|
||||
|
||||
opstr = opstr[:opstr.find("?")].lower() + insertion + opstr[opstr.find("?")+1:].lower()
|
||||
|
||||
# This version of the formatted string has labels. In
|
||||
# the future, the actual labels should be parsed
|
||||
# straight out of the "main.asm" file.
|
||||
target_address = number % 0x4000
|
||||
insertion = "asm_" + hex(target_address)
|
||||
|
||||
if str(target_address) in self.rom.labels.keys():
|
||||
insertion = self.rom.labels[str(target_address)]
|
||||
|
||||
opstr2 = base_opstr[:base_opstr.find("?")].lower() + insertion + base_opstr[base_opstr.find("?")+1:].lower()
|
||||
asm_command["formatted_with_labels"] = opstr2
|
||||
asm_command["target_address"] = target_address
|
||||
|
||||
current_byte_number += 2
|
||||
offset += 2
|
||||
|
||||
# Check for relative jumps, construct the formatted asm line.
|
||||
# Also set the usage of labels.
|
||||
if current_byte in [0x18, 0x20] + relative_jumps: # jr or jr nz
|
||||
# generate a label for the byte we're jumping to
|
||||
target_address = offset + 1 + c_int8(ord(rom[offset])).value
|
||||
|
||||
if target_address in asm_commands.keys():
|
||||
asm_commands[target_address]["references"] += 1
|
||||
remote_label = "asm_" + hex(target_address)
|
||||
asm_commands[target_address]["current_label"] = remote_label
|
||||
asm_command["remote_label"] = remote_label
|
||||
|
||||
# Not sure how to set this, can't be True because an
|
||||
# address referenced multiple times will use a label
|
||||
# despite the label not necessarily being used in the
|
||||
# output. The "use_remote_label" values should be
|
||||
# calculated when rendering the asm output, based on
|
||||
# which addresses and which op codes will be displayed
|
||||
# (within the range).
|
||||
asm_command["use_remote_label"] = "unknown"
|
||||
else:
|
||||
remote_label = "asm_" + hex(target_address)
|
||||
|
||||
# This remote address might not be part of this
|
||||
# function.
|
||||
asm_commands[target_address] = {
|
||||
"references": 1,
|
||||
"current_label": remote_label,
|
||||
"address": target_address,
|
||||
}
|
||||
# Also, target_address can be negative (before the
|
||||
# start_address that the user originally requested),
|
||||
# and it shouldn't be shown on asm output because the
|
||||
# intermediate bytes (between a negative target_address
|
||||
# and start_address) won't be disassembled.
|
||||
|
||||
# Don't know yet if this remote address is part of this
|
||||
# function or not. When the remote address is not part
|
||||
# of this function, the label name should not be used,
|
||||
# because that label will not be disassembled in the
|
||||
# output, until the user asks it to.
|
||||
asm_command["use_remote_label"] = "unknown"
|
||||
asm_command["remote_label"] = remote_label
|
||||
elif current_byte == 0x3e:
|
||||
last_a_address = ord(rom[offset + 1])
|
||||
|
||||
# store the formatted string for the output later
|
||||
asm_command["formatted"] = opstr
|
||||
|
||||
if current_byte == 0x21:
|
||||
last_hl_address = byte1 + (byte2 << 8)
|
||||
|
||||
# this is leftover from pokered, might be meaningless
|
||||
if current_byte == 0xcd:
|
||||
if number == 0x3d97:
|
||||
used_3d97 = True
|
||||
|
||||
if current_byte == 0xc3 or current_byte in relative_unconditional_jumps:
|
||||
if current_byte == 0xc3:
|
||||
if number == 0x3d97:
|
||||
used_3d97 = True
|
||||
|
||||
# stop reading at a jump, relative jump or return
|
||||
if current_byte in end_08_scripts_with:
|
||||
is_data = False
|
||||
|
||||
if not self.has_outstanding_labels(asm_commands, offset):
|
||||
keep_reading = False
|
||||
break
|
||||
else:
|
||||
keep_reading = True
|
||||
else:
|
||||
keep_reading = True
|
||||
|
||||
else:
|
||||
# This shouldn't really happen, and means that this area of the
|
||||
# ROM probably doesn't represent instructions.
|
||||
asm_command["type"] = "data" # db
|
||||
asm_command["value"] = current_byte
|
||||
keep_reading = False
|
||||
|
||||
# save this new command in the list
|
||||
asm_commands[asm_command["address"]] = asm_command
|
||||
|
||||
# jump forward by a byte
|
||||
offset += 1
|
||||
|
||||
# also save the last command if necessary
|
||||
if len(asm_commands.keys()) > 0 and asm_commands[asm_commands.keys()[-1]] is not asm_command:
|
||||
asm_commands[asm_command["address"]] = asm_command
|
||||
|
||||
# store the set of commands on this object
|
||||
self.asm_commands = asm_commands
|
||||
|
||||
self.end_address = offset + 1
|
||||
self.last_address = self.end_address
|
||||
|
||||
def has_outstanding_labels(self, asm_commands, offset):
|
||||
""" Checks if there are any labels that haven't yet been created.
|
||||
""" # is this really necessary??
|
||||
return False
|
||||
|
||||
def used_addresses(self):
|
||||
""" Returns a list of unique addresses that this function will probably
|
||||
call.
|
||||
"""
|
||||
addresses = set()
|
||||
|
||||
for (id, command) in self.asm_commands.items():
|
||||
if command.has_key("target_address") and command["id"] in call_commands:
|
||||
addresses.add(command["target_address"])
|
||||
|
||||
return addresses
|
||||
|
||||
def __str__(self):
|
||||
""" ASM pretty printer.
|
||||
"""
|
||||
output = ""
|
||||
|
||||
for (key, line) in self.asm_commands.items():
|
||||
# skip anything from before the beginning
|
||||
if key < self.start_address:
|
||||
continue
|
||||
|
||||
# show a label
|
||||
if line["references"] > 0 and "current_label" in line.keys():
|
||||
if line["address"] == self.start_address:
|
||||
output += "thing: ; " + hex(line["address"]) + "\n"
|
||||
else:
|
||||
output += "." + line["current_label"] + "\@ ; " + hex(line["address"]) + "\n"
|
||||
|
||||
# show the actual line
|
||||
if line.has_key("formatted_with_labels"):
|
||||
output += spacing + line["formatted_with_labels"]
|
||||
elif line.has_key("formatted"):
|
||||
output += spacing + line["formatted"]
|
||||
#output += " ; to " +
|
||||
output += "\n"
|
||||
|
||||
# show the next address after this chunk
|
||||
output += "; " + hex(self.end_address)
|
||||
|
||||
return output
|
||||
raise NotImplementedError("DisAsm was removed and never worked; hook up another disassembler please.")
|
||||
#return DisAsm(start_address=start_address, end_address=end_address, size=size, max_size=max_size, debug=debug, rom=self)
|
||||
|
||||
class AsmList(list):
|
||||
""" Simple wrapper to prevent all asm lines from being shown on screen.
|
||||
|
74
extras/test_dump_sections.py
Normal file
74
extras/test_dump_sections.py
Normal file
@ -0,0 +1,74 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
try:
|
||||
import unittest2 as unittest
|
||||
except ImportError:
|
||||
import unittest
|
||||
|
||||
# check for things we need in unittest
|
||||
if not hasattr(unittest.TestCase, 'setUpClass'):
|
||||
sys.stderr.write("The unittest2 module or Python 2.7 is required to run this script.")
|
||||
sys.exit(1)
|
||||
|
||||
from dump_sections import (
|
||||
upper_hex,
|
||||
format_bank_number,
|
||||
calculate_bank_quantity,
|
||||
dump_section,
|
||||
dump_incbin_for_section,
|
||||
)
|
||||
|
||||
class TestDumpSections(unittest.TestCase):
|
||||
def test_upper_hex(self):
|
||||
number = 0x52
|
||||
self.assertEquals(number, int("0x" + upper_hex(number), 16))
|
||||
|
||||
number = 0x1
|
||||
self.assertEquals(number, int("0x" + upper_hex(number), 16))
|
||||
|
||||
number = 0x0
|
||||
self.assertEquals(number, int("0x" + upper_hex(number), 16))
|
||||
|
||||
number = 0xAA
|
||||
self.assertEquals(number, int("0x" + upper_hex(number), 16))
|
||||
|
||||
number = 0xFFFFAAA0000
|
||||
self.assertEquals(number, int("0x" + upper_hex(number), 16))
|
||||
|
||||
def test_format_bank_number(self):
|
||||
address = 0x0
|
||||
self.assertEquals("0", format_bank_number(address))
|
||||
|
||||
address = 0x4000
|
||||
self.assertEquals("1", format_bank_number(address))
|
||||
|
||||
address = 0x1FC000
|
||||
self.assertEquals("7F", format_bank_number(address))
|
||||
|
||||
def test_dump_section(self):
|
||||
self.assertIn("SECTION", dump_section(str(0)))
|
||||
self.assertIn("HOME", dump_section(str(0)))
|
||||
self.assertNotIn("HOME", dump_section(str(1)))
|
||||
self.assertIn("DATA", dump_section(str(2)))
|
||||
self.assertIn("BANK", dump_section(str(40)))
|
||||
self.assertNotIn("BANK", dump_section(str(0)))
|
||||
|
||||
def test_dump_incbin_for_section(self):
|
||||
self.assertIn("INCBIN", dump_incbin_for_section(0))
|
||||
|
||||
def test_dump_incbin_for_section_separator(self):
|
||||
separator = "\n\n"
|
||||
self.assertIn(separator, dump_incbin_for_section(0, separator=separator))
|
||||
|
||||
separator = "\t\t" # dumb
|
||||
self.assertIn(separator, dump_incbin_for_section(0, separator=separator))
|
||||
|
||||
def test_dump_incbin_for_section_default(self):
|
||||
rom = "baserom.gbc"
|
||||
self.assertIn(rom, dump_incbin_for_section(0))
|
||||
|
||||
rom = "baserom"
|
||||
self.assertIn(rom, dump_incbin_for_section(0x4000))
|
||||
|
||||
if __name__ == "__main__":
|
||||
unittest.main()
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user