#!/usr/bin/env python3 # synacor.py - An implementation of the Synacor Challenge # Copyright © 2017 RunasSudo # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Affero General Public License for more details. # # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see . from libsynacor import * import argparse import struct parser = argparse.ArgumentParser() parser.add_argument('file', help='.asm file to read') parser.add_argument('output', help='.bin file to write') args = parser.parse_args() line_no = 0 def split_line(line): tokens = [] token = '' idx = 0 in_comment = False in_string = False in_escape = False while idx < len(line): if in_comment: pass elif in_string: if in_escape: token += line[idx] else: if line[idx] == '\\': in_escape = True token += line[idx] elif line[idx] == '"': in_string = False token += line[idx] else: token += line[idx] else: if line[idx] == ' ': if token != '': tokens.append(token) token = '' elif line[idx] == '"': in_string = True token += line[idx] elif line[idx] == ';': in_comment = True else: token += line[idx] idx += 1 # Final token if token != '': tokens.append(token) return tokens def unescape_char(char): return char.encode('utf-8').decode('unicode_escape') def assemble_next_instruction(source): line = source.readline() global line_no; line_no += 1 if line == '': return None, [] tokens = split_line(line.strip()) return assemble_instruction(source, tokens) def assemble_instruction(source, tokens): if len(tokens) == 0: return assemble_next_instruction(source) if tokens[0].endswith(':'): # Label label = tokens[0][:-1] instructions, inst_labels = assemble_instruction(source, tokens[1:]) return instructions, inst_labels + [label] else: # Instruction name = tokens[0] if name not in instructions_by_name: raise Exception('Unknown instruction {}'.format(name)) instruction = instructions_by_name[name]() # Special cases if isinstance(instruction, InstructionOut) and tokens[1].startswith('"'): chars = unescape_char(tokens[1][1:-1]) instructions = [] for char in chars: instruction = InstructionOut() instruction.args.append(OpLiteral(ord(char))) instructions.append(instruction) return instructions, [] elif isinstance(instruction, InstructionData): if tokens[1].startswith('"'): chars = unescape_char(tokens[1][1:-1]) instruction.args = [ord(char) for char in chars] return [instruction], [] else: instruction.args = [int(x, 16) for x in tokens[1:]] return [instruction], [] else: if len(tokens) != instruction.nargs + 1: raise Exception('Invalid number of arguments: Expected {}, got {}'.format(instruction.nargs, len(tokens) - 1)) for i in range(instruction.nargs): argstr = tokens[i + 1] if argstr.startswith('R'): # Register arg = OpRegister(int(argstr[1:])) else: # Hex literal arg = OpLiteral(int(argstr, 16)) instruction.args.append(arg) return [instruction], [] # TODO: First pass # Second pass SYN_MEM = [0] * 32768 SYN_PTR = 0 with open(args.file, 'r') as source: try: while True: instructions, labels = assemble_next_instruction(source) if instructions is None: break for instruction in instructions: code = instruction.assemble() SYN_MEM[SYN_PTR:SYN_PTR+len(code)] = code SYN_PTR += len(code) except Exception as ex: raise Exception('Error at line {}'.format(line_no)) from ex with open(args.output, 'wb') as f: f.write(struct.pack('<32768H', *SYN_MEM))