Source code for pennylane.devices.legacy_facade
# Copyright 2018-2024 Xanadu Quantum Technologies Inc.
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
# http://www.apache.org/licenses/LICENSE-2.0
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""
Defines a LegacyDeviceFacade class for converting legacy devices to the
new interface.
"""
# pylint: disable=not-callable, unused-argument
from contextlib import contextmanager
from copy import copy, deepcopy
from dataclasses import replace
import pennylane as qml
from pennylane.math import get_canonical_interface_name
from pennylane.measurements import MidMeasureMP, Shots
from pennylane.transforms.core.transform_program import TransformProgram
from .device_api import Device
from .execution_config import DefaultExecutionConfig
from .modifiers import single_tape_support
from .preprocess import (
decompose,
no_sampling,
validate_adjoint_trainable_params,
validate_measurements,
)
def _requests_adjoint(execution_config):
return execution_config.gradient_method == "adjoint" or (
execution_config.gradient_method == "device"
and execution_config.gradient_keyword_arguments.get("method", None) == "adjoint_jacobian"
)
@contextmanager
def _set_shots(device, shots):
"""Context manager to temporarily change the shots
of a device.
This context manager can be used in two ways.
As a standard context manager:
>>> with _set_shots(dev, shots=100):
... print(dev.shots)
100
>>> print(dev.shots)
None
Or as a decorator that acts on a function that uses the device:
>>> _set_shots(dev, shots=100)(lambda: dev.shots)()
100
"""
shots = qml.measurements.Shots(shots)
shots = shots.shot_vector if shots.has_partitioned_shots else shots.total_shots
if shots == device.shots:
yield
return
original_shots = device.shots
original_shot_vector = device._shot_vector # pylint: disable=protected-access
try:
device.shots = shots
yield
finally:
device.shots = original_shots
device._shot_vector = original_shot_vector # pylint: disable=protected-access
def null_postprocessing(results):
"""A postprocessing function with null behavior."""
return results[0]
@qml.transform
def legacy_device_expand_fn(tape, device):
"""Turn the ``expand_fn`` from the legacy device interface into a transform."""
new_tape = _set_shots(device, tape.shots)(device.expand_fn)(tape)
return (new_tape,), null_postprocessing
@qml.transform
def legacy_device_batch_transform(tape, device):
"""Turn the ``batch_transform`` from the legacy device interface into a transform."""
return _set_shots(device, tape.shots)(device.batch_transform)(tape)
def adjoint_ops(op: qml.operation.Operator) -> bool:
"""Specify whether or not an Operator is supported by adjoint differentiation."""
if isinstance(op, qml.QubitUnitary) and not qml.operation.is_trainable(op):
return True
return not isinstance(op, MidMeasureMP) and (
op.num_params == 0 or (op.num_params == 1 and op.has_generator)
)
def _add_adjoint_transforms(program: TransformProgram, name="adjoint"):
"""Add the adjoint specific transforms to the transform program."""
program.add_transform(no_sampling, name=name)
program.add_transform(
decompose,
stopping_condition=adjoint_ops,
name=name,
)
def accepted_adjoint_measurements(mp):
return isinstance(mp, qml.measurements.ExpectationMP)
program.add_transform(
validate_measurements,
analytic_measurements=accepted_adjoint_measurements,
name=name,
)
program.add_transform(qml.transforms.broadcast_expand)
program.add_transform(validate_adjoint_trainable_params)
[docs]@single_tape_support
class LegacyDeviceFacade(Device):
"""
A Facade that converts a device from the old ``qml.Device`` interface into the new interface.
Args:
device (qml.device.LegacyDevice): a device that follows the legacy device interface.
>>> from pennylane.devices import DefaultMixed, LegacyDeviceFacade
>>> legacy_dev = DefaultMixed(wires=2)
>>> new_dev = LegacyDeviceFacade(legacy_dev)
>>> new_dev.preprocess()
(TransformProgram(legacy_device_batch_transform, legacy_device_expand_fn, defer_measurements),
ExecutionConfig(grad_on_execution=None, use_device_gradient=None, use_device_jacobian_product=None,
gradient_method=None, gradient_keyword_arguments={}, device_options={}, interface=None,
derivative_order=1, mcm_config=MCMConfig(mcm_method=None, postselect_mode=None)))
>>> new_dev.shots
Shots(total_shots=None, shot_vector=())
>>> tape = qml.tape.QuantumScript([], [qml.sample(wires=0)], shots=5)
>>> new_dev.execute(tape)
array([0., 0., 0., 0., 0.])
"""
# pylint: disable=super-init-not-called
def __init__(self, device: "qml.devices.LegacyDevice"):
if isinstance(device, type(self)):
raise RuntimeError("An already-facaded device can not be wrapped in a facade again.")
if not isinstance(device, qml.devices.LegacyDevice):
raise ValueError(
"The LegacyDeviceFacade only accepts a device of type qml.devices.LegacyDevice."
)
self._device = device
self.config_filepath = getattr(self._device, "config_filepath", None)
@property
def tracker(self):
"""A :class:`~.Tracker` that can store information about device executions, shots, batches,
intermediate results, or any additional device dependent information.
"""
return self._device.tracker
@tracker.setter
def tracker(self, new_tracker):
self._device.tracker = new_tracker
@property
def name(self) -> str:
return self._device.short_name
def __repr__(self):
return f"<LegacyDeviceFacade: {repr(self._device)}>"
def __getattr__(self, name):
return getattr(self._device, name)
# These custom copy methods are needed for Catalyst
def __copy__(self):
return type(self)(copy(self.target_device))
def __deepcopy__(self, memo):
return type(self)(deepcopy(self.target_device, memo))
@property
def target_device(self) -> "qml.devices.LegacyDevice":
"""The device wrapped by the facade."""
return self._device
@property
def wires(self) -> qml.wires.Wires:
return self._device.wires
# pylint: disable=protected-access
@property
def shots(self) -> Shots:
if self._device._shot_vector:
return Shots(self._device._raw_shot_sequence)
return Shots(self._device.shots)
@property
def _debugger(self):
return self._device._debugger
@_debugger.setter
def _debugger(self, new_debugger):
self._device._debugger = new_debugger
[docs] def preprocess(self, execution_config=DefaultExecutionConfig):
execution_config = self._setup_execution_config(execution_config)
program = qml.transforms.core.TransformProgram()
program.add_transform(legacy_device_batch_transform, device=self._device)
program.add_transform(legacy_device_expand_fn, device=self._device)
if _requests_adjoint(execution_config):
_add_adjoint_transforms(program, name=f"{self.name} + adjoint")
if self._device.capabilities().get("supports_mid_measure", False):
program.add_transform(
qml.devices.preprocess.mid_circuit_measurements,
device=self,
mcm_config=execution_config.mcm_config,
)
else:
program.add_transform(qml.defer_measurements, allow_postselect=False)
return program, execution_config
def _setup_backprop_config(self, execution_config):
tape = qml.tape.QuantumScript()
if not self._validate_backprop_method(tape):
raise qml.DeviceError("device does not support backprop.")
if execution_config.use_device_gradient is None:
return replace(execution_config, use_device_gradient=True)
return execution_config
def _setup_adjoint_config(self, execution_config):
tape = qml.tape.QuantumScript([], [])
if not self._validate_adjoint_method(tape):
raise qml.DeviceError("device does not support device derivatives")
updated_values = {
"gradient_keyword_arguments": {"use_device_state": True, "method": "adjoint_jacobian"}
}
if execution_config.use_device_gradient is None:
updated_values["use_device_gradient"] = True
if execution_config.grad_on_execution is None:
updated_values["grad_on_execution"] = True
return replace(execution_config, **updated_values)
def _setup_device_config(self, execution_config):
tape = qml.tape.QuantumScript([], [])
if not self._validate_device_method(tape):
raise qml.DeviceError("device does not support device derivatives")
updated_values = {}
if execution_config.use_device_gradient is None:
updated_values["use_device_gradient"] = True
if execution_config.grad_on_execution is None:
updated_values["grad_on_execution"] = True
return replace(execution_config, **updated_values)
# pylint: disable=too-many-return-statements
def _setup_execution_config(self, execution_config):
if execution_config.gradient_method == "best":
tape = qml.tape.QuantumScript([], [])
if self._validate_device_method(tape):
config = replace(execution_config, gradient_method="device")
return self._setup_execution_config(config)
if self._validate_backprop_method(tape):
config = replace(execution_config, gradient_method="backprop")
return self._setup_backprop_config(config)
if execution_config.gradient_method == "backprop":
return self._setup_backprop_config(execution_config)
if _requests_adjoint(execution_config):
return self._setup_adjoint_config(execution_config)
if execution_config.gradient_method == "device":
return self._setup_device_config(execution_config)
return execution_config
[docs] def supports_derivatives(self, execution_config=None, circuit=None) -> bool:
circuit = qml.tape.QuantumScript([], [], shots=self.shots) if circuit is None else circuit
if execution_config is None or execution_config.gradient_method == "best":
validation_methods = (
self._validate_backprop_method,
self._validate_device_method,
)
return any(validate(circuit) for validate in validation_methods)
if execution_config.gradient_method == "backprop":
return self._validate_backprop_method(circuit)
if _requests_adjoint(execution_config):
return self._validate_adjoint_method(circuit)
if execution_config.gradient_method == "device":
return self._validate_device_method(circuit)
return False
def _validate_backprop_method(self, tape):
if tape.shots:
return False
params = tape.get_parameters(trainable_only=False)
interface = qml.math.get_interface(*params)
if interface != "numpy":
interface = get_canonical_interface_name(interface).value
if tape and any(isinstance(m.obs, qml.SparseHamiltonian) for m in tape.measurements):
return False
# determine if the device supports backpropagation
backprop_interface = self._device.capabilities().get("passthru_interface", None)
if backprop_interface is not None:
# device supports backpropagation natively
return interface in [backprop_interface, "numpy"]
# determine if the device has any child devices that support backpropagation
backprop_devices = self._device.capabilities().get("passthru_devices", None)
if backprop_devices is None:
return False
return interface in backprop_devices or interface == "numpy"
def _validate_adjoint_method(self, tape):
# The conditions below provide a minimal set of requirements that we can likely improve upon in
# future, or alternatively summarize within a single device capability. Moreover, we also
# need to inspect the circuit measurements to ensure only expectation values are taken. This
# cannot be done here since we don't yet know the composition of the circuit.
required_attrs = ["_apply_operation", "_apply_unitary", "adjoint_jacobian"]
supported_device = all(hasattr(self._device, attr) for attr in required_attrs)
supported_device = supported_device and self._device.capabilities().get("returns_state")
if not supported_device or bool(tape.shots):
return False
program = TransformProgram()
_add_adjoint_transforms(program, name=f"{self.name} + adjoint")
try:
program((tape,))
except (qml.operation.DecompositionUndefinedError, qml.DeviceError, AttributeError):
return False
return True
def _validate_device_method(self, _):
# determine if the device provides its own jacobian method
return self._device.capabilities().get("provides_jacobian", False)
[docs] def execute(self, circuits, execution_config=DefaultExecutionConfig):
dev = self.target_device
kwargs = {}
if dev.capabilities().get("supports_mid_measure", False):
kwargs["postselect_mode"] = execution_config.mcm_config.postselect_mode
first_shot = circuits[0].shots
if all(t.shots == first_shot for t in circuits):
return _set_shots(dev, first_shot)(dev.batch_execute)(circuits, **kwargs)
return tuple(
_set_shots(dev, t.shots)(dev.batch_execute)((t,), **kwargs)[0] for t in circuits
)
[docs] def execute_and_compute_derivatives(self, circuits, execution_config=DefaultExecutionConfig):
first_shot = circuits[0].shots
if all(t.shots == first_shot for t in circuits):
return _set_shots(self._device, first_shot)(self._device.execute_and_gradients)(
circuits, **execution_config.gradient_keyword_arguments
)
batched_res = tuple(
self.execute_and_compute_derivatives((c,), execution_config) for c in circuits
)
return tuple(zip(*batched_res))
[docs] def compute_derivatives(self, circuits, execution_config=DefaultExecutionConfig):
first_shot = circuits[0].shots
if all(t.shots == first_shot for t in circuits):
return _set_shots(self._device, first_shot)(self._device.gradients)(
circuits, **execution_config.gradient_keyword_arguments
)
return tuple(self.compute_derivatives((c,), execution_config) for c in circuits)
_modules/pennylane/devices/legacy_facade
Download Python script
Download Notebook
View on GitHub