New command-line system.

This is a full optparse-based parser for all the options we want
to have in Ophis 2.0, but the pass-disablers aren't working yet.

This also doesn't handle positional arguments the way we hope
to eventually; that will come later.

optparse is deprecated in 2.7, but its replacement isn't available
in any previous version of Python, so we avoid it so as to not
gratuitously break compatibility on older machines.

It would be nice to at least stay usable on stock Leopard Macs (2.5).
This commit is contained in:
Michael C. Martin 2012-05-30 22:18:45 -07:00
parent eae4ea7dcd
commit e47073bc1d
6 changed files with 90 additions and 81 deletions

View File

@ -7,4 +7,4 @@ path.insert(0, join(dirname(realpath(argv[0])), '..', 'src'))
import Ophis.Main import Ophis.Main
Ophis.Main.run_ophis() Ophis.Main.run_ophis(argv[1:])

View File

@ -1,17 +1,76 @@
"""Command line options data. """Command line options data."""
verbose: import optparse
0: Only report errors
1: Announce each file as it is read, and data count (default)
2: As above, but also announce each pass.
3: As above, but print the IR after each pass.
4: As above, but print the labels after each pass.
6510 compatibility and deprecation are handled in Ophis.Main."""
# Copyright 2002-2012 Michael C. Martin and additional contributors. # Copyright 2002-2012 Michael C. Martin and additional contributors.
# You may use, modify, and distribute this file under the MIT # You may use, modify, and distribute this file under the MIT
# license: See README for details. # license: See README for details.
verbose = 1; enable_collapse = True
enable_branch_extend = True
enable_undoc_ops = False
enable_65c02_exts = False
warn_on_branch_extend = True
print_summary = True
print_loaded_files = False
print_pass = False
print_ir = False
print_labels = False
infile = None
outfile = None
def parse_args(raw_args):
"Populate the module's globals based on the command-line options given."
global enable_collapse, enable_branch_extend, enable_undoc_ops, enable_65c02_exts
global warn_on_branch_extend
global print_summary, print_loaded_files, print_pass, print_ir, print_labels
global infile, outfile
parser = optparse.OptionParser(usage= "Usage: %prog [options] srcfile outfile", version="Ophis 6502 cross-assembler, version 2.0")
ingrp = optparse.OptionGroup(parser, "Input options")
ingrp.add_option("-u", "--undoc", action="store_true", default=False, help="Enable 6502 undocumented opcodes")
ingrp.add_option("-c", "--65c02", action="store_true", default=False, dest="c02", help="Enable 65c02 extended instruction set")
outgrp = optparse.OptionGroup(parser, "Console output options")
outgrp.add_option("-v", "--verbose", action="store_const", const=2, help="Verbose mode", default=1)
outgrp.add_option("-q", "--quiet", action="store_const", help="Quiet mode", dest="verbose", const=0)
outgrp.add_option("-d", "--debug", action="count", dest="verbose", help=optparse.SUPPRESS_HELP)
outgrp.add_option("--no-warn", action="store_false", dest="warn", default=True, help="Do not print warnings")
bingrp = optparse.OptionGroup(parser, "Binary output options")
bingrp.add_option("--no-collapse", action="store_false", dest="enable_collapse", default="True", help="Disable zero-page collapse pass")
bingrp.add_option("--no-branch-extend", action="store_false", dest="enable_branch_extend", default="True", help="Disable branch-extension pass")
parser.add_option_group(ingrp)
parser.add_option_group(outgrp)
parser.add_option_group(bingrp)
(options, args) = parser.parse_args(raw_args)
if len(args) > 2:
parser.error("Too many files specified")
if len(args) == 1:
parser.error("No output file specified")
if len(args) == 0:
parser.error("No files specified")
if options.c02 and options.undoc:
parser.error("--undoc and --65c02 are mutually exclusive")
infile = args[0]
outfile = args[1]
enable_collapse = options.enable_collapse
enable_branch_extend = options.enable_branch_extend
enable_undoc_ops = options.undoc
enable_65c02_exts = options.c02
warn_on_branch_extend = options.warn
print_summary = options.verbose > 0 # no options set
print_loaded_files = options.verbose > 1 # v
print_pass = options.verbose > 2 # dd
print_ir = options.verbose > 3 # ddd
print_labels = options.verbose > 4 # dddd
parser.destroy()

