gambatte/testrunner/qdgbas.py

285 lines
7.3 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
# Quick and Dirty Game Boy Assembler
import re
import sys
import collections
class InputError(RuntimeError): pass
Op = collections.namedtuple('Op', ['string', 'code', 'size', 'assemble'])
def mkrestr(s):
return re.sub(r'\s+', r'\s*', re.sub(r'([,)(+])', r' \\\1 ', s))
imm8 = r'[0-9a-fA-F][0-9a-fA-F]'
imm16 = imm8 + imm8
labelregexp = re.compile('\w+\s+(\w+)')
def assembleNormal(outdata, outpos, indata, inpos, targets, op):
pass
def assembleImm8(outdata, outpos, indata, inpos, targets, op):
m = re.match(mkrestr(op.string).replace('imm8', '('+imm8+')'), indata[inpos])
outdata[outpos] = int(m.group(1), 0x10)
def assembleImm16(outdata, outpos, indata, inpos, targets, op):
m = re.match(mkrestr(op.string).replace('imm16', '('+imm16+')'), indata[inpos])
outdata[outpos ] = int(m.group(1)[2:4], 0x10)
outdata[outpos+1] = int(m.group(1)[0:2], 0x10)
def assembleJr(outdata, outpos, indata, inpos, targets, op):
label = labelregexp.match(indata[inpos]).group(1)
t = targets.get(label)
t = t - (outpos + 1) if t != None else int(label, 0x10)
if t < -0x80 or t > 0x7f:
raise InputError('line ' + str(i) + ': ' + indata[inpos], 'Relative jump target too far away')
outdata[outpos] = t & 0xff
def assembleJp(outdata, outpos, indata, inpos, targets, op):
label = labelregexp.match(indata[inpos]).group(1)
t = targets.get(label)
if t == None:
t = int(label, 0x10)
elif t >= 0x8000:
t = 0x4000 | (t & 0x3fff)
outdata[outpos ] = t & 0xFF
outdata[outpos+1] = t >> 8
def makeoplist():
l = []
def addop(opstr, opcode):
size = (opcode >= 0) + (opcode >= 0x100)
moperand = re.match(r'(.*imm8)|(jr.* target)|(.*imm16)|(call.* target|jp.* target)', opstr)
if moperand:
optype = [(0, assembleNormal), (1, assembleImm8), (1, assembleJr), (2, assembleImm16), (2, assembleJp)][moperand.lastindex]
l.append(Op(opstr, opcode, size + optype[0], optype[1]))
else:
l.append(Op(opstr, opcode, size, assembleNormal))
addop('nop', 0x00)
addop('ld bc, imm16', 0x01)
addop('ld(bc), a', 0x02)
addop('inc bc', 0x03)
addop('inc b', 0x04)
addop('dec b', 0x05)
addop('ld b, imm8', 0x06)
addop('ld a, (bc)', 0x0a)
addop('inc c', 0x0c)
addop('dec c', 0x0d)
addop('ld c, imm8', 0x0e)
addop('rrca', 0x0f)
addop('stop, imm8', 0x10)
addop('ld de, imm16', 0x11)
addop('ld(de), a', 0x12)
addop('inc d', 0x14)
addop('dec d', 0x15)
addop('ld d, imm8', 0x16)
addop('jr target', 0x18)
addop('add hl, de', 0x19)
addop('ld a, (de)', 0x1a)
addop('inc e', 0x1c)
addop('dec e', 0x1d)
addop('jrnz target', 0x20)
addop('ld hl, imm16', 0x21)
addop('ld(hl++), a', 0x22)
addop('inc h', 0x24)
addop('dec h', 0x25)
addop('inc l', 0x2c)
addop('dec l', 0x2d)
addop('ld sp, imm16', 0x31)
addop('ld(hl--), a', 0x32)
addop('ld(hl), imm8', 0x36)
addop('ld a, (hl--)', 0x3a)
addop('inc a', 0x3c)
addop('dec a', 0x3d)
addop('ld a, imm8', 0x3e)
addop('ld b, a', 0x47)
addop('ld d, a', 0x57)
addop('halt', 0x76)
addop('ld(hl), a', 0x77)
addop('ld a, b', 0x78)
addop('ld a, c', 0x79)
addop('ld a, d', 0x7a)
addop('ld a, e', 0x7b)
addop('ld a, (hl)', 0x7e)
addop('sub a, b', 0x90)
addop('sub a, c', 0x91)
addop('sub a, d', 0x92)
addop('and a, b', 0xa0)
addop('and a, c', 0xa1)
addop('and a, d', 0xa2)
addop('xor a, a', 0xaf)
addop('or a, a', 0xb7)
addop('cmp a, b', 0xb8)
addop('cmp a, c', 0xb9)
addop('cmp a, d', 0xba)
addop('cmp a, e', 0xbb)
addop('cmp a, h', 0xbc)
addop('jpnz target', 0xc2)
addop('jp target', 0xc3)
addop('ret', 0xc9)
addop('call target', 0xcd)
addop('pop de', 0xd1)
addop('sub a, imm8', 0xd6)
addop('ldff(imm8), a', 0xe0)
addop('pop hl', 0xe1)
addop('ldff(c), a', 0xe2)
addop('push hl', 0xe5)
addop('and a, imm8', 0xe6)
addop('ld(imm16), a', 0xea)
addop('xor a, imm8', 0xee)
addop('ldff a, (imm8)', 0xf0)
addop('ldff a, (c)', 0xf2)
addop('ld a, (imm16)', 0xfa)
addop('ei', 0xfb)
addop('cmp a, imm8', 0xfe)
addop('srl a', 0xcb3f)
addop('\w* : ', -1)
return l
def makeopregexp(oplist):
rxps = ''
for e in oplist:
rxps += '(' + mkrestr(e.string) + ')|'
return re.compile(rxps[0:len(rxps)-1].replace('imm8', imm8).replace('imm16', imm16).replace('target', r'\s\w+'))
oplist = makeoplist()
opregexp = makeopregexp(oplist)
def maptargets(targets, indata, instart, addr):
for i in xrange(instart, len(indata)):
match = opregexp.match(indata[i])
if match == None:
break
op = oplist[match.lastindex - 1]
if op.size == 0:
targets[re.match('(\w*)\s*:', indata[i]).group(1)] = addr
addr += op.size
return i
def ascode(outdata, addr, indata, instart, targets):
for i in xrange(instart, len(indata)):
match = opregexp.match(indata[i])
if match == None:
break
op = oplist[match.lastindex - 1]
outpos = addr
if op.code >= 0x100:
outdata[outpos ] = op.code >> 8
outdata[outpos+1] = op.code & 0xff
outpos += 2
elif op.code >= 0:
outdata[outpos] = op.code
outpos += 1
op.assemble(outdata, outpos, indata, i, targets, op)
addr += op.size
return i
def asdata(outdata, addr, indata, instart):
try:
for i in xrange(instart, len(indata)):
ints = [int(x, 0x10) for x in indata[i].split()]
outdata[addr:addr+len(ints)] = ints
addr += len(ints)
except ValueError:
pass
return i
def putgblogo(outdata):
outdata[0x104:0x134] = \
[0xce, 0xed, 0x66, 0x66, 0xcc, 0x0d, 0x00, 0x0b,
0x03, 0x73, 0x00, 0x83, 0x00, 0x0c, 0x00, 0x0d,
0x00, 0x08, 0x11, 0x1f, 0x88, 0x89, 0x00, 0x0e,
0xdc, 0xcc, 0x6e, 0xe6, 0xdd, 0xdd, 0xd9, 0x99,
0xbb, 0xbb, 0x67, 0x63, 0x6e, 0x0e, 0xec, 0xcc,
0xdd, 0xdc, 0x99, 0x9f, 0xbb, 0xb9, 0x33, 0x3e]
def putheadercsum(outdata):
outdata[0x14d] = sum([-(x+1) for x in outdata[0x134:0x14d]]) & 0xFF
def putromcsum(outdata):
csum = sum(outdata)
outdata[0x14e:0x150] = [csum >> 8 & 0xff, csum & 0xff]
def assembleFile(indata):
size = 0x8000
startpos = 0
for i in xrange(0, len(indata)):
spl = indata[i].split()
if len(spl) > 0 and spl[0] == '.size':
size = int(spl[1], 0x10)
startpos = i + 1
break
targets = {}
i = startpos
while i < len(indata):
spl = [x.strip() for x in indata[i].split('@')]
i += 1;
if spl[0] == '.code':
i = maptargets(targets, indata, i, int(spl[1], 0x10))
outdata = bytearray(size)
i = startpos
while i < len(indata):
if indata[i] != '':
spl = [x.strip() for x in indata[i].split('@')]
if spl[0] == '.code':
i = ascode(outdata, int(spl[1], 0x10), indata, i+1, targets)
elif spl[0] == '.data':
i = asdata(outdata, int(spl[1], 0x10), indata, i+1)
else:
raise RuntimeError('line ' + str(i) + ': ' + indata[i])
else:
i += 1
putgblogo(outdata)
putheadercsum(outdata)
putromcsum(outdata)
return outdata
def readDataFromFile(filename):
infile = open(filename, 'r')
indata = [line.strip() for line in infile]
infile.close()
return indata
def writeDataToFile(filename, data):
outfile = open(filename, 'wb')
outfile.write(data)
outfile.close()
def outFilenameFromInFilename(inname, h143):
return inname.rsplit('.', 1)[0] + ('.gbc' if (h143 & 0x80) else '.gb')
def main():
for arg in sys.argv[1:]:
print 'processing ' + arg
outdata = assembleFile(readDataFromFile(arg))
writeDataToFile(outFilenameFromInFilename(arg, outdata[0x143]), outdata)
print "\nassembled " + str(len(sys.argv) - 1) + (' files' if len(sys.argv) != 2 else ' file')
if __name__ == "__main__":
main()