From ac7aca32f0a7fa2e74fd7576fdc9ef00a267f717 Mon Sep 17 00:00:00 2001 From: Akash M <97802146+Ak-ash22@users.noreply.github.com> Date: Thu, 7 Dec 2023 11:25:31 +0100 Subject: [PATCH] Remove deprecated ParametricPulse classes (#11024) * This is a demo * commiting new file * resolved conflict * removed ParametricPulse instance from files * style changes * Delete demo.txt * Update releasenotes/notes/releasenotes/notes/remove-deprecated-parametric-pulses-class-667e4b970e1163b3.yaml Co-authored-by: Luciano Bello * Update releasenotes/notes/releasenotes/notes/remove-deprecated-parametric-pulses-class-667e4b970e1163b3.yaml Co-authored-by: Luciano Bello * Update releasenotes/notes/releasenotes/notes/remove-deprecated-parametric-pulses-class-667e4b970e1163b3.yaml Co-authored-by: Luciano Bello --------- Co-authored-by: Luciano Bello Co-authored-by: Naoki Kanazawa --- qiskit/assembler/assemble_schedules.py | 2 +- qiskit/pulse/__init__.py | 1 - qiskit/pulse/library/__init__.py | 2 - qiskit/pulse/library/parametric_pulses.py | 629 ------------------ qiskit/pulse/library/symbolic_pulses.py | 3 +- qiskit/pulse/parameter_manager.py | 21 +- qiskit/qobj/converters/pulse_instruction.py | 6 +- qiskit/visualization/pulse_v2/core.py | 10 +- .../pulse_v2/generators/waveform.py | 4 +- qiskit/visualization/pulse_v2/interface.py | 9 +- qiskit/visualization/pulse_v2/layouts.py | 10 +- ...ametric-pulses-class-667e4b970e1163b3.yaml | 6 + test/python/pulse/test_pulse_lib.py | 10 +- 13 files changed, 29 insertions(+), 684 deletions(-) delete mode 100644 qiskit/pulse/library/parametric_pulses.py create mode 100644 releasenotes/notes/releasenotes/notes/remove-deprecated-parametric-pulses-class-667e4b970e1163b3.yaml diff --git a/qiskit/assembler/assemble_schedules.py b/qiskit/assembler/assemble_schedules.py index 6ddf4c140d56..c60c28ff9a50 100644 --- a/qiskit/assembler/assemble_schedules.py +++ b/qiskit/assembler/assemble_schedules.py @@ -224,7 +224,7 @@ def _assemble_instructions( for time, instruction in sched.instructions: if isinstance(instruction, instructions.Play): - if isinstance(instruction.pulse, (library.ParametricPulse, library.SymbolicPulse)): + if isinstance(instruction.pulse, library.SymbolicPulse): is_backend_supported = True try: pulse_shape = ParametricPulseShapes.from_instance(instruction.pulse).name diff --git a/qiskit/pulse/__init__.py b/qiskit/pulse/__init__.py index 34ef5c072020..685c1b14acbf 100644 --- a/qiskit/pulse/__init__.py +++ b/qiskit/pulse/__init__.py @@ -161,7 +161,6 @@ GaussianDeriv, Sech, SechDeriv, - ParametricPulse, SymbolicPulse, ScalableSymbolicPulse, Waveform, diff --git a/qiskit/pulse/library/__init__.py b/qiskit/pulse/library/__init__.py index 89778b240137..99d382e63f91 100644 --- a/qiskit/pulse/library/__init__.py +++ b/qiskit/pulse/library/__init__.py @@ -48,7 +48,6 @@ Waveform SymbolicPulse - ParametricPulse .. _symbolic_pulses: @@ -76,7 +75,6 @@ """ -from .parametric_pulses import ParametricPulse from .symbolic_pulses import ( SymbolicPulse, ScalableSymbolicPulse, diff --git a/qiskit/pulse/library/parametric_pulses.py b/qiskit/pulse/library/parametric_pulses.py deleted file mode 100644 index 03876b73ebfd..000000000000 --- a/qiskit/pulse/library/parametric_pulses.py +++ /dev/null @@ -1,629 +0,0 @@ -# This code is part of Qiskit. -# -# (C) Copyright IBM 2020. -# -# This code is licensed under the Apache License, Version 2.0. You may -# obtain a copy of this license in the LICENSE.txt file in the root directory -# of this source tree or at http://www.apache.org/licenses/LICENSE-2.0. -# -# Any modifications or derivative works of this code must retain this -# copyright notice, and modified files need to carry a notice indicating -# that they have been altered from the originals. - -"""Parametric waveforms module. These are pulses which are described by a specified -parameterization. - -If a backend supports parametric pulses, it will have the attribute -`backend.configuration().parametric_pulses`, which is a list of supported pulse shapes, such as -`['gaussian', 'gaussian_square', 'drag']`. A Pulse Schedule, using parametric pulses, which is -assembled for a backend which supports those pulses, will result in a Qobj which is dramatically -smaller than one which uses Waveforms. - -This module can easily be extended to describe more pulse shapes. The new class should: - - have a descriptive name - - be a well known and/or well described formula (include the formula in the class docstring) - - take some parameters (at least `duration`) and validate them, if necessary - - implement a ``get_waveform`` method which returns a corresponding Waveform in the case that - it is assembled for a backend which does not support it. Ends are zeroed to avoid steep jumps at - pulse edges. By default, the ends are defined such that ``f(-1), f(duration+1) = 0``. - -The new pulse must then be registered by the assembler in -`qiskit/qobj/converters/pulse_instruction.py:ParametricPulseShapes` -by following the existing pattern: - - class ParametricPulseShapes(Enum): - gaussian = library.Gaussian - ... - new_supported_pulse_name = library.YourPulseWaveformClass -""" -from abc import abstractmethod -from typing import Any, Dict, Optional, Union - -import math -import numpy as np - -from qiskit.circuit.parameterexpression import ParameterExpression -from qiskit.pulse.exceptions import PulseError -from qiskit.pulse.library import continuous -from qiskit.pulse.library.symbolic_pulses import Gaussian as SymGaussian -from qiskit.pulse.library.symbolic_pulses import GaussianSquare as SymGaussianSquare -from qiskit.pulse.library.symbolic_pulses import Drag as SymDrag -from qiskit.pulse.library.symbolic_pulses import Constant as SymConstant -from qiskit.pulse.library.pulse import Pulse -from qiskit.pulse.library.waveform import Waveform -from qiskit.utils.deprecation import deprecate_func - - -class ParametricPulse(Pulse): - """The abstract superclass for parametric pulses. - - .. warning:: - - This class is superseded by :class:`.SymbolicPulse` and will be deprecated - and eventually removed in the future because of the poor flexibility - for defining a new waveform type and serializing it through the :mod:`qiskit.qpy` framework. - - """ - - @abstractmethod - @deprecate_func( - additional_msg=( - "Instead, use SymbolicPulse because of QPY serialization support. See " - "qiskit.pulse.library.symbolic_pulses for details." - ), - since="0.22", - package_name="qiskit-terra", - pending=True, - ) - def __init__( - self, - duration: Union[int, ParameterExpression], - name: Optional[str] = None, - limit_amplitude: Optional[bool] = None, - ): - """Create a parametric pulse and validate the input parameters. - - Args: - duration: Pulse length in terms of the sampling period `dt`. - name: Display name for this pulse envelope. - limit_amplitude: If ``True``, then limit the amplitude of the - waveform to 1. The default is ``True`` and the - amplitude is constrained to 1. - """ - super().__init__(duration=duration, name=name, limit_amplitude=limit_amplitude) - self.validate_parameters() - - @abstractmethod - def get_waveform(self) -> Waveform: - """Return a Waveform with samples filled according to the formula that the pulse - represents and the parameter values it contains. - """ - raise NotImplementedError - - @abstractmethod - def validate_parameters(self) -> None: - """ - Validate parameters. - - Raises: - PulseError: If the parameters passed are not valid. - """ - raise NotImplementedError - - def is_parameterized(self) -> bool: - """Return True iff the instruction is parameterized.""" - return any(_is_parameterized(val) for val in self.parameters.values()) - - def __eq__(self, other: Pulse) -> bool: - return super().__eq__(other) and self.parameters == other.parameters - - def __hash__(self) -> int: - return hash(tuple(self.parameters[k] for k in sorted(self.parameters))) - - -class Gaussian(ParametricPulse): - r"""A lifted and truncated pulse envelope shaped according to the Gaussian function whose - mean is centered at the center of the pulse (duration / 2): - - .. math:: - - f'(x) &= \exp\Bigl( -\frac12 \frac{{(x - \text{duration}/2)}^2}{\text{sigma}^2} \Bigr)\\ - f(x) &= \text{amp} \times \frac{f'(x) - f'(-1)}{1-f'(-1)}, \quad 0 \le x < \text{duration} - - where :math:`f'(x)` is the gaussian waveform without lifting or amplitude scaling. - - This pulse would be more accurately named as ``LiftedGaussian``, however, for historical - and practical DSP reasons it has the name ``Gaussian``. - """ - - @deprecate_func( - additional_msg=( - "Instead, use Gaussian from qiskit.pulse.library.symbolic_pulses because of " - "QPY serialization support." - ), - since="0.22", - package_name="qiskit-terra", - pending=True, - ) - def __init__( - self, - duration: Union[int, ParameterExpression], - amp: Union[complex, ParameterExpression], - sigma: Union[float, ParameterExpression], - name: Optional[str] = None, - limit_amplitude: Optional[bool] = None, - ): - """Initialize the gaussian pulse. - - Args: - duration: Pulse length in terms of the sampling period `dt`. - amp: The amplitude of the Gaussian envelope. - sigma: A measure of how wide or narrow the Gaussian peak is; described mathematically - in the class docstring. - name: Display name for this pulse envelope. - limit_amplitude: If ``True``, then limit the amplitude of the - waveform to 1. The default is ``True`` and the - amplitude is constrained to 1. - """ - if not _is_parameterized(amp): - amp = complex(amp) - self._amp = amp - self._sigma = sigma - super().__init__(duration=duration, name=name, limit_amplitude=limit_amplitude) - - @property - def amp(self) -> Union[complex, ParameterExpression]: - """The Gaussian amplitude.""" - return self._amp - - @property - def sigma(self) -> Union[float, ParameterExpression]: - """The Gaussian standard deviation of the pulse width.""" - return self._sigma - - def get_waveform(self) -> Waveform: - return SymGaussian( - duration=self.duration, - amp=np.abs(self.amp), - angle=np.angle(self.amp), - sigma=self.sigma, - zero_ends=True, - ).get_waveform() - - def validate_parameters(self) -> None: - if not _is_parameterized(self.amp) and abs(self.amp) > 1.0 and self._limit_amplitude: - raise PulseError( - f"The amplitude norm must be <= 1, found: {abs(self.amp)}" - + "This can be overruled by setting Pulse.limit_amplitude." - ) - if not _is_parameterized(self.sigma) and self.sigma <= 0: - raise PulseError("Sigma must be greater than 0.") - - @property - def parameters(self) -> Dict[str, Any]: - return {"duration": self.duration, "amp": self.amp, "sigma": self.sigma} - - def __repr__(self) -> str: - return "{}(duration={}, amp={}, sigma={}{})".format( - self.__class__.__name__, - self.duration, - self.amp, - self.sigma, - f", name='{self.name}'" if self.name is not None else "", - ) - - -class GaussianSquare(ParametricPulse): - # Not a raw string because we need to be able to split lines. - """A square pulse with a Gaussian shaped risefall on both sides lifted such that - its first sample is zero. - - Either the ``risefall_sigma_ratio`` or ``width`` parameter has to be specified. - - If ``risefall_sigma_ratio`` is not None and ``width`` is None: - - .. math:: - - \\text{risefall} &= \\text{risefall_sigma_ratio} \\times \\text{sigma}\\\\ - \\text{width} &= \\text{duration} - 2 \\times \\text{risefall} - - If ``width`` is not None and ``risefall_sigma_ratio`` is None: - - .. math:: \\text{risefall} = \\frac{\\text{duration} - \\text{width}}{2} - - In both cases, the lifted gaussian square pulse :math:`f'(x)` is defined as: - - .. math:: - - f'(x) &= \\begin{cases}\ - \\exp\\biggl(-\\frac12 \\frac{(x - \\text{risefall})^2}{\\text{sigma}^2}\\biggr)\ - & x < \\text{risefall}\\\\ - 1\ - & \\text{risefall} \\le x < \\text{risefall} + \\text{width}\\\\ - \\exp\\biggl(-\\frac12\ - \\frac{{\\bigl(x - (\\text{risefall} + \\text{width})\\bigr)}^2}\ - {\\text{sigma}^2}\ - \\biggr)\ - & \\text{risefall} + \\text{width} \\le x\ - \\end{cases}\\\\ - f(x) &= \\text{amp} \\times \\frac{f'(x) - f'(-1)}{1-f'(-1)},\ - \\quad 0 \\le x < \\text{duration} - - where :math:`f'(x)` is the gaussian square waveform without lifting or amplitude scaling. - - This pulse would be more accurately named as ``LiftedGaussianSquare``, however, for historical - and practical DSP reasons it has the name ``GaussianSquare``. - """ - - @deprecate_func( - additional_msg=( - "Instead, use GaussianSquare from qiskit.pulse.library.symbolic_pulses because of " - "QPY serialization support." - ), - since="0.22", - package_name="qiskit-terra", - pending=True, - ) - def __init__( - self, - duration: Union[int, ParameterExpression], - amp: Union[complex, ParameterExpression], - sigma: Union[float, ParameterExpression], - width: Union[float, ParameterExpression] = None, - risefall_sigma_ratio: Union[float, ParameterExpression] = None, - name: Optional[str] = None, - limit_amplitude: Optional[bool] = None, - ): - """Initialize the gaussian square pulse. - - Args: - duration: Pulse length in terms of the sampling period `dt`. - amp: The amplitude of the Gaussian and of the square pulse. - sigma: A measure of how wide or narrow the Gaussian risefall is; see the class - docstring for more details. - width: The duration of the embedded square pulse. - risefall_sigma_ratio: The ratio of each risefall duration to sigma. - name: Display name for this pulse envelope. - limit_amplitude: If ``True``, then limit the amplitude of the - waveform to 1. The default is ``True`` and the - amplitude is constrained to 1. - - Raises: - PulseError: If the parameters passed are not valid. - """ - if not _is_parameterized(amp): - amp = complex(amp) - self._amp = amp - self._sigma = sigma - self._risefall_sigma_ratio = risefall_sigma_ratio - self._width = width - - if self.width is not None and self.risefall_sigma_ratio is not None: - raise PulseError( - "Either the pulse width or the risefall_sigma_ratio parameter can be specified" - " but not both." - ) - - super().__init__(duration=duration, name=name, limit_amplitude=limit_amplitude) - - @property - def amp(self) -> Union[complex, ParameterExpression]: - """The Gaussian amplitude.""" - return self._amp - - @property - def sigma(self) -> Union[float, ParameterExpression]: - """The Gaussian standard deviation of the pulse width.""" - return self._sigma - - @property - def risefall_sigma_ratio(self) -> Union[float, ParameterExpression]: - """The duration of each risefall in terms of sigma.""" - return self._risefall_sigma_ratio - - @property - def width(self) -> Union[float, ParameterExpression]: - """The width of the square portion of the pulse.""" - return self._width - - def get_waveform(self) -> Waveform: - return SymGaussianSquare( - duration=self.duration, - amp=np.abs(self.amp), - angle=np.angle(self.amp), - width=self.width, - sigma=self.sigma, - zero_ends=True, - ).get_waveform() - - def validate_parameters(self) -> None: - if not _is_parameterized(self.amp) and abs(self.amp) > 1.0 and self._limit_amplitude: - raise PulseError( - f"The amplitude norm must be <= 1, found: {abs(self.amp)}" - + "This can be overruled by setting Pulse.limit_amplitude." - ) - if not _is_parameterized(self.sigma) and self.sigma <= 0: - raise PulseError("Sigma must be greater than 0.") - if self.width is None and self.risefall_sigma_ratio is None: - raise PulseError( - "Either the pulse width or the risefall_sigma_ratio parameter must be specified." - ) - if self.width is not None: - if not _is_parameterized(self.width) and self.width < 0: - raise PulseError("The pulse width must be at least 0.") - if ( - not (_is_parameterized(self.width) or _is_parameterized(self.duration)) - and self.width >= self.duration - ): - raise PulseError("The pulse width must be less than its duration.") - self._risefall_sigma_ratio = (self.duration - self.width) / (2.0 * self.sigma) - - else: - if not _is_parameterized(self.risefall_sigma_ratio) and self.risefall_sigma_ratio <= 0: - raise PulseError("The parameter risefall_sigma_ratio must be greater than 0.") - if not ( - _is_parameterized(self.risefall_sigma_ratio) - or _is_parameterized(self.duration) - or _is_parameterized(self.sigma) - ) and self.risefall_sigma_ratio >= self.duration / (2.0 * self.sigma): - raise PulseError( - "The parameter risefall_sigma_ratio must be less than duration/(" - "2*sigma)={}.".format(self.duration / (2.0 * self.sigma)) - ) - self._width = self.duration - 2.0 * self.risefall_sigma_ratio * self.sigma - - @property - def parameters(self) -> Dict[str, Any]: - return { - "duration": self.duration, - "amp": self.amp, - "sigma": self.sigma, - "width": self.width, - } - - def __repr__(self) -> str: - return "{}(duration={}, amp={}, sigma={}, width={}{})".format( - self.__class__.__name__, - self.duration, - self.amp, - self.sigma, - self.width, - f", name='{self.name}'" if self.name is not None else "", - ) - - -class Drag(ParametricPulse): - """The Derivative Removal by Adiabatic Gate (DRAG) pulse is a standard Gaussian pulse - with an additional Gaussian derivative component and lifting applied. - - It is designed to reduce the frequency spectrum of a standard Gaussian pulse near - the :math:`|1\\rangle\\leftrightarrow|2\\rangle` transition, - reducing the chance of leakage to the :math:`|2\\rangle` state. - - .. math:: - - g(x) &= \\exp\\Bigl(-\\frac12 \\frac{(x - \\text{duration}/2)^2}{\\text{sigma}^2}\\Bigr)\\\\ - g'(x) &= \\text{amp}\\times\\frac{g(x)-g(-1)}{1-g(-1)}\\\\ - f(x) &= g'(x) \\times \\Bigl(1 + 1j \\times \\text{beta} \\times\ - \\Bigl(-\\frac{x - \\text{duration}/2}{\\text{sigma}^2}\\Bigr) \\Bigr), - \\quad 0 \\le x < \\text{duration} - - where :math:`g(x)` is a standard unlifted Gaussian waveform and - :math:`g'(x)` is the lifted :class:`~qiskit.pulse.library.Gaussian` waveform. - - This pulse, defined by :math:`f(x)`, would be more accurately named as ``LiftedDrag``, however, - for historical and practical DSP reasons it has the name ``Drag``. - - References: - 1. |citation1|_ - - .. _citation1: https://link.aps.org/doi/10.1103/PhysRevA.83.012308 - - .. |citation1| replace:: *Gambetta, J. M., Motzoi, F., Merkel, S. T. & Wilhelm, F. K. - Analytic control methods for high-fidelity unitary operations - in a weakly nonlinear oscillator. Phys. Rev. A 83, 012308 (2011).* - - 2. |citation2|_ - - .. _citation2: https://link.aps.org/doi/10.1103/PhysRevLett.103.110501 - - .. |citation2| replace:: *F. Motzoi, J. M. Gambetta, P. Rebentrost, and F. K. Wilhelm - Phys. Rev. Lett. 103, 110501 – Published 8 September 2009.* - """ - - @deprecate_func( - additional_msg=( - "Instead, use Drag from qiskit.pulse.library.symbolic_pulses because of " - "QPY serialization support." - ), - since="0.22", - package_name="qiskit-terra", - pending=True, - ) - def __init__( - self, - duration: Union[int, ParameterExpression], - amp: Union[complex, ParameterExpression], - sigma: Union[float, ParameterExpression], - beta: Union[float, ParameterExpression], - name: Optional[str] = None, - limit_amplitude: Optional[bool] = None, - ): - """Initialize the drag pulse. - - Args: - duration: Pulse length in terms of the sampling period `dt`. - amp: The amplitude of the Drag envelope. - sigma: A measure of how wide or narrow the Gaussian peak is; described mathematically - in the class docstring. - beta: The correction amplitude. - name: Display name for this pulse envelope. - limit_amplitude: If ``True``, then limit the amplitude of the - waveform to 1. The default is ``True`` and the - amplitude is constrained to 1. - """ - if not _is_parameterized(amp): - amp = complex(amp) - self._amp = amp - self._sigma = sigma - self._beta = beta - super().__init__(duration=duration, name=name, limit_amplitude=limit_amplitude) - - @property - def amp(self) -> Union[complex, ParameterExpression]: - """The Gaussian amplitude.""" - return self._amp - - @property - def sigma(self) -> Union[float, ParameterExpression]: - """The Gaussian standard deviation of the pulse width.""" - return self._sigma - - @property - def beta(self) -> Union[float, ParameterExpression]: - """The weighing factor for the Gaussian derivative component of the waveform.""" - return self._beta - - def get_waveform(self) -> Waveform: - return SymDrag( - duration=self.duration, - amp=np.abs(self.amp), - angle=np.angle(self.amp), - sigma=self.sigma, - beta=self.beta, - zero_ends=True, - ).get_waveform() - - def validate_parameters(self) -> None: - if not _is_parameterized(self.amp) and abs(self.amp) > 1.0 and self._limit_amplitude: - raise PulseError( - f"The amplitude norm must be <= 1, found: {abs(self.amp)}" - + "This can be overruled by setting Pulse.limit_amplitude." - ) - if not _is_parameterized(self.sigma) and self.sigma <= 0: - raise PulseError("Sigma must be greater than 0.") - if not _is_parameterized(self.beta) and isinstance(self.beta, complex): - raise PulseError("Beta must be real.") - # Check if beta is too large: the amplitude norm must be <=1 for all points - if ( - not _is_parameterized(self.beta) - and not _is_parameterized(self.sigma) - and np.abs(self.beta) > self.sigma - and self._limit_amplitude - ): - # If beta <= sigma, then the maximum amplitude is at duration / 2, which is - # already constrained by self.amp <= 1 - - # 1. Find the first maxima associated with the beta * d/dx gaussian term - # This eq is derived from solving for the roots of the norm of the drag function. - # There is a second maxima mirrored around the center of the pulse with the same - # norm as the first, so checking the value at the first x maxima is sufficient. - argmax_x = self.duration / 2 - (self.sigma / self.beta) * math.sqrt( - self.beta**2 - self.sigma**2 - ) - # If the max point is out of range, either end of the pulse will do - argmax_x = max(argmax_x, 0) - - # 2. Find the value at that maximum - max_val = continuous.drag( - np.array(argmax_x), - sigma=self.sigma, - beta=self.beta, - amp=self.amp, - center=self.duration / 2, - ) - if abs(max_val) > 1.0: - raise PulseError("Beta is too large; pulse amplitude norm exceeds 1.") - - @property - def parameters(self) -> Dict[str, Any]: - return {"duration": self.duration, "amp": self.amp, "sigma": self.sigma, "beta": self.beta} - - def __repr__(self) -> str: - return "{}(duration={}, amp={}, sigma={}, beta={}{})".format( - self.__class__.__name__, - self.duration, - self.amp, - self.sigma, - self.beta, - f", name='{self.name}'" if self.name is not None else "", - ) - - -class Constant(ParametricPulse): - """ - A simple constant pulse, with an amplitude value and a duration: - - .. math:: - - f(x) = amp , 0 <= x < duration - f(x) = 0 , elsewhere - """ - - @deprecate_func( - additional_msg=( - "Instead, use Constant from qiskit.pulse.library.symbolic_pulses because of " - "QPY serialization support." - ), - since="0.22", - package_name="qiskit-terra", - pending=True, - ) - def __init__( - self, - duration: Union[int, ParameterExpression], - amp: Union[complex, ParameterExpression], - name: Optional[str] = None, - limit_amplitude: Optional[bool] = None, - ): - """ - Initialize the constant-valued pulse. - - Args: - duration: Pulse length in terms of the sampling period `dt`. - amp: The amplitude of the constant square pulse. - name: Display name for this pulse envelope. - limit_amplitude: If ``True``, then limit the amplitude of the - waveform to 1. The default is ``True`` and the - amplitude is constrained to 1. - """ - if not _is_parameterized(amp): - amp = complex(amp) - self._amp = amp - super().__init__(duration=duration, name=name, limit_amplitude=limit_amplitude) - - @property - def amp(self) -> Union[complex, ParameterExpression]: - """The constant value amplitude.""" - return self._amp - - def get_waveform(self) -> Waveform: - return SymConstant( - duration=self.duration, amp=np.abs(self.amp), angle=np.angle(self.amp) - ).get_waveform() - - def validate_parameters(self) -> None: - if not _is_parameterized(self.amp) and abs(self.amp) > 1.0 and self._limit_amplitude: - raise PulseError( - f"The amplitude norm must be <= 1, found: {abs(self.amp)}" - + "This can be overruled by setting Pulse.limit_amplitude." - ) - - @property - def parameters(self) -> Dict[str, Any]: - return {"duration": self.duration, "amp": self.amp} - - def __repr__(self) -> str: - return "{}(duration={}, amp={}{})".format( - self.__class__.__name__, - self.duration, - self.amp, - f", name='{self.name}'" if self.name is not None else "", - ) - - -def _is_parameterized(value: Any) -> bool: - """Shorthand for a frequently checked predicate. ParameterExpressions cannot be - validated until they are numerically assigned. - """ - return isinstance(value, ParameterExpression) diff --git a/qiskit/pulse/library/symbolic_pulses.py b/qiskit/pulse/library/symbolic_pulses.py index a943bb1acf60..4621aa2a4f38 100644 --- a/qiskit/pulse/library/symbolic_pulses.py +++ b/qiskit/pulse/library/symbolic_pulses.py @@ -1465,8 +1465,7 @@ def __new__( # Note this is implemented using Piecewise instead of just returning amp # directly because otherwise the expression has no t dependence and sympy's # lambdify will produce a function f that for an array t returns amp - # instead of amp * np.ones(t.shape). This does not work well with - # ParametricPulse.get_waveform(). + # instead of amp * np.ones(t.shape). # # See: https://github.com/sympy/sympy/issues/5642 envelope_expr = ( diff --git a/qiskit/pulse/parameter_manager.py b/qiskit/pulse/parameter_manager.py index 4fac995b22f7..721fd9501a60 100644 --- a/qiskit/pulse/parameter_manager.py +++ b/qiskit/pulse/parameter_manager.py @@ -57,7 +57,7 @@ from qiskit.circuit.parameterexpression import ParameterExpression, ParameterValueType from qiskit.pulse import instructions, channels from qiskit.pulse.exceptions import PulseError -from qiskit.pulse.library import ParametricPulse, SymbolicPulse, Waveform +from qiskit.pulse.library import SymbolicPulse, Waveform from qiskit.pulse.schedule import Schedule, ScheduleBlock from qiskit.pulse.transforms.alignments import AlignmentKind from qiskit.pulse.utils import format_parameter_value @@ -207,19 +207,6 @@ def visit_Channel(self, node: channels.Channel): return node - def visit_ParametricPulse(self, node: ParametricPulse): - """Assign parameters to ``ParametricPulse`` object.""" - if node.is_parameterized(): - new_parameters = {} - for op, op_value in node.parameters.items(): - if isinstance(op_value, ParameterExpression): - op_value = self._assign_parameter_expression(op_value) - new_parameters[op] = op_value - - return node.__class__(**new_parameters, name=node.name) - - return node - def visit_SymbolicPulse(self, node: SymbolicPulse): """Assign parameters to ``SymbolicPulse`` object.""" if node.is_parameterized(): @@ -334,12 +321,6 @@ def visit_Channel(self, node: channels.Channel): """Get parameters from ``Channel`` object.""" self.parameters |= node.parameters - def visit_ParametricPulse(self, node: ParametricPulse): - """Get parameters from ``ParametricPulse`` object.""" - for op_value in node.parameters.values(): - if isinstance(op_value, ParameterExpression): - self.parameters |= op_value.parameters - def visit_SymbolicPulse(self, node: SymbolicPulse): """Get parameters from ``SymbolicPulse`` object.""" for op_value in node.parameters.values(): diff --git a/qiskit/qobj/converters/pulse_instruction.py b/qiskit/qobj/converters/pulse_instruction.py index 091d3aabc417..e99d497dd314 100644 --- a/qiskit/qobj/converters/pulse_instruction.py +++ b/qiskit/qobj/converters/pulse_instruction.py @@ -50,12 +50,12 @@ class ParametricPulseShapes(Enum): @classmethod def from_instance( cls, - instance: Union[library.ParametricPulse, library.SymbolicPulse], + instance: library.SymbolicPulse, ) -> "ParametricPulseShapes": """Get Qobj name from the pulse class instance. Args: - instance: Symbolic or ParametricPulse class. + instance: SymbolicPulse class. Returns: Qobj name. @@ -354,7 +354,7 @@ def _convert_play( Returns: Qobj instruction data. """ - if isinstance(instruction.pulse, (library.ParametricPulse, library.SymbolicPulse)): + if isinstance(instruction.pulse, library.SymbolicPulse): params = dict(instruction.pulse.parameters) # IBM backends expect "amp" to be the complex amplitude if "amp" in params and "angle" in params: diff --git a/qiskit/visualization/pulse_v2/core.py b/qiskit/visualization/pulse_v2/core.py index 42cfa224c5d4..d60f2db030d0 100644 --- a/qiskit/visualization/pulse_v2/core.py +++ b/qiskit/visualization/pulse_v2/core.py @@ -205,11 +205,7 @@ def time_breaks(self, new_breaks: list[tuple[int, int]]): def load_program( self, - program: pulse.Waveform - | pulse.ParametricPulse - | pulse.SymbolicPulse - | pulse.Schedule - | pulse.ScheduleBlock, + program: pulse.Waveform | pulse.SymbolicPulse | pulse.Schedule | pulse.ScheduleBlock, ): """Load a program to draw. @@ -221,7 +217,7 @@ def load_program( """ if isinstance(program, (pulse.Schedule, pulse.ScheduleBlock)): self._schedule_loader(program) - elif isinstance(program, (pulse.Waveform, pulse.ParametricPulse, pulse.SymbolicPulse)): + elif isinstance(program, (pulse.Waveform, pulse.SymbolicPulse)): self._waveform_loader(program) else: raise VisualizationError("Data type %s is not supported." % type(program)) @@ -234,7 +230,7 @@ def load_program( def _waveform_loader( self, - program: pulse.Waveform | pulse.ParametricPulse | pulse.SymbolicPulse, + program: pulse.Waveform | pulse.SymbolicPulse, ): """Load Waveform instance. diff --git a/qiskit/visualization/pulse_v2/generators/waveform.py b/qiskit/visualization/pulse_v2/generators/waveform.py index 7d14b522957f..b0d90b895c7a 100644 --- a/qiskit/visualization/pulse_v2/generators/waveform.py +++ b/qiskit/visualization/pulse_v2/generators/waveform.py @@ -271,7 +271,7 @@ def gen_waveform_max_value( if isinstance(data.inst, instructions.Play): # pulse operand = data.inst.pulse - if isinstance(operand, (pulse.ParametricPulse, pulse.SymbolicPulse)): + if isinstance(operand, pulse.SymbolicPulse): pulse_data = operand.get_waveform() else: pulse_data = operand @@ -563,7 +563,7 @@ def _parse_waveform( if isinstance(inst, instructions.Play): # pulse operand = inst.pulse - if isinstance(operand, (pulse.ParametricPulse, pulse.SymbolicPulse)): + if isinstance(operand, pulse.SymbolicPulse): # parametric pulse params = operand.parameters duration = params.pop("duration", None) diff --git a/qiskit/visualization/pulse_v2/interface.py b/qiskit/visualization/pulse_v2/interface.py index 70b3922d8501..d11a02863d2e 100644 --- a/qiskit/visualization/pulse_v2/interface.py +++ b/qiskit/visualization/pulse_v2/interface.py @@ -22,7 +22,7 @@ from typing import Union, Optional, Dict, Any, Tuple, List from qiskit.providers import Backend -from qiskit.pulse import Waveform, ParametricPulse, SymbolicPulse, Schedule, ScheduleBlock +from qiskit.pulse import Waveform, SymbolicPulse, Schedule, ScheduleBlock from qiskit.pulse.channels import Channel from qiskit.visualization.exceptions import VisualizationError from qiskit.visualization.pulse_v2 import core, device_info, stylesheet, types @@ -30,7 +30,7 @@ def draw( - program: Union[Waveform, ParametricPulse, SymbolicPulse, Schedule, ScheduleBlock], + program: Union[Waveform, SymbolicPulse, Schedule, ScheduleBlock], style: Optional[Dict[str, Any]] = None, backend: Optional[Backend] = None, time_range: Optional[Tuple[int, int]] = None, @@ -47,9 +47,8 @@ def draw( Args: program: Program to visualize. This program can be arbitrary Qiskit Pulse program, - such as :py:class:`~qiskit.pulse.Waveform`, :py:class:`~qiskit.pulse.ParametricPulse`, - :py:class:`~qiskit.pulse.SymbolicPulse`, :py:class:`~qiskit.pulse.Schedule` - and :py:class:`~qiskit.pulse.ScheduleBlock`. + such as :py:class:`~qiskit.pulse.Waveform`, :py:class:`~qiskit.pulse.SymbolicPulse`, + :py:class:`~qiskit.pulse.Schedule` and :py:class:`~qiskit.pulse.ScheduleBlock`. style: Stylesheet options. This can be dictionary or preset stylesheet classes. See :py:class:`~qiskit.visualization.pulse_v2.stylesheets.IQXStandard`, :py:class:`~qiskit.visualization.pulse_v2.stylesheets.IQXSimple`, and diff --git a/qiskit/visualization/pulse_v2/layouts.py b/qiskit/visualization/pulse_v2/layouts.py index 9aa368466215..6b39dceaf567 100644 --- a/qiskit/visualization/pulse_v2/layouts.py +++ b/qiskit/visualization/pulse_v2/layouts.py @@ -67,7 +67,7 @@ def my_horizontal_axis(time_window: Tuple[int, int], This data provides input program and backend system configurations. ```python - def my_figure_title(program: Union[pulse.Waveform, pulse.ParametricPulse, pulse.Schedule], + def my_figure_title(program: Union[pulse.Waveform, pulse.Schedule], device: DrawerBackendInfo) -> str: return 'title' @@ -361,9 +361,7 @@ def time_map_in_ns( ) -def detail_title( - program: Union[pulse.Waveform, pulse.ParametricPulse, pulse.Schedule], device: DrawerBackendInfo -) -> str: +def detail_title(program: Union[pulse.Waveform, pulse.Schedule], device: DrawerBackendInfo) -> str: """Layout function for generating figure title. This layout writes program name, program duration, and backend name in the title. @@ -388,8 +386,6 @@ def detail_title( return ", ".join(title_str) -def empty_title( - program: Union[pulse.Waveform, pulse.ParametricPulse, pulse.Schedule], device: DrawerBackendInfo -) -> str: +def empty_title(program: Union[pulse.Waveform, pulse.Schedule], device: DrawerBackendInfo) -> str: """Layout function for generating an empty figure title.""" return "" diff --git a/releasenotes/notes/releasenotes/notes/remove-deprecated-parametric-pulses-class-667e4b970e1163b3.yaml b/releasenotes/notes/releasenotes/notes/remove-deprecated-parametric-pulses-class-667e4b970e1163b3.yaml new file mode 100644 index 000000000000..7ba9d188f55a --- /dev/null +++ b/releasenotes/notes/releasenotes/notes/remove-deprecated-parametric-pulses-class-667e4b970e1163b3.yaml @@ -0,0 +1,6 @@ +--- +upgrade: + - | + The module ``ParametricPulse`` is removed. + All classes in it were deprecated since Qiskit 0.39 (with qiskit-terra 0.22), released on Oct 2022. + Instead, use :class:`.SymbolicPulse` and check its documentation for details. \ No newline at end of file diff --git a/test/python/pulse/test_pulse_lib.py b/test/python/pulse/test_pulse_lib.py index eb082327964a..c0a4a43e7440 100644 --- a/test/python/pulse/test_pulse_lib.py +++ b/test/python/pulse/test_pulse_lib.py @@ -127,11 +127,11 @@ def test_pulse_limits(self): self.fail("Waveform incorrectly failed to approximately unit norm samples.") -class TestParametricPulses(QiskitTestCase): - """Tests for all subclasses of ParametricPulse.""" +class TestSymbolicPulses(QiskitTestCase): + """Tests for all subclasses of SymbolicPulse.""" def test_construction(self): - """Test that parametric pulses can be constructed without error.""" + """Test that symbolic pulses can be constructed without error.""" Gaussian(duration=25, sigma=4, amp=0.5, angle=np.pi / 2) GaussianSquare(duration=150, amp=0.2, sigma=8, width=140) GaussianSquare(duration=150, amp=0.2, sigma=8, risefall_sigma_ratio=2.5) @@ -493,7 +493,7 @@ def test_parameters(self): self.assertEqual(set(const.parameters.keys()), {"duration", "amp", "angle"}) def test_repr(self): - """Test the repr methods for parametric pulses.""" + """Test the repr methods for symbolic pulses.""" gaus = Gaussian(duration=25, amp=0.7, sigma=4, angle=0.3) self.assertEqual(repr(gaus), "Gaussian(duration=25, sigma=4, amp=0.7, angle=0.3)") gaus_square = GaussianSquare(duration=20, sigma=30, amp=1.0, width=3) @@ -565,7 +565,7 @@ def test_repr(self): ) def test_param_validation(self): - """Test that parametric pulse parameters are validated when initialized.""" + """Test that symbolic pulse parameters are validated when initialized.""" with self.assertRaises(PulseError): Gaussian(duration=25, sigma=0, amp=0.5, angle=np.pi / 2) with self.assertRaises(PulseError):