View File

@ -310,7 +310,7 @@ def parse_line(ppt, lexemelist):
def parse_file(ppt, filename): def parse_file(ppt, filename):
"Loads an Ophis source file, and returns an IR list." "Loads an Ophis source file, and returns an IR list."
Err.currentpoint = ppt Err.currentpoint = ppt
if Cmd.verbose > 0: if Cmd.print_loaded_files:
if filename != '-': if filename != '-':
print>>sys.stderr, "Loading "+filename print>>sys.stderr, "Loading "+filename
else: else:

View File

@ -1,7 +1,7 @@
"""Main controller routines for the Ophis assembler. """Main controller routines for the Ophis assembler.
When invoked as main, interprets its command line and goes from there. When invoked as main, interprets its command line and goes from there.
Otherwise, use run_all to interpret a file set.""" Otherwise, use run_ophis(cmdline-list) to use it inside a script."""
# Copyright 2002-2012 Michael C. Martin and additional contributors. # Copyright 2002-2012 Michael C. Martin and additional contributors.
# You may use, modify, and distribute this file under the MIT # You may use, modify, and distribute this file under the MIT
@ -17,18 +17,6 @@ import Ophis.Environment
import Ophis.CmdLine import Ophis.CmdLine
import Ophis.Opcodes import Ophis.Opcodes
def usage():
"Prints a usage message and quits."
print>>sys.stderr, "Usage:"
print>>sys.stderr, "\tOphis [options] infile outfile"
print>>sys.stderr, ""
print>>sys.stderr, "Options:"
print>>sys.stderr, "\t-6510 Allow 6510 undocumented opcodes"
print>>sys.stderr, "\t-65c02 Enable 65c02 extensions"
print>>sys.stderr, "\t-v n Set verbosity to n (0-4, 1=default)"
sys.exit(1)
def run_all(infile, outfile): def run_all(infile, outfile):
"Transforms the source infile to a binary outfile." "Transforms the source infile to a binary outfile."
Err.count = 0 Err.count = 0
@ -69,56 +57,17 @@ def run_all(infile, outfile):
else: else:
Err.report() Err.report()
def run_ophis(): def run_ophis(args):
infile = None Ophis.CmdLine.parse_args(args)
outfile = None
p65_compatibility_mode = 0
chip_extension = None
reading_arg = 0
for x in sys.argv[1:]:
if reading_arg:
try:
Ophis.CmdLine.verbose = int(x)
reading_arg = 0
except ValueError:
print>>sys.stderr, "FATAL: Non-integer passed as argument to -v"
usage()
elif x[0] == '-' and x != '-':
if x == '-v':
reading_arg = 1
elif x == '-6510':
chip_extension = Ophis.Opcodes.undocops
elif x == '-65c02':
chip_extension = Ophis.Opcodes.c02extensions
else:
print>>sys.stderr, "FATAL: Unknown option "+x
usage()
elif infile == None:
infile = x
elif outfile == None:
outfile = x
else:
print>>sys.stderr, "FATAL: Too many files specified"
usage()
if infile is None:
print>>sys.stderr, "FATAL: No files specified"
usage()
if outfile is None:
print>>sys.stderr, "FATAL: No output file specified"
usage()
Ophis.Frontend.pragma_modules.append(Ophis.CorePragmas) Ophis.Frontend.pragma_modules.append(Ophis.CorePragmas)
if chip_extension is not None: if Ophis.CmdLine.enable_undoc_ops:
Ophis.Opcodes.opcodes.update(chip_extension) Ophis.Opcodes.opcodes.update(Ophis.Opcodes.undocops)
elif Ophis.CmdLine.enable_65c02_exts:
Ophis.Opcodes.opcodes.update(Ophis.Opcodes.c02extensions)
Ophis.CorePragmas.reset() Ophis.CorePragmas.reset()
run_all(infile, outfile) run_all(Ophis.CmdLine.infile, Ophis.CmdLine.outfile)
if __name__ == '__main__': if __name__ == '__main__':
run_ophis() run_ophis(sys.argv[1:])

View File

