logisim-stuff/multicycle cpu/assembler.py

188 lines
5.9 KiB
Python
Raw Normal View History

2015-01-04 15:24:53 -06:00
#Assembler for new Terpstra CPU
# Dec 20 2014
# CAK
import re
import sys
if len(sys.argv) == 2:
#first argument is second in list
SOURCE_FILENAME = sys.argv[1]
else:
SOURCE_FILENAME = raw_input('input file:')
t = SOURCE_FILENAME.strip( '.asm' )
OUTPUT_FILENAME = t+ ".code"
VALID_LABEL = re.compile("[a-zA-Z_][a-zA-Z0-9_]*")
#key is lowercase instruction name
#value is (numeric code, True is needs argument)
INSTRUCTION_TABLE = {
'lda':(0, True),
'ldb':(1, True),
'ldai':(2, True),
'ldbi':(3, True),
'add':(4, False),
'sub':(5, False),
'mult':(6, False),
'div':(7, False),
'sta':(8, True),
'stb':(9, True),
'str':(10, True),
'jmp':(11, True),
'jmpc':(12, True),
'jmpz':(13, True)
}
output_file = open(OUTPUT_FILENAME, "w")
labels = {}
memcounter = 0
def convert_number(instring):
if instring.startswith(("0x", "0X")):
return int(instring[2:], 16)
elif instring.startswith(("x", "X")):
return int(instring[1:], 16)
elif instring.startswith(("b", "B")):
return int(instring[1:], 2)
else:
#try good old decimal conversion
return int(instring)
def assert_label(instring):
#throw an AsserionError if the string is not a valid label or
#instruction name
if VALID_LABEL.match(instring) is None:
raise AssertionError(
"'{0}' is not a valid label/token".format(instring))
asm_file = open(SOURCE_FILENAME)
asm_lines = asm_file.readlines()
#first pass works out values for labels and variables
linenum = 1
for line in asm_lines:
try:
#take out any comment
line = line.strip()
line = line.split(";")[0]
tokens = line.split()
print tokens
#check for label declaration
if len(tokens) > 0 and tokens[0][-1] == ":":
#a label, put it in the table
label_name = tokens[0][:-1]
assert_label(label_name)
labels[label_name] = memcounter
tokens = tokens[1:]
#check for a definition
if len(tokens) > 0 and tokens[0] == ".def":
if len(tokens) >= 3:
(name, value) = tokens[1:]
assert_label(name)
labels[name] = convert_number(value)
#remove the three tokens in the .def statement
tokens = tokens[3:]
else:
raise AssertionError("Bad .def statement")
#check for a variable definition, dont do anything until
#next pass
if len(tokens) > 0 and tokens[0] == ".var":
if len(tokens) >= 3:
tokens = tokens[3:]
else:
raise AssertionError("Bad .var statement")
#check if there is a token that is a valid instruction
if len(tokens) > 0:
instruction = tokens[0].lower()
if instruction in INSTRUCTION_TABLE:
#increment program counter (by 2)
memcounter += 2
tokens = tokens[1:]
else:
raise AssertionError(
"Unknown instruction '{0}'".format(instruction))
except:
#catch in error and include line number
print "Error at .asm file line {0}".format(linenum)
#keep passing the error along
raise
linenum += 1
#second pass generates code
output_code = []
code_list = []
linenum = 1
init_values = [] #for data variables
for line in asm_lines:
try:
#take out any comment
line = line.strip()
without_comments = line.split(";")[0]
tokens = without_comments.split()
if len(tokens) > 0 and tokens[0][-1] == ":":
#label at start, remove from token list
tokens = tokens[1:]
if len(tokens) > 0 and tokens[0][0] != ".":
#should be an instruction
instruction = tokens[0].lower()
if instruction in INSTRUCTION_TABLE:
(opcode, needs_argument) = INSTRUCTION_TABLE[instruction]
output_code.append(opcode)
if needs_argument:
if len(tokens) > 1:
argument = tokens[1]
if argument in labels:
argument = labels[argument]
else:
argument = convert_number(argument)
else:
raise AssertionError(
"Argument Missing for {0}".format(instruction))
else:
argument = 0x00 #space for no argument
output_code.append(argument)
code_list.append("({0} {1}) {2}".format(
hex(opcode), hex(argument),line.strip()) )
else:
raise AssertionError(
"{0} is not a valid instruction".format(instruction))
elif len(tokens) > 0 and tokens[0] == ".var":
#variable definition, make room at the end of the program
(name, init_value) = tokens[1:]
labels[name] = memcounter #end of program
memcounter += 1
init_values.append(convert_number(init_value))
else:
#should still add to code listing
code_list.append(line)
except:
print "Error at .asm file line {0}".format(linenum)
raise
linenum += 1
#append variables with initial values to end of code
for val in init_values:
output_code.append(val)
print labels
print output_code
for line in code_list:
print line
#header for logisim image
output_file.write("v2.0 raw\n")
for code in output_code:
output_file.write("{0:02x}\n".format(code))
asm_file.close()
output_file.close()