2017-01-30 09:24:43 +08:00
|
|
|
#!/usr/bin/env python3
|
2015-09-07 06:08:57 +08:00
|
|
|
# Copyright (C) 2014, 2015 M-Labs Limited
|
|
|
|
# Copyright (C) 2014, 2015 Robert Jordens <jordens@gmail.com>
|
2014-12-03 18:20:30 +08:00
|
|
|
|
|
|
|
import argparse
|
|
|
|
import sys
|
2023-10-19 17:16:26 +08:00
|
|
|
import tarfile
|
2014-12-03 18:20:30 +08:00
|
|
|
from operator import itemgetter
|
2015-04-05 10:42:37 +08:00
|
|
|
import logging
|
2016-02-25 06:34:34 +08:00
|
|
|
from collections import defaultdict
|
2014-12-03 18:20:30 +08:00
|
|
|
|
2015-02-04 18:46:55 +08:00
|
|
|
import h5py
|
|
|
|
|
2021-08-12 11:54:36 +08:00
|
|
|
from llvmlite import binding as llvm
|
2016-02-25 06:34:34 +08:00
|
|
|
|
2024-02-26 19:30:31 +08:00
|
|
|
from sipyco import common_args, pyon
|
2019-11-10 15:55:17 +08:00
|
|
|
|
2019-11-14 11:42:31 +08:00
|
|
|
from artiq import __version__ as artiq_version
|
2016-04-16 19:31:07 +08:00
|
|
|
from artiq.language.environment import EnvExperiment, ProcessArgumentManager
|
2016-06-27 14:37:29 +08:00
|
|
|
from artiq.language.types import TBool
|
2015-10-12 17:18:23 +08:00
|
|
|
from artiq.master.databases import DeviceDB, DatasetDB
|
|
|
|
from artiq.master.worker_db import DeviceManager, DatasetManager
|
2016-01-16 01:11:26 +08:00
|
|
|
from artiq.coredevice.core import CompileError, host_only
|
2016-05-16 22:30:21 +08:00
|
|
|
from artiq.compiler.embedding import EmbeddingMap
|
2016-08-06 12:01:49 +08:00
|
|
|
from artiq.compiler import import_cache
|
2015-08-28 14:54:51 +08:00
|
|
|
from artiq.tools import *
|
2014-12-03 18:20:30 +08:00
|
|
|
|
2016-04-12 15:16:48 +08:00
|
|
|
|
2015-04-05 10:42:37 +08:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2016-04-12 15:16:48 +08:00
|
|
|
|
2016-02-25 06:34:34 +08:00
|
|
|
class StubObject:
|
|
|
|
def __setattr__(self, name, value):
|
|
|
|
pass
|
|
|
|
|
2016-04-12 15:16:48 +08:00
|
|
|
|
2016-05-16 22:30:21 +08:00
|
|
|
class StubEmbeddingMap:
|
2016-02-25 06:34:34 +08:00
|
|
|
def __init__(self):
|
|
|
|
stub_object = StubObject()
|
2016-05-16 22:30:21 +08:00
|
|
|
self.object_forward_map = defaultdict(lambda: stub_object)
|
|
|
|
self.object_forward_map[1] = lambda _: None # return RPC
|
|
|
|
self.object_current_id = -1
|
2016-02-25 06:34:34 +08:00
|
|
|
|
2016-05-16 22:30:21 +08:00
|
|
|
def retrieve_object(self, object_id):
|
|
|
|
return self.object_forward_map[object_id]
|
2015-04-05 10:42:37 +08:00
|
|
|
|
2016-05-16 22:30:21 +08:00
|
|
|
def store_object(self, value):
|
|
|
|
self.object_forward_map[self.object_current_id] = value
|
|
|
|
self.object_current_id -= 1
|
2016-02-25 06:34:34 +08:00
|
|
|
|
2016-04-12 15:16:48 +08:00
|
|
|
|
2016-02-25 06:34:34 +08:00
|
|
|
class FileRunner(EnvExperiment):
|
2016-08-17 18:20:04 +08:00
|
|
|
def build(self, file):
|
2015-10-04 00:18:21 +08:00
|
|
|
self.setattr_device("core")
|
2016-08-17 18:20:04 +08:00
|
|
|
self.file = file
|
2021-08-12 11:37:45 +08:00
|
|
|
self.target = self.core.target_cls()
|
2014-12-03 18:20:30 +08:00
|
|
|
|
2015-04-05 10:42:37 +08:00
|
|
|
def run(self):
|
2016-02-25 06:34:34 +08:00
|
|
|
kernel_library = self.compile()
|
2015-08-28 14:43:46 +08:00
|
|
|
|
|
|
|
self.core.comm.load(kernel_library)
|
|
|
|
self.core.comm.run()
|
2016-05-16 22:30:21 +08:00
|
|
|
self.core.comm.serve(StubEmbeddingMap(),
|
2016-08-17 18:20:04 +08:00
|
|
|
lambda addresses: self.target.symbolize(kernel_library, addresses), \
|
|
|
|
lambda symbols: self.target.demangle(symbols))
|
2016-02-25 06:34:34 +08:00
|
|
|
|
2016-04-12 15:16:48 +08:00
|
|
|
|
2016-02-25 06:34:34 +08:00
|
|
|
class ELFRunner(FileRunner):
|
|
|
|
def compile(self):
|
|
|
|
with open(self.file, "rb") as f:
|
|
|
|
return f.read()
|
|
|
|
|
2016-04-12 15:16:48 +08:00
|
|
|
|
2016-02-25 06:34:34 +08:00
|
|
|
class LLVMIRRunner(FileRunner):
|
|
|
|
def compile(self):
|
|
|
|
with open(self.file, "r") as f:
|
|
|
|
llmodule = llvm.parse_assembly(f.read())
|
|
|
|
llmodule.verify()
|
2016-11-08 20:08:20 +08:00
|
|
|
return self.target.link([self.target.assemble(llmodule)])
|
2016-04-12 15:16:48 +08:00
|
|
|
|
2016-02-25 06:34:34 +08:00
|
|
|
|
|
|
|
class LLVMBitcodeRunner(FileRunner):
|
|
|
|
def compile(self):
|
|
|
|
with open(self.file, "rb") as f:
|
|
|
|
llmodule = llvm.parse_bitcode(f.read())
|
|
|
|
llmodule.verify()
|
2016-11-08 20:08:20 +08:00
|
|
|
return self.target.link([self.target.assemble(llmodule)])
|
2014-12-03 18:20:30 +08:00
|
|
|
|
|
|
|
|
2023-10-19 17:16:26 +08:00
|
|
|
class TARRunner(FileRunner):
|
|
|
|
def compile(self):
|
|
|
|
with tarfile.open(self.file, "r:") as tar:
|
|
|
|
for entry in tar:
|
|
|
|
if entry.name == 'main.elf':
|
|
|
|
main_lib = tar.extractfile(entry).read()
|
|
|
|
else:
|
|
|
|
subkernel_name = entry.name.removesuffix(".elf")
|
|
|
|
sid, dest = tuple(map(lambda x: int(x), subkernel_name.split(" ")))
|
|
|
|
subkernel_lib = tar.extractfile(entry).read()
|
|
|
|
self.core.comm.upload_subkernel(subkernel_lib, sid, dest)
|
|
|
|
return main_lib
|
|
|
|
|
|
|
|
|
2015-02-20 03:09:37 +08:00
|
|
|
class DummyScheduler:
|
2015-07-14 04:08:20 +08:00
|
|
|
def __init__(self):
|
2015-10-30 13:41:18 +08:00
|
|
|
self.rid = 0
|
2015-05-28 17:21:20 +08:00
|
|
|
self.pipeline_name = "main"
|
|
|
|
self.priority = 0
|
2015-07-14 04:08:20 +08:00
|
|
|
self.expid = None
|
2015-02-20 03:09:37 +08:00
|
|
|
|
2015-10-30 13:41:18 +08:00
|
|
|
self._next_rid = 1
|
|
|
|
|
2016-10-18 13:49:43 +08:00
|
|
|
def submit(self, pipeline_name=None, expid=None, priority=None, due_date=None, flush=False):
|
2015-10-30 13:41:18 +08:00
|
|
|
rid = self._next_rid
|
|
|
|
self._next_rid += 1
|
2015-05-28 17:21:20 +08:00
|
|
|
logger.info("Submitting: %s, RID=%s", expid, rid)
|
2015-02-20 03:09:37 +08:00
|
|
|
return rid
|
|
|
|
|
2015-05-28 17:21:20 +08:00
|
|
|
def delete(self, rid):
|
|
|
|
logger.info("Deleting RID %s", rid)
|
2015-02-20 03:09:37 +08:00
|
|
|
|
2015-10-30 13:41:18 +08:00
|
|
|
def request_termination(self, rid):
|
|
|
|
logger.info("Requesting termination of RID %s", rid)
|
|
|
|
|
|
|
|
def get_status(self):
|
|
|
|
return dict()
|
|
|
|
|
2016-06-27 14:37:29 +08:00
|
|
|
def check_pause(self, rid=None) -> TBool:
|
|
|
|
return False
|
|
|
|
|
2016-01-16 00:35:12 +08:00
|
|
|
@host_only
|
2015-08-15 09:16:00 +08:00
|
|
|
def pause(self):
|
|
|
|
pass
|
|
|
|
|
2015-02-20 03:09:37 +08:00
|
|
|
|
2016-09-05 00:53:44 +08:00
|
|
|
class DummyCCB:
|
|
|
|
def issue(self, service, *args, **kwargs):
|
|
|
|
logger.info("CCB for service '%s' (args %s, kwargs %s)",
|
|
|
|
service, args, kwargs)
|
|
|
|
|
|
|
|
|
2015-04-05 11:17:24 +08:00
|
|
|
def get_argparser(with_file=True):
|
2015-01-12 18:51:23 +08:00
|
|
|
parser = argparse.ArgumentParser(
|
|
|
|
description="Local experiment running tool")
|
2019-11-14 11:42:31 +08:00
|
|
|
parser.add_argument("--version", action="version",
|
|
|
|
version="ARTIQ v{}".format(artiq_version),
|
|
|
|
help="print the ARTIQ version number")
|
2014-12-03 18:20:30 +08:00
|
|
|
|
2019-11-10 15:55:17 +08:00
|
|
|
common_args.verbosity_args(parser)
|
2017-05-18 23:14:20 +08:00
|
|
|
parser.add_argument("--device-db", default="device_db.py",
|
2015-10-12 17:18:23 +08:00
|
|
|
help="device database file (default: '%(default)s')")
|
2023-04-24 17:34:30 +08:00
|
|
|
parser.add_argument("--dataset-db", default="dataset_db.mdb",
|
2015-10-12 17:18:23 +08:00
|
|
|
help="dataset file (default: '%(default)s')")
|
2014-12-03 18:20:30 +08:00
|
|
|
|
2019-09-09 15:16:33 +08:00
|
|
|
parser.add_argument("-c", "--class-name", default=None,
|
|
|
|
help="name of the class to run")
|
2015-02-04 18:46:55 +08:00
|
|
|
parser.add_argument("-o", "--hdf5", default=None,
|
|
|
|
help="write results to specified HDF5 file"
|
|
|
|
" (default: print them)")
|
2015-04-05 10:42:37 +08:00
|
|
|
if with_file:
|
2016-04-22 16:33:44 +08:00
|
|
|
parser.add_argument("file", metavar="FILE",
|
2015-04-05 10:42:37 +08:00
|
|
|
help="file containing the experiment to run")
|
2016-04-22 16:33:44 +08:00
|
|
|
parser.add_argument("arguments", metavar="ARGUMENTS", nargs="*",
|
2015-01-07 19:21:17 +08:00
|
|
|
help="run arguments")
|
2014-12-03 18:20:30 +08:00
|
|
|
|
2015-01-23 00:52:13 +08:00
|
|
|
return parser
|
2014-12-03 18:20:30 +08:00
|
|
|
|
|
|
|
|
2024-02-26 19:30:31 +08:00
|
|
|
class ArgumentManager(ProcessArgumentManager):
|
2024-03-11 17:07:25 +08:00
|
|
|
def get_interactive(self, interactive_arglist, title):
|
|
|
|
print(title)
|
2024-02-26 19:30:31 +08:00
|
|
|
result = dict()
|
|
|
|
for key, processor, group, tooltip in interactive_arglist:
|
|
|
|
success = False
|
|
|
|
while not success:
|
|
|
|
user_input = input("{}:{} (group={}, tooltip={}): ".format(
|
|
|
|
key, type(processor).__name__, group, tooltip))
|
|
|
|
try:
|
|
|
|
user_input_deser = pyon.decode(user_input)
|
|
|
|
value = processor.process(user_input_deser)
|
|
|
|
except:
|
2024-02-27 11:10:21 +08:00
|
|
|
logger.error("failed to process user input, retrying",
|
|
|
|
exc_info=True)
|
2024-02-26 19:30:31 +08:00
|
|
|
else:
|
|
|
|
success = True
|
|
|
|
result[key] = value
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
2015-10-12 17:18:23 +08:00
|
|
|
def _build_experiment(device_mgr, dataset_mgr, args):
|
2016-08-17 18:20:04 +08:00
|
|
|
arguments = parse_arguments(args.arguments)
|
2024-02-26 19:30:31 +08:00
|
|
|
argument_mgr = ArgumentManager(arguments)
|
2019-03-13 23:20:41 +08:00
|
|
|
managers = (device_mgr, dataset_mgr, argument_mgr, {})
|
2015-04-05 10:42:37 +08:00
|
|
|
if hasattr(args, "file"):
|
2023-10-19 17:16:26 +08:00
|
|
|
is_tar = tarfile.is_tarfile(args.file)
|
2016-02-25 06:34:34 +08:00
|
|
|
is_elf = args.file.endswith(".elf")
|
|
|
|
is_ll = args.file.endswith(".ll")
|
|
|
|
is_bc = args.file.endswith(".bc")
|
|
|
|
if is_elf or is_ll or is_bc:
|
2015-04-05 10:42:37 +08:00
|
|
|
if args.arguments:
|
2016-02-25 06:34:34 +08:00
|
|
|
raise ValueError("arguments not supported for precompiled kernels")
|
2019-09-09 15:16:33 +08:00
|
|
|
if args.class_name:
|
|
|
|
raise ValueError("class-name not supported "
|
2016-02-25 06:34:34 +08:00
|
|
|
"for precompiled kernels")
|
2023-10-19 17:16:26 +08:00
|
|
|
if is_tar:
|
|
|
|
return TARRunner(managers, file=args.file)
|
|
|
|
elif is_elf:
|
2016-08-17 18:20:04 +08:00
|
|
|
return ELFRunner(managers, file=args.file)
|
2016-02-25 06:34:34 +08:00
|
|
|
elif is_ll:
|
2016-08-17 18:20:04 +08:00
|
|
|
return LLVMIRRunner(managers, file=args.file)
|
2016-02-25 06:34:34 +08:00
|
|
|
elif is_bc:
|
2016-08-17 18:20:04 +08:00
|
|
|
return LLVMBitcodeRunner(managers, file=args.file)
|
2015-04-05 10:42:37 +08:00
|
|
|
else:
|
2016-08-06 12:01:49 +08:00
|
|
|
import_cache.install_hook()
|
2015-08-28 15:22:59 +08:00
|
|
|
module = file_import(args.file, prefix="artiq_run_")
|
2015-04-05 10:42:37 +08:00
|
|
|
file = args.file
|
|
|
|
else:
|
|
|
|
module = sys.modules["__main__"]
|
|
|
|
file = getattr(module, "__file__")
|
2015-05-28 17:21:20 +08:00
|
|
|
expid = {
|
|
|
|
"file": file,
|
2019-09-09 15:16:33 +08:00
|
|
|
"class_name": args.class_name,
|
2015-05-28 17:21:20 +08:00
|
|
|
"arguments": arguments
|
|
|
|
}
|
2015-10-12 17:18:23 +08:00
|
|
|
device_mgr.virtual_devices["scheduler"].expid = expid
|
2022-11-23 22:34:21 +08:00
|
|
|
exp_inst = get_experiment(module, args.class_name)(managers)
|
|
|
|
argument_mgr.check_unprocessed_arguments()
|
|
|
|
return exp_inst
|
2015-04-05 10:42:37 +08:00
|
|
|
|
|
|
|
|
|
|
|
def run(with_file=False):
|
|
|
|
args = get_argparser(with_file).parse_args()
|
2019-11-10 15:55:17 +08:00
|
|
|
common_args.init_logger_from_args(args)
|
2014-12-03 18:20:30 +08:00
|
|
|
|
2015-10-12 17:18:23 +08:00
|
|
|
device_mgr = DeviceManager(DeviceDB(args.device_db),
|
2016-09-05 00:53:44 +08:00
|
|
|
virtual_devices={"scheduler": DummyScheduler(),
|
|
|
|
"ccb": DummyCCB()})
|
2015-10-12 17:18:23 +08:00
|
|
|
dataset_db = DatasetDB(args.dataset_db)
|
2015-04-05 17:49:41 +08:00
|
|
|
try:
|
2023-04-24 17:34:30 +08:00
|
|
|
dataset_mgr = DatasetManager(dataset_db)
|
|
|
|
|
|
|
|
try:
|
|
|
|
exp_inst = _build_experiment(device_mgr, dataset_mgr, args)
|
|
|
|
exp_inst.prepare()
|
|
|
|
exp_inst.run()
|
2023-12-13 14:27:04 +08:00
|
|
|
device_mgr.notify_run_end()
|
2023-04-24 17:34:30 +08:00
|
|
|
exp_inst.analyze()
|
|
|
|
except CompileError as error:
|
|
|
|
return
|
|
|
|
except Exception as exn:
|
|
|
|
if hasattr(exn, "artiq_core_exception"):
|
|
|
|
print(exn.artiq_core_exception, file=sys.stderr)
|
|
|
|
raise exn
|
|
|
|
finally:
|
|
|
|
device_mgr.close_devices()
|
|
|
|
|
|
|
|
if args.hdf5 is not None:
|
|
|
|
with h5py.File(args.hdf5, "w") as f:
|
|
|
|
dataset_mgr.write_hdf5(f)
|
|
|
|
else:
|
|
|
|
for k, v in sorted(dataset_mgr.local.items(), key=itemgetter(0)):
|
|
|
|
print("{}: {}".format(k, v))
|
|
|
|
dataset_db.save()
|
2015-04-05 17:49:41 +08:00
|
|
|
finally:
|
2023-04-24 17:34:30 +08:00
|
|
|
dataset_db.close_db()
|
2015-04-05 10:42:37 +08:00
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
return run(with_file=True)
|
|
|
|
|
2014-12-03 18:20:30 +08:00
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|