2015-10-16 09:30:24 +01:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
"""Usage: sixtypical [OPTIONS] FILES
|
|
|
|
|
2018-02-05 13:17:23 +00:00
|
|
|
Analyzes and compiles a Sixtypical program.
|
2015-10-16 09:30:24 +01:00
|
|
|
"""
|
|
|
|
|
|
|
|
from os.path import realpath, dirname, join
|
|
|
|
import sys
|
|
|
|
|
|
|
|
sys.path.insert(0, join(dirname(realpath(sys.argv[0])), '..', 'src'))
|
|
|
|
|
|
|
|
# ----------------------------------------------------------------- #
|
|
|
|
|
|
|
|
import codecs
|
2018-03-06 16:28:34 +00:00
|
|
|
from argparse import ArgumentParser
|
2015-10-22 15:45:16 +01:00
|
|
|
from pprint import pprint
|
2015-10-16 09:30:24 +01:00
|
|
|
import sys
|
|
|
|
import traceback
|
|
|
|
|
2018-03-27 16:23:22 +01:00
|
|
|
from sixtypical.parser import Parser, ParsingContext
|
2015-10-21 15:45:14 +01:00
|
|
|
from sixtypical.analyzer import Analyzer
|
2015-10-17 12:28:39 +01:00
|
|
|
from sixtypical.emitter import Emitter, Byte, Word
|
2015-10-17 15:06:50 +01:00
|
|
|
from sixtypical.compiler import Compiler
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
|
2018-03-27 16:23:22 +01:00
|
|
|
def merge_programs(programs):
|
|
|
|
"""Assumes that the programs do not have any conflicts."""
|
|
|
|
|
|
|
|
from sixtypical.ast import Program
|
|
|
|
|
|
|
|
full = Program(1, defns=[], routines=[])
|
|
|
|
for p in programs:
|
|
|
|
full.defns.extend(p.defns)
|
|
|
|
full.routines.extend(p.routines)
|
|
|
|
|
|
|
|
return full
|
|
|
|
|
|
|
|
|
2018-03-27 15:55:29 +01:00
|
|
|
def process_input_files(filenames, options):
|
2018-03-27 16:23:22 +01:00
|
|
|
context = ParsingContext()
|
|
|
|
|
2018-03-27 15:55:29 +01:00
|
|
|
programs = []
|
|
|
|
|
|
|
|
for filename in options.filenames:
|
|
|
|
text = open(filename).read()
|
2018-03-27 16:23:22 +01:00
|
|
|
parser = Parser(context, text, filename)
|
|
|
|
if options.debug:
|
|
|
|
print(context)
|
2018-03-27 15:55:29 +01:00
|
|
|
program = parser.program()
|
|
|
|
programs.append(program)
|
|
|
|
|
|
|
|
if options.parse_only:
|
|
|
|
return
|
|
|
|
|
2018-03-27 16:23:22 +01:00
|
|
|
program = merge_programs(programs)
|
2018-03-27 15:55:29 +01:00
|
|
|
|
|
|
|
analyzer = Analyzer(debug=options.debug)
|
|
|
|
analyzer.analyze_program(program)
|
|
|
|
|
2018-03-29 15:07:44 +01:00
|
|
|
if options.dump_fallthru_map:
|
|
|
|
import json
|
|
|
|
sys.stdout.write(json.dumps(program.fallthru_map, indent=4, sort_keys=True))
|
|
|
|
sys.stdout.write("\n")
|
|
|
|
|
2018-03-27 15:55:29 +01:00
|
|
|
if options.analyze_only:
|
|
|
|
return
|
|
|
|
|
|
|
|
fh = sys.stdout
|
|
|
|
|
|
|
|
if options.origin.startswith('0x'):
|
|
|
|
start_addr = int(options.origin, 16)
|
|
|
|
else:
|
|
|
|
start_addr = int(options.origin, 10)
|
|
|
|
|
|
|
|
output_format = options.output_format
|
|
|
|
|
|
|
|
prelude = []
|
|
|
|
if options.prelude == 'c64':
|
|
|
|
output_format = 'prg'
|
|
|
|
start_addr = 0x0801
|
|
|
|
prelude = [0x10, 0x08, 0xc9, 0x07, 0x9e, 0x32,
|
|
|
|
0x30, 0x36, 0x31, 0x00, 0x00, 0x00]
|
|
|
|
elif options.prelude == 'vic20':
|
|
|
|
output_format = 'prg'
|
|
|
|
start_addr = 0x1001
|
|
|
|
prelude = [0x0b, 0x10, 0xc9, 0x07, 0x9e, 0x34,
|
|
|
|
0x31, 0x30, 0x39, 0x00, 0x00, 0x00]
|
2018-03-28 14:20:53 +01:00
|
|
|
elif options.prelude == 'atari2600':
|
|
|
|
output_format = 'crtbb'
|
|
|
|
start_addr = 0xf000
|
|
|
|
prelude = [0x78, 0xd8, 0xa2, 0xff, 0x9a, 0xa9,
|
|
|
|
0x00,0x95, 0x00, 0xca, 0xd0, 0xfb]
|
|
|
|
|
2018-03-27 15:55:29 +01:00
|
|
|
elif options.prelude:
|
|
|
|
raise NotImplementedError("Unknown prelude: {}".format(options.prelude))
|
|
|
|
|
|
|
|
# If we are outputting a .PRG, we output the load address first.
|
|
|
|
# We don't use the Emitter for this b/c not part of addr space.
|
|
|
|
if output_format == 'prg':
|
|
|
|
fh.write(Word(start_addr).serialize(0))
|
|
|
|
|
|
|
|
emitter = Emitter(start_addr)
|
|
|
|
for byte in prelude:
|
|
|
|
emitter.emit(Byte(byte))
|
|
|
|
compiler = Compiler(emitter)
|
|
|
|
compiler.compile_program(program)
|
2018-03-28 14:20:53 +01:00
|
|
|
|
|
|
|
# If we are outputting a cartridge with boot and BRK address
|
|
|
|
# at the end, pad to ROM size minus 4 bytes, and emit addresses.
|
|
|
|
if output_format == 'crtbb':
|
|
|
|
emitter.pad_to_size(4096 - 4)
|
|
|
|
emitter.emit(Word(start_addr))
|
|
|
|
emitter.emit(Word(start_addr))
|
|
|
|
|
2018-03-27 15:55:29 +01:00
|
|
|
if options.debug:
|
|
|
|
pprint(emitter.accum)
|
|
|
|
else:
|
|
|
|
emitter.serialize(fh)
|
|
|
|
|
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
if __name__ == '__main__':
|
2018-03-06 16:28:34 +00:00
|
|
|
argparser = ArgumentParser(__doc__.strip())
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2018-03-06 16:28:34 +00:00
|
|
|
argparser.add_argument(
|
|
|
|
'filenames', metavar='FILENAME', type=str, nargs='+',
|
|
|
|
help="The SixtyPical source files to compile."
|
|
|
|
)
|
2018-03-29 15:07:44 +01:00
|
|
|
|
2018-03-13 13:33:01 +00:00
|
|
|
argparser.add_argument(
|
|
|
|
"--origin", type=str, default='0xc000',
|
|
|
|
help="Location in memory where the `main` routine will be "
|
|
|
|
"located. Default: 0xc000."
|
|
|
|
)
|
|
|
|
argparser.add_argument(
|
|
|
|
"--output-format", type=str, default='prg',
|
2018-03-28 14:20:53 +01:00
|
|
|
help="Executable format to produce. Options are: prg, crtbb. "
|
|
|
|
"Default: prg."
|
2018-03-13 13:33:01 +00:00
|
|
|
)
|
2018-03-06 16:28:34 +00:00
|
|
|
argparser.add_argument(
|
2018-03-06 17:00:39 +00:00
|
|
|
"--prelude", type=str,
|
2018-03-28 14:20:53 +01:00
|
|
|
help="Insert a snippet of code before the compiled program so that "
|
|
|
|
"it can be booted automatically on a particular platform. "
|
2018-03-13 13:33:01 +00:00
|
|
|
"Also sets the origin and format. "
|
2018-03-28 14:20:53 +01:00
|
|
|
"Options are: c64, vic20, atari2600."
|
2018-03-06 16:28:34 +00:00
|
|
|
)
|
2018-03-29 15:07:44 +01:00
|
|
|
|
2018-03-06 16:28:34 +00:00
|
|
|
argparser.add_argument(
|
2018-03-29 15:07:44 +01:00
|
|
|
"--analyze-only",
|
2018-03-06 16:28:34 +00:00
|
|
|
action="store_true",
|
2018-03-29 15:07:44 +01:00
|
|
|
help="Only parse and analyze the program; do not compile it."
|
|
|
|
)
|
|
|
|
argparser.add_argument(
|
|
|
|
"--dump-fallthru-map",
|
|
|
|
action="store_true",
|
|
|
|
help="Dump the fallthru map to stdout after analyzing the program."
|
2018-03-06 16:28:34 +00:00
|
|
|
)
|
|
|
|
argparser.add_argument(
|
|
|
|
"--parse-only",
|
|
|
|
action="store_true",
|
|
|
|
help="Only parse the program; do not analyze or compile it."
|
|
|
|
)
|
2018-03-29 15:07:44 +01:00
|
|
|
argparser.add_argument(
|
|
|
|
"--debug",
|
|
|
|
action="store_true",
|
|
|
|
help="Display debugging information when analyzing and compiling."
|
|
|
|
)
|
2018-03-06 16:28:34 +00:00
|
|
|
argparser.add_argument(
|
|
|
|
"--traceback",
|
|
|
|
action="store_true",
|
|
|
|
help="When an error occurs, display a full Python traceback."
|
|
|
|
)
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2018-03-06 16:28:34 +00:00
|
|
|
options, unknown = argparser.parse_known_args(sys.argv[1:])
|
|
|
|
remainder = ' '.join(unknown)
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2018-03-27 15:55:29 +01:00
|
|
|
try:
|
|
|
|
process_input_files(options.filenames, options)
|
|
|
|
except Exception as e:
|
|
|
|
if options.traceback:
|
|
|
|
raise
|
2018-02-06 16:14:44 +00:00
|
|
|
else:
|
2018-03-27 15:55:29 +01:00
|
|
|
traceback.print_exception(e.__class__, e, None)
|
|
|
|
sys.exit(1)
|