mirror of
https://github.com/jefftranter/udis.git
synced 2025-01-27 20:29:51 +00:00
Merge pull request #3 from robmcmullen/udis_only
Preliminary support for 0xfdcb and 0xddcb bit operations
This commit is contained in:
commit
e0a5e76ed3
12
udis.py
12
udis.py
@ -26,6 +26,7 @@ import signal
|
|||||||
|
|
||||||
pcr = 1
|
pcr = 1
|
||||||
und = 2
|
und = 2
|
||||||
|
z80bit = 4
|
||||||
|
|
||||||
# Functions
|
# Functions
|
||||||
|
|
||||||
@ -181,7 +182,7 @@ while True:
|
|||||||
# Handle relative addresses. Indicated by the flag pcr being set.
|
# Handle relative addresses. Indicated by the flag pcr being set.
|
||||||
# Assumes the operand that needs to be PC relative is the last one.
|
# Assumes the operand that needs to be PC relative is the last one.
|
||||||
# Note: Code will need changes if more flags are added.
|
# Note: Code will need changes if more flags are added.
|
||||||
if flags & 1 == pcr:
|
if flags & pcr:
|
||||||
if op[length-1] < 128:
|
if op[length-1] < 128:
|
||||||
op[length-1] = address + op[length-1] + length
|
op[length-1] = address + op[length-1] + length
|
||||||
else:
|
else:
|
||||||
@ -195,7 +196,14 @@ while True:
|
|||||||
elif length == 2:
|
elif length == 2:
|
||||||
operand = format.format(op[1])
|
operand = format.format(op[1])
|
||||||
elif length == 3:
|
elif length == 3:
|
||||||
operand = format.format(op[1], op[2])
|
if flags & z80bit:
|
||||||
|
opcode = (opcode << 16) + op[2]
|
||||||
|
# reread opcode table for real format string
|
||||||
|
length, mnemonic, mode, flags = opcodeTable[opcode]
|
||||||
|
format = addressModeTable[mode]
|
||||||
|
operand = format.format(op[1])
|
||||||
|
else:
|
||||||
|
operand = format.format(op[1], op[2])
|
||||||
elif length == 4:
|
elif length == 4:
|
||||||
operand = format.format(op[1], op[2], op[3])
|
operand = format.format(op[1], op[2], op[3])
|
||||||
elif length == 5:
|
elif length == 5:
|
||||||
|
44
z80.py
44
z80.py
@ -299,7 +299,6 @@ addressModeTable = {
|
|||||||
"z,nn" : "z,${1:02X}{0:02X}",
|
"z,nn" : "z,${1:02X}{0:02X}",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
# Op Code Table
|
# Op Code Table
|
||||||
# Key is numeric opcode (possibly multiple bytes)
|
# Key is numeric opcode (possibly multiple bytes)
|
||||||
# Value is a list:
|
# Value is a list:
|
||||||
@ -880,10 +879,6 @@ opcodeTable = {
|
|||||||
|
|
||||||
0xdd8e : [3, "adc", "indix+d" ],
|
0xdd8e : [3, "adc", "indix+d" ],
|
||||||
|
|
||||||
# The below are a set of instructions that all start with 0xddcb. They
|
|
||||||
# are not supported yet.
|
|
||||||
0xddcb : [ 4, "unimplemented", "implied" ],
|
|
||||||
|
|
||||||
0xed40 : [ 2, "in", "b,indc" ],
|
0xed40 : [ 2, "in", "b,indc" ],
|
||||||
0xed41 : [ 2, "out", "indc,b" ],
|
0xed41 : [ 2, "out", "indc,b" ],
|
||||||
0xed42 : [ 2, "sbc", "hl,bc" ],
|
0xed42 : [ 2, "sbc", "hl,bc" ],
|
||||||
@ -976,11 +971,42 @@ opcodeTable = {
|
|||||||
0xfdb6 : [ 3, "or", "indiy+d" ],
|
0xfdb6 : [ 3, "or", "indiy+d" ],
|
||||||
0xfdbe : [ 3, "cp", "indiy+d" ],
|
0xfdbe : [ 3, "cp", "indiy+d" ],
|
||||||
|
|
||||||
# The below are a set of instructions that all start with 0xfdcb. They
|
# Placeholder 2-byte leadins for the 4-byte ix/iy bit instructions fully
|
||||||
# are not supported yet.
|
# defined below. The z80bit flag triggers a special case in the disassembler
|
||||||
0xfdcb : [ 4, "unimplemented", "implied" ],
|
# to look up the 4 byte instruction.
|
||||||
|
0xddcb : [ 4, "ixbit", "implied", z80bit ],
|
||||||
|
0xfdcb : [ 4, "iybit", "implied", z80bit ],
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def extra_opcodes(addr_table, op_table):
|
||||||
|
# Create all the 0xddcb and 0xfdcb addressing modes. The modes look like [0-7],(i[xy]+*)[,[abcdehl]]?
|
||||||
|
for index in ['x', 'y']:
|
||||||
|
for bit in range(8):
|
||||||
|
k = "%d,indi%s+d" % (bit, index)
|
||||||
|
v = "%d,(i%s+${0:02X})" % (bit, index)
|
||||||
|
addr_table[k] = v
|
||||||
|
for reg in ['a', 'b', 'c', 'd', 'e', 'h', 'l']:
|
||||||
|
k = "%d,indi%s+d,%s" % (bit, index, reg)
|
||||||
|
v = "%d,(i%s+${0:02X}),%s" % (bit, index, reg)
|
||||||
|
addr_table[k] = v
|
||||||
|
|
||||||
|
# Create all the 0xddcb and 0xfdcb opcodes. These are all 4 byte opcodes
|
||||||
|
# where the 3rd byte is a -128 - +127 offset. For the purposes of using
|
||||||
|
# this table, the 3rd byte will be marked as zero and the disassembler will
|
||||||
|
# have to insert the real 3rd byte the check of the z80bit special case
|
||||||
|
for first_byte, x_or_y in [(0xdd, 'x'), (0xfd, 'y')]:
|
||||||
|
# groups of 8, expand to full 256
|
||||||
|
mnemonics_8 = ['rlc', 'rrc', 'rl', 'rr', 'sla', 'sra', 'sll', 'sr1'] + ['bit'] * 8 + ['res'] * 8 + ['set'] * 8
|
||||||
|
mnemonics = [m for mnemonic in mnemonics_8 for m in [mnemonic]*8]
|
||||||
|
|
||||||
|
# create all 256 addressing modes, in groups of 64
|
||||||
|
addrmodes = ['indi%s+d' + a for a in [',b', ',c', ',d', ',e', ',h', ',l', '', ',a']] * 8 + [f % d for d in range(8) for f in ['%d,indi%%s+d'] * 8] + [f % d for d in range(8) for f in ['%d,indi%%s+d' + a for a in [',b', ',c', ',d', ',e', ',h', ',l', '', ',a']]] * 2
|
||||||
|
|
||||||
|
for fourth_byte, (instruction, addrmode) in enumerate(zip(mnemonics, addrmodes)):
|
||||||
|
opcode = (first_byte << 24) + (0xcb << 16) + fourth_byte
|
||||||
|
op_table[opcode] = [ 4, instruction, addrmode % x_or_y, z80bit ]
|
||||||
|
extra_opcodes(addressModeTable, opcodeTable)
|
||||||
|
del extra_opcodes
|
||||||
|
|
||||||
# End of processor specific code
|
# End of processor specific code
|
||||||
##########################################################################
|
##########################################################################
|
||||||
|
Loading…
x
Reference in New Issue
Block a user