forked from M-Labs/artiq
198 lines
14 KiB
ReStructuredText
198 lines
14 KiB
ReStructuredText
Management system
|
|
=================
|
|
|
|
.. note::
|
|
The ARTIQ management system as described here is optional. Experiments can be run one-by-one using :mod:`~artiq.frontend.artiq_run`, and controllers can be run without a controller manager. For their very first steps with ARTIQ or in simple or particular cases, users do not need to deploy the management system. For an introduction to the system and how to use it, see :doc:`getting_started_mgmt`.
|
|
|
|
Components
|
|
----------
|
|
|
|
Master
|
|
^^^^^^
|
|
|
|
The :ref:`ARTIQ master <frontend-artiq-master>` is responsible for managing the parameter and device databases, the experiment repository, scheduling and running experiments, archiving results, and distributing real-time results. It is a headless component, and one or several clients (command-line or GUI) use the network to interact with it. It also uses the network to interact with any relevant controller managers.
|
|
|
|
It should not be confused with the 'master' device in a DRTIO system, which is simply a designation for the particular core device acting as central node in a distributed configuration of ARTIQ. The two concepts are otherwise unrelated.
|
|
|
|
Command-line client
|
|
^^^^^^^^^^^^^^^^^^^
|
|
|
|
The :ref:`command-line client <frontend-artiq-client>` connects to the master and permits modification and monitoring of the databases, monitoring the experiment schedule and log, and submitting experiments.
|
|
|
|
Dashboard
|
|
^^^^^^^^^
|
|
|
|
The :ref:`dashboard <frontend-artiq-dashboard>` connects to the master and is the main method of interacting with it. The main features of the dashboard are scheduling of experiments, setting of their arguments, examining the schedule, displaying real-time results, and debugging TTL and DDS channels in real time.
|
|
|
|
Controller manager
|
|
^^^^^^^^^^^^^^^^^^
|
|
|
|
The controller manager is provided in the ``artiq-comtools`` package and started with the ``artiq_ctlmgr`` command. It is responsible for running and stopping controllers on a machine. One controller manager must be run by each network node that runs controllers.
|
|
|
|
A controller manager connects to the master and accesses the device database through it to determine what controllrs need to be run. The local network address of the connection is used to filter for only those controllers allocated to the current node. Hostname resolution is supported. Changes to the device database are tracked and controllers will be stopped and started accordingly.
|
|
|
|
.. warning::
|
|
With some network setups, the current machine's hostname without the domain name resolves to a localhost address (127.0.0.1 or even 127.0.1.1). If you wish to use controllers across a network, make sure that the hostname you provide resolves to an IP address visible on the network (e.g. try providing the full hostname including the domain name).
|
|
|
|
Git integration
|
|
---------------
|
|
|
|
The master may use a Git repository to store experiment source code. Using Git has many advantages. For example, each result file (HDF5) contains the commit ID corresponding to the exact source code it was produced by, which helps reproducibility.
|
|
|
|
Although the master also supports non-bare repositories, it is recommended to use a bare repository (e.g. ``git init --bare``) to easily support push transactions from clients.
|
|
|
|
You will want Git to notify the master every time the repository is pushed to (e.g. updated), so that the master knows to rescan the repository for new or changed experiments. This is easiest done with the ``post-receive`` hook, as described in :ref:`master-setting-up-git`.
|
|
|
|
.. note::
|
|
If you plan to run the ARTIQ system entirely on a single machine, you may also consider using a non-bare repository and the ``post-commit`` hook to trigger repository scans every time you commit changes (locally). In this case, note that the ARTIQ master never uses the repository's working directory, but only what is committed. More precisely, when scanning the repository, it fetches the last (atomically) completed commit at that time of repository scan and checks it out in a temporary folder. This commit ID is used by default when subsequently submitting experiments. There is one temporary folder by commit ID currently referenced in the system, so concurrently running experiments from different repository revisions is fully supported by the master.
|
|
|
|
The dashboard always runs experiments from the repository. By default the command-line client (``artiq_client submit``) runs experiments from the raw filesystem (which is useful for iterating rapidly without creating many disorganized commits). If you want to use the repository instead, simply pass the ``-R`` option.
|
|
|
|
.. _experiment-scheduling:
|
|
|
|
Experiment scheduling
|
|
---------------------
|
|
|
|
Basics
|
|
^^^^^^
|
|
|
|
To make more efficient use of hardware resources, experiments are generally split into three phases and pipelined, such that potentially compute-intensive pre-computation or analysis phases may be executed in parallel with the bodies of other experiments, which access hardware.
|
|
|
|
.. seealso::
|
|
These steps are implemented in :class:`~artiq.language.environment.Experiment`. However, user-written experiments should usually derive from (sub-class) :class:`artiq.language.environment.EnvExperiment`.
|
|
|
|
There are three stages of a standard experiment users may write code in:
|
|
|
|
1. The **preparation** stage, which pre-fetches and pre-computes any data that necessary to run the experiment. Users may implement this stage by overloading the :meth:`~artiq.language.environment.Experiment.prepare` method. It is not permitted to access hardware in this stage, as doing so may conflict with other experiments using the same devices.
|
|
2. The **run** stage, which corresponds to the body of the experiment and generally accesses hardware. Users must implement this stage and overload the :meth:`~artiq.language.environment.Experiment.run` method.
|
|
3. The **analysis** stage, where raw results collected in the running stage are post-processed and may lead to updates of the parameter database. This stage may be implemented by overloading the :meth:`~artiq.language.environment.Experiment.analyze` method.
|
|
|
|
Only the :meth:`~artiq.language.environment.Experiment.run` method implementation is mandatory; if the experiment does not fit into the pipelined scheduling model, it can leave one or both of the other methods empty (which is the default).
|
|
|
|
Consecutive experiments are then executed in a pipelined manner by the ARTIQ master's scheduler: first experiment A runs its preparation stage, than experiment A executes its running stage while experiment B executes its preparation stage, and so on.
|
|
|
|
.. note::
|
|
The next experiment (B) may start its :meth:`~artiq.language.environment.Experiment.run` before all events placed into (core device) RTIO buffers by the previous experiment (A) have been executed. These events may then execute while experiment B's :meth:`~artiq.language.environment.Experiment.run` is already in progress. Using :meth:`~artiq.coredevice.core.Core.reset` in experiment B will clear the RTIO buffers, discarding pending events, including those left over from A.
|
|
|
|
Interactions between events of different experiments can be avoided by preventing the :meth:`~artiq.language.environment.Experiment.run` method of experiment A from returning until all events have been executed. This is discussed in the section on RTIO :ref:`rtio-handover-synchronization`.
|
|
|
|
Priorities and timed runs
|
|
^^^^^^^^^^^^^^^^^^^^^^^^^
|
|
|
|
When determining what experiment should begin executing next (i.e. enter the preparation stage), the scheduling looks at the following factors, by decreasing order of precedence:
|
|
|
|
1. Experiments may be scheduled with a due date. This is considered the *earliest possible* time of their execution (rather than a deadline, or latest possible -- ARTIQ makes no guarantees about experiments being started or completed before any specified time). If a due date is set and it has not yet been reached, the experiment is not eligible for preparation.
|
|
2. The integer priority value specified by the user.
|
|
3. The due date itself. The earliest (reached) due date will be scheduled first.
|
|
4. The run identifier (RID), an integer that is incremented at each experiment submission. This ensures that, all else being equal, experiments are scheduled in the same order as they are submitted.
|
|
|
|
Multiple pipelines
|
|
^^^^^^^^^^^^^^^^^^
|
|
|
|
Experiments must be placed into a pipeline at submission time, set by the "Pipeline" field. Multiple pipelines can operate in parallel within the same master. Pipelines are identified by their names, and are automtically created (when an experiment is scheduled with a pipeline name that does not yet exist) and destroyed (when they run empty). By default, all experiments are submitted into the same pipeline, ``main``.
|
|
|
|
When using multiple pipelines it is the responsibility of the user to ensure that experiments scheduled in parallel will never conflict with those of another pipeline over resources (e.g. attempt to use the same devices simultaneously).
|
|
|
|
Pauses
|
|
^^^^^^
|
|
|
|
In the run stage, an experiment may yield to the scheduler by calling the ``pause()`` method of the scheduler.
|
|
If there are other experiments with higher priority (e.g. a high-priority experiment has been newly submitted, or reached its due date and become eligible for execution), the higher-priority experiments are executed first, and then ``pause()`` returns. If there are no such experiments, ``pause()`` returns immediately. To check whether ``pause()`` would in fact *not* return immediately, use :meth:`artiq.master.scheduler.Scheduler.check_pause`.
|
|
|
|
The experiment must place the hardware in a safe state and disconnect from the core device (typically, by calling ``self.core.comm.close()`` from the kernel, which is equivalent to :meth:`artiq.coredevice.core.Core.close`) before calling ``pause()``.
|
|
|
|
Accessing the ``pause()`` and :meth:`~artiq.master.scheduler.Scheduler.check_pause` methods is done through a virtual device called ``scheduler`` that is accessible to all experiments. The scheduler virtual device is requested like regular devices using :meth:`~artiq.language.environment.HasEnvironment.get_device` (``self.get_device()``) or :meth:`~artiq.language.environment.HasEnvironment.setattr_device` (``self.setattr_device()``).
|
|
|
|
:meth:`~artiq.master.scheduler.Scheduler.check_pause` can be called (via RPC) from a kernel, but ``pause()`` must not be.
|
|
|
|
Scheduler API reference
|
|
-----------------------
|
|
|
|
The scheduler is exposed to the experiments via a virtual device called ``scheduler``. It can be requested like any regular device, and then the methods below can be called on the returned object.
|
|
|
|
The scheduler virtual device also contains the attributes ``rid``, ``pipeline_name``, ``priority`` and ``expid``, which contain the corresponding information about the current run.
|
|
|
|
.. autoclass:: artiq.master.scheduler.Scheduler
|
|
:members:
|
|
|
|
Client control broadcasts (CCBs)
|
|
--------------------------------
|
|
|
|
Client control broadcasts are requests made by experiments for clients to perform some action. Experiments do so by requesting the ``ccb`` virtual device and calling its ``issue`` method. The first argument of the issue method is the name of the broadcast, and any further positional and keyword arguments are passed to the broadcast.
|
|
|
|
CCBs are used by experiments to configure applets in the dashboard, for example for plotting purposes.
|
|
|
|
.. autoclass:: artiq.dashboard.applets_ccb.AppletsCCBDock
|
|
:members:
|
|
|
|
.. _applet-references:
|
|
|
|
Applet request interfaces
|
|
-------------------------
|
|
|
|
Applet request interfaces allow applets to perform actions on the master database and set arguments in the dashboard. Applets may inherit from the ``artiq.applets.simple.SimpleApplet`` and call the methods defined below through the `req` attribute.
|
|
|
|
Embedded applets should use `AppletRequestIPC` while standalone applets use `AppletRequestRPC`. `SimpleApplet` automatically chooses the correct interface on initialization.
|
|
|
|
.. autoclass:: artiq.applets.simple._AppletRequestInterface
|
|
:members:
|
|
|
|
|
|
Applet entry area
|
|
-----------------
|
|
|
|
Argument widgets can be used in applets through the `EntryArea` class.
|
|
|
|
Below is a simple example code snippet using the `EntryArea` class: ::
|
|
|
|
entry_area = EntryArea()
|
|
|
|
# Create a new widget
|
|
entry_area.setattr_argument("bl", BooleanValue(True))
|
|
|
|
# Get the value of the widget (output: True)
|
|
print(entry_area.bl)
|
|
|
|
# Set the value
|
|
entry_area.set_value("bl", False)
|
|
|
|
# False
|
|
print(entry_area.bl)
|
|
|
|
The `EntryArea` object can then be added to a layout and integrated with the applet GUI. Multiple `EntryArea` objects can be used in a single applet.
|
|
|
|
.. class:: artiq.gui.applets.EntryArea
|
|
|
|
.. method:: setattr_argument(name, proc, group=None, tooltip=None)
|
|
|
|
Sets an argument as attribute. The names of the argument and of the
|
|
attribute are the same.
|
|
|
|
:param name: Argument name
|
|
:param proc: Argument processor, for example ``NumberValue``
|
|
:param group: Used to group together arguments in the GUI under a common category
|
|
:param tooltip: Tooltip displayed when hovering over the entry widget
|
|
|
|
.. method:: get_value(name)
|
|
|
|
Get the value of an entry widget.
|
|
|
|
:param name: Argument name
|
|
|
|
.. method:: get_values()
|
|
|
|
Get all values in the ``EntryArea`` as a dictionary. Names are stored as keys, and argument values as values.
|
|
|
|
.. method:: set_value(name, value)
|
|
|
|
Set the value of an entry widget. The change is temporary and will reset to default when the reset button is clicked.
|
|
|
|
:param name: Argument name
|
|
:param value: Object representing the new value of the argument. For ``Scannable`` arguments, this parameter
|
|
should be a ``ScanObject``. The type of the ``ScanObject`` will be set as the selected type when this function is called.
|
|
|
|
.. method:: set_values(values)
|
|
|
|
Set multiple values from a dictionary input. Calls ``set_value()`` for each key-value pair.
|
|
|
|
:param values: Dictionary with names as keys and new argument values as values. |