Source code for quantify_scheduler.compilation
# Repository: https://gitlab.com/quantify-os/quantify-scheduler
# Licensed according to the LICENCE file on the main branch
"""Compiler for the quantify_scheduler."""
from __future__ import annotations
import logging
from typing import Literal, Optional
from quantify_scheduler.backends.graph_compilation import (
CompilationConfig,
)
from quantify_scheduler.json_utils import load_json_schema, validate_json
from quantify_scheduler.schedules.schedule import Schedule
[docs]logger = logging.getLogger(__name__)
[docs]def determine_absolute_timing(
schedule: Schedule,
time_unit: Literal[
"physical", "ideal", None
] = "physical", # should be included in CompilationConfig
config: Optional[CompilationConfig] = None,
) -> Schedule:
"""
Determines the absolute timing of a schedule based on the timing constraints.
This function determines absolute timings for every operation in the
:attr:`~.ScheduleBase.schedulables`. It does this by:
1. iterating over all and elements in the :attr:`~.ScheduleBase.schedulables`.
2. determining the absolute time of the reference operation
- reference point :code:`"ref_pt"` of the reference operation defaults to :code:`"end"` in case it is not set (i.e., is :code:`None`).
3. determining the start of the operation based on the :code:`rel_time` and :code:`duration`
of operations
- reference point :code:`"ref_pt_new"` of the added operation defaults to :code:`"start"` in case it is not set.
Parameters
----------
schedule
The schedule for which to determine timings.
config
Compilation config for
:class:`~quantify_scheduler.backends.graph_compilation.QuantifyCompiler`,
which is currently not used in this compilation step.
time_unit
Whether to use physical units to determine the absolute time or ideal time.
When :code:`time_unit == "physical"` the duration attribute is used.
When :code:`time_unit == "ideal"` the duration attribute is ignored and treated
as if it is :code:`1`.
When :code:`time_unit == None` it will revert to :code:`"physical"`.
Returns
-------
:
a new schedule object where the absolute time for each operation has been
determined.
"""
if len(schedule.schedulables) == 0:
raise ValueError(f"schedule '{schedule.name}' contains no schedulables.")
if time_unit is None:
time_unit = "physical"
valid_time_units = ("physical", "ideal")
if time_unit not in valid_time_units:
raise ValueError(
f"Undefined time_unit '{time_unit}'! Must be one of {valid_time_units}"
)
# iterate over the objects in the schedule.
last_schedulable = next(iter(schedule.schedulables.values()))
last_op = schedule.operations[last_schedulable["operation_repr"]]
last_schedulable["abs_time"] = 0
for schedulable in list(schedule.schedulables.values())[1:]:
curr_op = schedule.operations[schedulable["operation_repr"]]
if len(schedulable["timing_constraints"]) == 0:
schedulable.add_timing_constraint(ref_schedulable=last_schedulable)
for t_constr in schedulable["timing_constraints"]:
if t_constr["ref_schedulable"] is None:
ref_schedulable = last_schedulable
ref_op = last_op
else:
# this assumes the reference op exists. This is ensured in schedule.add
ref_schedulable = schedule.schedulables[
str(t_constr["ref_schedulable"])
]
ref_op = schedule.operations[ref_schedulable["operation_repr"]]
# duration = 1 is useful when e.g., drawing a circuit diagram.
duration_ref_op = ref_op.duration if time_unit == "physical" else 1
ref_pt = t_constr["ref_pt"] or "end"
if ref_pt == "start":
t0 = ref_schedulable["abs_time"]
elif ref_pt == "center":
t0 = ref_schedulable["abs_time"] + duration_ref_op / 2
elif ref_pt == "end":
t0 = ref_schedulable["abs_time"] + duration_ref_op
else:
raise NotImplementedError(
f'Timing "{ref_schedulable["abs_time"]}" not supported by backend.'
)
duration_new_op = curr_op.duration if time_unit == "physical" else 1
ref_pt_new = t_constr["ref_pt_new"] or "start"
if ref_pt_new == "start":
abs_time = t0 + t_constr["rel_time"]
elif ref_pt_new == "center":
abs_time = t0 + t_constr["rel_time"] - duration_new_op / 2
elif ref_pt_new == "end":
abs_time = t0 + t_constr["rel_time"] - duration_new_op
if "abs_time" not in schedulable or abs_time > schedulable["abs_time"]:
schedulable["abs_time"] = abs_time
# update last_constraint and operation for next iteration of the loop
last_schedulable = schedulable
last_op = curr_op
return schedule
[docs]def validate_config(config: dict, scheme_fn: str) -> bool:
"""
Validate a configuration using a schema.
Parameters
----------
config
The configuration to validate
scheme_fn
The name of a json schema in the quantify_scheduler.schemas folder.
Returns
----------
:
True if valid
"""
scheme = load_json_schema(__file__, scheme_fn)
validate_json(config, scheme)
return True