Added instruction set selector to CLI

This commit is contained in:
Anton Lydike 2021-04-19 12:51:43 +02:00
parent 8b1cbd97ec
commit cd17c65ce7
3 changed files with 31 additions and 6 deletions

View File

@ -35,7 +35,9 @@ Basic IO should work, as open, read, write and close are supported for stdin/std
This is how the interface is used:
```
riscemu [--options OPTIONS] [--syscall-option SYSCALL_OPTIONS] [--default_stack_size] file.asm [file.asm ...]
usage: riscemu [-h] [--options OPTIONS] [--syscall-opts SYSCALL_OPTS] [--instruction-sets INSTRUCTION_SETS] [--default_stack_size [default-stack-size]] file.asm [file.asm ...]
OPTIONS and SYSCALL_OPTIONS is a list of comma-separated flags that will be enabled
@ -44,9 +46,14 @@ disable_debug Disable the ebreak and sbreak instructions
no_syscall_symbols Don't make syscall symbols globally available
fail_on_ex Do not launch an interactive debugger when the CPU loop catches an exception
SYSCALL_OPTIONS:
SYSCALL_OPTS: Options to control syscall behaviour
fs_access Allow access to the filesystem
disable_io Disallow reading/writing from stdin/stdout/stderr
INSTRUCTION_SETS: A list of comma separated instruction sets you want to load:
Currently implemented: RV32I, RV32M
```
If multiple files are specified, all are loaded into memeory, but only the last one is executed. This might be improved

View File

@ -1,13 +1,14 @@
if __name__ == '__main__':
from . import *
from .helpers import *
from .instructions import RV32I, RV32M
from .instructions import InstructionSetDict
import argparse
import sys
all_ins_names = list(InstructionSetDict.keys())
class OptionStringAction(argparse.Action):
def __init__(self, option_strings, dest, keys=None, **kwargs):
def __init__(self, option_strings, dest, keys=None, omit_empty=False, **kwargs):
if keys is None:
raise ValueError('must define "keys" argument')
if isinstance(keys, dict):
@ -24,13 +25,17 @@ if __name__ == '__main__':
keys_d = dict()
super().__init__(option_strings, dest, default=keys_d, **kwargs)
self.keys = keys_d
self.omit_empty = omit_empty
def __call__(self, parser, namespace, values, option_string=None):
d = {}
d.update(self.keys)
if not self.omit_empty:
d.update(self.keys)
for x in values.split(','):
if x in self.keys:
d[x] = True
else:
raise ValueError('Invalid parameter supplied: ' + x)
setattr(namespace, self.dest, d)
@ -44,6 +49,10 @@ if __name__ == '__main__':
parser.add_argument('--syscall-opts', '-so', action=OptionStringAction,
keys=('fs_access', 'disable_input'))
parser.add_argument('--instruction-sets', '-is', action=OptionStringAction, help="Instruction sets to load, available are: {}. "
"All are enabled by default"
.format(", ".join(all_ins_names)), keys={k: True for k in all_ins_names}, omit_empty=True)
parser.add_argument('--default_stack_size', type=int, help='Default stack size of loaded programs', default=None,
metavar='default-stack-size', nargs='?')
@ -60,8 +69,13 @@ if __name__ == '__main__':
FMT_PRINT = FMT_BOLD + FMT_MAGENTA
# parse required instruction sets
ins_to_load = [
InstructionSetDict[name] for name, b in args.ins.items() if b
]
try:
cpu = CPU(cfg, [RV32I, RV32M])
cpu = CPU(cfg, ins_to_load)
loaded_exe = None
for file in args.files:
tk = cpu.get_tokenizer(RiscVInput.from_file(file))

View File

@ -1,3 +1,7 @@
from .InstructionSet import InstructionSet
from .RV32M import RV32M
from .RV32I import RV32I
InstructionSetDict = {
v.__name__: v for v in [RV32I, RV32M]
}