@ -54,16 +54,16 @@ class Pass:
"""Prepares the environment and runs this pass, possibly """Prepares the environment and runs this pass, possibly
printing debugging information.""" printing debugging information."""
if Err.count == 0: if Err.count == 0:
if Cmd.verbose > 1: print>>sys.stderr, "Running: "+self.name if Cmd.print_pass: print>>sys.stderr, "Running: "+self.name
env.reset() env.reset()
self.prePass() self.prePass()
node.accept(self, env) node.accept(self, env)
self.postPass() self.postPass()
env.reset() env.reset()
if Cmd.verbose > 3: if Cmd.print_labels:
print>>sys.stderr, "Current labels:" print>>sys.stderr, "Current labels:"
print>>sys.stderr, env print>>sys.stderr, env
if Cmd.verbose > 2: if Cmd.print_ir:
print>>sys.stderr, "Current IR:" print>>sys.stderr, "Current IR:"
print>>sys.stderr, node print>>sys.stderr, node
@ -83,7 +83,7 @@ class FixPoint:
p.go(node, env) p.go(node, env)
if Err.count != 0: break if Err.count != 0: break
if self.fixpoint(): break if self.fixpoint(): break
if Cmd.verbose > 1: print>>sys.stderr, "Fixpoint failed, looping back" if Cmd.print_pass: print>>sys.stderr, "Fixpoint failed, looping back"
else: else:
Err.log("Can't make %s converge! Maybe there's a recursive dependency somewhere?" % self.name) Err.log("Can't make %s converge! Maybe there's a recursive dependency somewhere?" % self.name)
@ -96,7 +96,7 @@ class DefineMacros(Pass):
def postPass(self): def postPass(self):
if self.inDef: if self.inDef:
Err.log("Unmatched .macro") Err.log("Unmatched .macro")
elif Cmd.verbose > 2: elif Cmd.print_ir:
print>>sys.stderr, "Macro definitions:" print>>sys.stderr, "Macro definitions:"
Macro.dump() Macro.dump()
def visitMacroBegin(self, node, env): def visitMacroBegin(self, node, env):
@ -392,7 +392,7 @@ class ExtendBranches(PCTracker):
# If BRA - BRanch Always - is out of range, it's a JMP. # If BRA - BRanch Always - is out of range, it's a JMP.
node.data = ('jmp', expr) node.data = ('jmp', expr)
node.nodetype = "Absolute" node.nodetype = "Absolute"
if Cmd.verbose > 0: if Cmd.warn_on_branch_extend:
print>>sys.stderr, str(node.ppt) + ": WARNING: bra out of range, replacing with jmp" print>>sys.stderr, str(node.ppt) + ": WARNING: bra out of range, replacing with jmp"
else: else:
# Otherwise, we replace it with a 'macro' of sorts by hand: # Otherwise, we replace it with a 'macro' of sorts by hand:
@ -403,7 +403,7 @@ class ExtendBranches(PCTracker):
IR.Node(node.ppt, "Absolute", 'jmp', expr)] IR.Node(node.ppt, "Absolute", 'jmp', expr)]
node.nodetype='SEQUENCE' node.nodetype='SEQUENCE'
node.data = expansion node.data = expansion
if Cmd.verbose > 0: if Cmd.warn_on_branch_extend:
print>>sys.stderr, str(node.ppt) + ": WARNING: "+opcode+" out of range, replacing with "+ExtendBranches.reversed[opcode] +"/jmp combo" print>>sys.stderr, str(node.ppt) + ": WARNING: "+opcode+" out of range, replacing with "+ExtendBranches.reversed[opcode] +"/jmp combo"
self.expanded += 1 self.expanded += 1
node.accept(self, env) node.accept(self, env)
@ -435,7 +435,7 @@ class Assembler(Pass):
self.filler = 0 self.filler = 0
def postPass(self): def postPass(self):
if Cmd.verbose > 0 and Err.count == 0: if Cmd.print_summary and Err.count == 0:
print>>sys.stderr, "Assembly complete: %s bytes output (%s code, %s data, %s filler)" \ print>>sys.stderr, "Assembly complete: %s bytes output (%s code, %s data, %s filler)" \
% (len(self.output), self.code, self.data, self.filler) % (len(self.output), self.code, self.data, self.filler)

View File

@ -1,4 +1,5 @@
#!/usr/local/bin/python #!/usr/local/bin/python
import sys
import Ophis.Main import Ophis.Main
Ophis.Main.run_ophis() Ophis.Main.run_ophis(sys.argv[1:])