Quellcode für qiskit.circuit.classicalfunction.classicalfunction
# 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.
"""ClassicalFunction class"""
import ast
try:
from tweedledum import synthesize_xag, simulate # pylint: disable=no-name-in-module
HAS_TWEEDLEDUM = True
except Exception: # pylint: disable=broad-except
HAS_TWEEDLEDUM = False
from qiskit.circuit import quantumregister
from qiskit.circuit import gate
from qiskit.exceptions import QiskitError
from .utils import tweedledum2qiskit
from .classical_function_visitor import ClassicalFunctionVisitor
[Doku]class ClassicalFunction(gate.Gate):
"""Represent a classical function function and its logic network."""
[Doku] def __init__(self, source, name=None):
"""Creates a ``ClassicalFunction`` from Python source code in ``source``.
The code should be a single function with types.
Args:
source (str): Python code with type hints.
name (str): Optional. Default: "*classicalfunction*". ClassicalFunction name.
Raises:
ImportError: If tweedledum is not installed.
QiskitError: If source is not a string.
"""
if not isinstance(source, str):
raise QiskitError('ClassicalFunction needs a source code as a string.')
if not HAS_TWEEDLEDUM:
raise ImportError("To use the classicalfunction compiler, tweedledum "
"must be installed. To install tweedledum run "
'"pip install tweedledum".')
self._ast = ast.parse(source)
self._network = None
self._scopes = None
self._args = None
super().__init__(name or '*classicalfunction*',
num_qubits=sum([qreg.size for qreg in self.qregs]),
params=[])
[Doku] def compile(self):
"""Parses and creates the logical circuit"""
_classical_function_visitor = ClassicalFunctionVisitor()
_classical_function_visitor.visit(self._ast)
self._network = _classical_function_visitor._network
self._scopes = _classical_function_visitor.scopes
self._args = _classical_function_visitor.args
self.name = _classical_function_visitor.name
@property
def network(self):
"""Returns the logical network"""
if self._network is None:
self.compile()
return self._network
@property
def scopes(self):
"""Returns the scope dict"""
if self._scopes is None:
self.compile()
return self._scopes
@property
def args(self):
"""Returns the classicalfunction arguments"""
if self._args is None:
self.compile()
return self._args
@property
def types(self):
"""Dumps a list of scopes with their variables and types.
Returns:
list(dict): A list of scopes as dicts, where key is the variable name and
value is its type.
"""
ret = []
for scope in self.scopes:
ret.append({k: v[0] for k, v in scope.items()})
return ret
[Doku] def simulate(self):
"""Runs ``tweedledum.simulate`` on the logic network."""
return simulate(self._network)
[Doku] def synth(self, registerless=True):
"""Synthesis the logic network into a :class:`~qiskit.circuit.QuantumCircuit`.
Args:
registerless (bool): Default ``True``. If ``False`` uses the parameter names to create
registers with those names. Otherwise, creates a circuit with a flat quantum register.
Returns:
QuantumCircuit: A circuit implementing the logic network.
"""
if registerless:
qregs = None
else:
qregs = self.qregs
return tweedledum2qiskit(synthesize_xag(self._network), name=self.name, qregs=qregs)
def _define(self):
"""The definition of the classicalfunction is its synthesis"""
self.definition = self.synth()
@property
def qregs(self):
"""The list of qregs used by the classicalfunction"""
qregs = [
quantumregister.QuantumRegister(
1, name=arg) for arg in self.args if self.types[0][arg] == 'Int1']
qregs.reverse()
if self.types[0]['return'] == 'Int1':
qregs.append(quantumregister.QuantumRegister(1, name='return'))
return qregs