Quellcode für qiskit.pulse.library.waveform
# 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.
"""A pulse that is described by complex-valued sample points."""
from typing import Dict, List, Optional, Union, Any
import numpy as np
from qiskit.pulse.exceptions import PulseError
from qiskit.pulse.library.pulse import Pulse
[Doku]class Waveform(Pulse):
"""A pulse specified completely by complex-valued samples; each sample is played for the
duration of the backend cycle-time, dt.
"""
def __init__(
self,
samples: Union[np.ndarray, List[complex]],
name: Optional[str] = None,
epsilon: float = 1e-7,
limit_amplitude: Optional[bool] = None,
):
"""Create new sample pulse command.
Args:
samples: Complex array of the samples in the pulse envelope.
name: Unique name to identify the pulse.
epsilon: Pulse sample norm tolerance for clipping.
If any sample's norm exceeds unity by less than or equal to epsilon
it will be clipped to unit norm. If the sample
norm is greater than 1+epsilon an error will be raised.
limit_amplitude: Passed to parent Pulse
"""
super().__init__(duration=len(samples), name=name, limit_amplitude=limit_amplitude)
samples = np.asarray(samples, dtype=np.complex128)
self.epsilon = epsilon
self._samples = self._clip(samples, epsilon=epsilon)
@property
def samples(self) -> np.ndarray:
"""Return sample values."""
return self._samples
def _clip(self, samples: np.ndarray, epsilon: float = 1e-7) -> np.ndarray:
"""If samples are within epsilon of unit norm, clip sample by reducing norm by (1-epsilon).
If difference is greater than epsilon error is raised.
Args:
samples: Complex array of the samples in the pulse envelope.
epsilon: Pulse sample norm tolerance for clipping.
If any sample's norm exceeds unity by less than or equal to epsilon
it will be clipped to unit norm. If the sample
norm is greater than 1+epsilon an error will be raised.
Returns:
Clipped pulse samples.
Raises:
PulseError: If there exists a pulse sample with a norm greater than 1+epsilon.
"""
samples_norm = np.abs(samples)
to_clip = (samples_norm > 1.0) & (samples_norm <= 1.0 + epsilon)
if np.any(to_clip):
# first try normalizing by the abs value
clip_where = np.argwhere(to_clip)
clip_angle = np.angle(samples[clip_where])
clipped_samples = np.exp(1j * clip_angle, dtype=np.complex128)
# if norm still exceed one subtract epsilon
# required for some platforms
clipped_sample_norms = np.abs(clipped_samples)
to_clip_epsilon = clipped_sample_norms > 1.0
if np.any(to_clip_epsilon):
clip_where_epsilon = np.argwhere(to_clip_epsilon)
clipped_samples_epsilon = (1 - epsilon) * np.exp(
1j * clip_angle[clip_where_epsilon], dtype=np.complex128
)
clipped_samples[clip_where_epsilon] = clipped_samples_epsilon
# update samples with clipped values
samples[clip_where] = clipped_samples
samples_norm[clip_where] = np.abs(clipped_samples)
if np.any(samples_norm > 1.0) and self._limit_amplitude:
amp = np.max(samples_norm)
raise PulseError(
f"Pulse contains sample with norm {amp} greater than 1+epsilon."
" This can be overruled by setting Pulse.limit_amplitude."
)
return samples
[Doku] def is_parameterized(self) -> bool:
"""Return True iff the instruction is parameterized."""
return False
@property
def parameters(self) -> Dict[str, Any]:
"""Return a dictionary containing the pulse's parameters."""
return {}
def __eq__(self, other: Pulse) -> bool:
return (
super().__eq__(other)
and self.samples.shape == other.samples.shape
and np.allclose(self.samples, other.samples, rtol=0, atol=self.epsilon)
)
def __hash__(self) -> int:
return hash(self.samples.tobytes())
def __repr__(self) -> str:
opt = np.get_printoptions()
np.set_printoptions(threshold=50)
np.set_printoptions(**opt)
return "{}({}{})".format(
self.__class__.__name__,
repr(self.samples),
f", name='{self.name}'" if self.name is not None else "",
)