qiskit 1.3.0__cp39-abi3-macosx_11_0_arm64.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- qiskit/VERSION.txt +1 -0
- qiskit/__init__.py +146 -0
- qiskit/_accelerate.abi3.so +0 -0
- qiskit/_numpy_compat.py +73 -0
- qiskit/assembler/__init__.py +42 -0
- qiskit/assembler/assemble_circuits.py +451 -0
- qiskit/assembler/assemble_schedules.py +367 -0
- qiskit/assembler/disassemble.py +310 -0
- qiskit/assembler/run_config.py +77 -0
- qiskit/circuit/__init__.py +1313 -0
- qiskit/circuit/_classical_resource_map.py +148 -0
- qiskit/circuit/_standard_gates_commutations.py +3849 -0
- qiskit/circuit/_utils.py +167 -0
- qiskit/circuit/add_control.py +274 -0
- qiskit/circuit/annotated_operation.py +279 -0
- qiskit/circuit/barrier.py +50 -0
- qiskit/circuit/bit.py +94 -0
- qiskit/circuit/classical/__init__.py +41 -0
- qiskit/circuit/classical/expr/__init__.py +238 -0
- qiskit/circuit/classical/expr/constructors.py +556 -0
- qiskit/circuit/classical/expr/expr.py +397 -0
- qiskit/circuit/classical/expr/visitors.py +300 -0
- qiskit/circuit/classical/types/__init__.py +109 -0
- qiskit/circuit/classical/types/ordering.py +222 -0
- qiskit/circuit/classical/types/types.py +117 -0
- qiskit/circuit/classicalfunction/__init__.py +140 -0
- qiskit/circuit/classicalfunction/boolean_expression.py +129 -0
- qiskit/circuit/classicalfunction/classical_element.py +54 -0
- qiskit/circuit/classicalfunction/classical_function_visitor.py +155 -0
- qiskit/circuit/classicalfunction/classicalfunction.py +173 -0
- qiskit/circuit/classicalfunction/exceptions.py +35 -0
- qiskit/circuit/classicalfunction/types.py +18 -0
- qiskit/circuit/classicalfunction/utils.py +91 -0
- qiskit/circuit/classicalregister.py +57 -0
- qiskit/circuit/commutation_checker.py +106 -0
- qiskit/circuit/commutation_library.py +20 -0
- qiskit/circuit/controlflow/__init__.py +28 -0
- qiskit/circuit/controlflow/_builder_utils.py +207 -0
- qiskit/circuit/controlflow/break_loop.py +56 -0
- qiskit/circuit/controlflow/builder.py +691 -0
- qiskit/circuit/controlflow/continue_loop.py +58 -0
- qiskit/circuit/controlflow/control_flow.py +84 -0
- qiskit/circuit/controlflow/for_loop.py +217 -0
- qiskit/circuit/controlflow/if_else.py +511 -0
- qiskit/circuit/controlflow/switch_case.py +417 -0
- qiskit/circuit/controlflow/while_loop.py +171 -0
- qiskit/circuit/controlledgate.py +274 -0
- qiskit/circuit/delay.py +123 -0
- qiskit/circuit/duration.py +95 -0
- qiskit/circuit/equivalence.py +94 -0
- qiskit/circuit/equivalence_library.py +18 -0
- qiskit/circuit/exceptions.py +19 -0
- qiskit/circuit/gate.py +263 -0
- qiskit/circuit/instruction.py +697 -0
- qiskit/circuit/instructionset.py +179 -0
- qiskit/circuit/library/__init__.py +668 -0
- qiskit/circuit/library/arithmetic/__init__.py +34 -0
- qiskit/circuit/library/arithmetic/adders/__init__.py +18 -0
- qiskit/circuit/library/arithmetic/adders/adder.py +210 -0
- qiskit/circuit/library/arithmetic/adders/cdkm_ripple_carry_adder.py +123 -0
- qiskit/circuit/library/arithmetic/adders/draper_qft_adder.py +129 -0
- qiskit/circuit/library/arithmetic/adders/vbe_ripple_carry_adder.py +95 -0
- qiskit/circuit/library/arithmetic/exact_reciprocal.py +88 -0
- qiskit/circuit/library/arithmetic/functional_pauli_rotations.py +114 -0
- qiskit/circuit/library/arithmetic/integer_comparator.py +243 -0
- qiskit/circuit/library/arithmetic/linear_amplitude_function.py +196 -0
- qiskit/circuit/library/arithmetic/linear_pauli_rotations.py +189 -0
- qiskit/circuit/library/arithmetic/multipliers/__init__.py +17 -0
- qiskit/circuit/library/arithmetic/multipliers/hrs_cumulative_multiplier.py +145 -0
- qiskit/circuit/library/arithmetic/multipliers/multiplier.py +192 -0
- qiskit/circuit/library/arithmetic/multipliers/rg_qft_multiplier.py +108 -0
- qiskit/circuit/library/arithmetic/piecewise_chebyshev.py +353 -0
- qiskit/circuit/library/arithmetic/piecewise_linear_pauli_rotations.py +277 -0
- qiskit/circuit/library/arithmetic/piecewise_polynomial_pauli_rotations.py +317 -0
- qiskit/circuit/library/arithmetic/polynomial_pauli_rotations.py +335 -0
- qiskit/circuit/library/arithmetic/quadratic_form.py +198 -0
- qiskit/circuit/library/arithmetic/weighted_adder.py +337 -0
- qiskit/circuit/library/basis_change/__init__.py +15 -0
- qiskit/circuit/library/basis_change/qft.py +313 -0
- qiskit/circuit/library/blueprintcircuit.py +280 -0
- qiskit/circuit/library/boolean_logic/__init__.py +18 -0
- qiskit/circuit/library/boolean_logic/inner_product.py +155 -0
- qiskit/circuit/library/boolean_logic/quantum_and.py +200 -0
- qiskit/circuit/library/boolean_logic/quantum_or.py +202 -0
- qiskit/circuit/library/boolean_logic/quantum_xor.py +165 -0
- qiskit/circuit/library/data_preparation/__init__.py +57 -0
- qiskit/circuit/library/data_preparation/_z_feature_map.py +115 -0
- qiskit/circuit/library/data_preparation/_zz_feature_map.py +150 -0
- qiskit/circuit/library/data_preparation/initializer.py +107 -0
- qiskit/circuit/library/data_preparation/pauli_feature_map.py +656 -0
- qiskit/circuit/library/data_preparation/state_preparation.py +336 -0
- qiskit/circuit/library/fourier_checking.py +158 -0
- qiskit/circuit/library/generalized_gates/__init__.py +30 -0
- qiskit/circuit/library/generalized_gates/diagonal.py +159 -0
- qiskit/circuit/library/generalized_gates/gms.py +174 -0
- qiskit/circuit/library/generalized_gates/gr.py +215 -0
- qiskit/circuit/library/generalized_gates/isometry.py +370 -0
- qiskit/circuit/library/generalized_gates/linear_function.py +318 -0
- qiskit/circuit/library/generalized_gates/mcg_up_to_diagonal.py +143 -0
- qiskit/circuit/library/generalized_gates/mcmt.py +316 -0
- qiskit/circuit/library/generalized_gates/pauli.py +85 -0
- qiskit/circuit/library/generalized_gates/permutation.py +194 -0
- qiskit/circuit/library/generalized_gates/rv.py +96 -0
- qiskit/circuit/library/generalized_gates/uc.py +213 -0
- qiskit/circuit/library/generalized_gates/uc_pauli_rot.py +164 -0
- qiskit/circuit/library/generalized_gates/ucrx.py +32 -0
- qiskit/circuit/library/generalized_gates/ucry.py +32 -0
- qiskit/circuit/library/generalized_gates/ucrz.py +32 -0
- qiskit/circuit/library/generalized_gates/unitary.py +215 -0
- qiskit/circuit/library/graph_state.py +169 -0
- qiskit/circuit/library/grover_operator.py +579 -0
- qiskit/circuit/library/hamiltonian_gate.py +142 -0
- qiskit/circuit/library/hidden_linear_function.py +161 -0
- qiskit/circuit/library/iqp.py +175 -0
- qiskit/circuit/library/n_local/__init__.py +45 -0
- qiskit/circuit/library/n_local/efficient_su2.py +277 -0
- qiskit/circuit/library/n_local/evolved_operator_ansatz.py +515 -0
- qiskit/circuit/library/n_local/excitation_preserving.py +297 -0
- qiskit/circuit/library/n_local/n_local.py +1472 -0
- qiskit/circuit/library/n_local/pauli_two_design.py +243 -0
- qiskit/circuit/library/n_local/qaoa_ansatz.py +366 -0
- qiskit/circuit/library/n_local/real_amplitudes.py +306 -0
- qiskit/circuit/library/n_local/two_local.py +289 -0
- qiskit/circuit/library/overlap.py +182 -0
- qiskit/circuit/library/pauli_evolution.py +186 -0
- qiskit/circuit/library/phase_estimation.py +175 -0
- qiskit/circuit/library/phase_oracle.py +153 -0
- qiskit/circuit/library/quantum_volume.py +167 -0
- qiskit/circuit/library/standard_gates/__init__.py +142 -0
- qiskit/circuit/library/standard_gates/dcx.py +78 -0
- qiskit/circuit/library/standard_gates/ecr.py +130 -0
- qiskit/circuit/library/standard_gates/equivalence_library.py +1800 -0
- qiskit/circuit/library/standard_gates/global_phase.py +85 -0
- qiskit/circuit/library/standard_gates/h.py +258 -0
- qiskit/circuit/library/standard_gates/i.py +76 -0
- qiskit/circuit/library/standard_gates/iswap.py +134 -0
- qiskit/circuit/library/standard_gates/multi_control_rotation_gates.py +405 -0
- qiskit/circuit/library/standard_gates/p.py +441 -0
- qiskit/circuit/library/standard_gates/r.py +117 -0
- qiskit/circuit/library/standard_gates/rx.py +303 -0
- qiskit/circuit/library/standard_gates/rxx.py +183 -0
- qiskit/circuit/library/standard_gates/ry.py +298 -0
- qiskit/circuit/library/standard_gates/ryy.py +183 -0
- qiskit/circuit/library/standard_gates/rz.py +319 -0
- qiskit/circuit/library/standard_gates/rzx.py +229 -0
- qiskit/circuit/library/standard_gates/rzz.py +196 -0
- qiskit/circuit/library/standard_gates/s.py +428 -0
- qiskit/circuit/library/standard_gates/swap.py +288 -0
- qiskit/circuit/library/standard_gates/sx.py +315 -0
- qiskit/circuit/library/standard_gates/t.py +179 -0
- qiskit/circuit/library/standard_gates/u.py +403 -0
- qiskit/circuit/library/standard_gates/u1.py +501 -0
- qiskit/circuit/library/standard_gates/u2.py +149 -0
- qiskit/circuit/library/standard_gates/u3.py +436 -0
- qiskit/circuit/library/standard_gates/x.py +1529 -0
- qiskit/circuit/library/standard_gates/xx_minus_yy.py +235 -0
- qiskit/circuit/library/standard_gates/xx_plus_yy.py +239 -0
- qiskit/circuit/library/standard_gates/y.py +262 -0
- qiskit/circuit/library/standard_gates/z.py +348 -0
- qiskit/circuit/library/templates/__init__.py +92 -0
- qiskit/circuit/library/templates/clifford/__init__.py +33 -0
- qiskit/circuit/library/templates/clifford/clifford_2_1.py +34 -0
- qiskit/circuit/library/templates/clifford/clifford_2_2.py +35 -0
- qiskit/circuit/library/templates/clifford/clifford_2_3.py +34 -0
- qiskit/circuit/library/templates/clifford/clifford_2_4.py +34 -0
- qiskit/circuit/library/templates/clifford/clifford_3_1.py +35 -0
- qiskit/circuit/library/templates/clifford/clifford_4_1.py +38 -0
- qiskit/circuit/library/templates/clifford/clifford_4_2.py +37 -0
- qiskit/circuit/library/templates/clifford/clifford_4_3.py +38 -0
- qiskit/circuit/library/templates/clifford/clifford_4_4.py +37 -0
- qiskit/circuit/library/templates/clifford/clifford_5_1.py +40 -0
- qiskit/circuit/library/templates/clifford/clifford_6_1.py +40 -0
- qiskit/circuit/library/templates/clifford/clifford_6_2.py +40 -0
- qiskit/circuit/library/templates/clifford/clifford_6_3.py +40 -0
- qiskit/circuit/library/templates/clifford/clifford_6_4.py +38 -0
- qiskit/circuit/library/templates/clifford/clifford_6_5.py +40 -0
- qiskit/circuit/library/templates/clifford/clifford_8_1.py +42 -0
- qiskit/circuit/library/templates/clifford/clifford_8_2.py +42 -0
- qiskit/circuit/library/templates/clifford/clifford_8_3.py +41 -0
- qiskit/circuit/library/templates/nct/__init__.py +67 -0
- qiskit/circuit/library/templates/nct/template_nct_2a_1.py +34 -0
- qiskit/circuit/library/templates/nct/template_nct_2a_2.py +35 -0
- qiskit/circuit/library/templates/nct/template_nct_2a_3.py +37 -0
- qiskit/circuit/library/templates/nct/template_nct_4a_1.py +43 -0
- qiskit/circuit/library/templates/nct/template_nct_4a_2.py +41 -0
- qiskit/circuit/library/templates/nct/template_nct_4a_3.py +39 -0
- qiskit/circuit/library/templates/nct/template_nct_4b_1.py +41 -0
- qiskit/circuit/library/templates/nct/template_nct_4b_2.py +39 -0
- qiskit/circuit/library/templates/nct/template_nct_5a_1.py +40 -0
- qiskit/circuit/library/templates/nct/template_nct_5a_2.py +40 -0
- qiskit/circuit/library/templates/nct/template_nct_5a_3.py +40 -0
- qiskit/circuit/library/templates/nct/template_nct_5a_4.py +39 -0
- qiskit/circuit/library/templates/nct/template_nct_6a_1.py +40 -0
- qiskit/circuit/library/templates/nct/template_nct_6a_2.py +41 -0
- qiskit/circuit/library/templates/nct/template_nct_6a_3.py +41 -0
- qiskit/circuit/library/templates/nct/template_nct_6a_4.py +41 -0
- qiskit/circuit/library/templates/nct/template_nct_6b_1.py +41 -0
- qiskit/circuit/library/templates/nct/template_nct_6b_2.py +41 -0
- qiskit/circuit/library/templates/nct/template_nct_6c_1.py +41 -0
- qiskit/circuit/library/templates/nct/template_nct_7a_1.py +43 -0
- qiskit/circuit/library/templates/nct/template_nct_7b_1.py +43 -0
- qiskit/circuit/library/templates/nct/template_nct_7c_1.py +43 -0
- qiskit/circuit/library/templates/nct/template_nct_7d_1.py +43 -0
- qiskit/circuit/library/templates/nct/template_nct_7e_1.py +43 -0
- qiskit/circuit/library/templates/nct/template_nct_9a_1.py +45 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_1.py +43 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_10.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_11.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_12.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_2.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_3.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_4.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_5.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_6.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_7.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_8.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9c_9.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9d_1.py +43 -0
- qiskit/circuit/library/templates/nct/template_nct_9d_10.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9d_2.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9d_3.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9d_4.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9d_5.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9d_6.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9d_7.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9d_8.py +44 -0
- qiskit/circuit/library/templates/nct/template_nct_9d_9.py +44 -0
- qiskit/circuit/library/templates/rzx/__init__.py +25 -0
- qiskit/circuit/library/templates/rzx/rzx_cy.py +47 -0
- qiskit/circuit/library/templates/rzx/rzx_xz.py +54 -0
- qiskit/circuit/library/templates/rzx/rzx_yz.py +45 -0
- qiskit/circuit/library/templates/rzx/rzx_zz1.py +69 -0
- qiskit/circuit/library/templates/rzx/rzx_zz2.py +59 -0
- qiskit/circuit/library/templates/rzx/rzx_zz3.py +59 -0
- qiskit/circuit/measure.py +44 -0
- qiskit/circuit/operation.py +67 -0
- qiskit/circuit/parameter.py +178 -0
- qiskit/circuit/parameterexpression.py +692 -0
- qiskit/circuit/parametertable.py +119 -0
- qiskit/circuit/parametervector.py +120 -0
- qiskit/circuit/quantumcircuit.py +6829 -0
- qiskit/circuit/quantumcircuitdata.py +136 -0
- qiskit/circuit/quantumregister.py +75 -0
- qiskit/circuit/random/__init__.py +15 -0
- qiskit/circuit/random/utils.py +358 -0
- qiskit/circuit/register.py +233 -0
- qiskit/circuit/reset.py +34 -0
- qiskit/circuit/singleton.py +606 -0
- qiskit/circuit/store.py +97 -0
- qiskit/circuit/tools/__init__.py +16 -0
- qiskit/circuit/tools/pi_check.py +190 -0
- qiskit/circuit/twirling.py +145 -0
- qiskit/compiler/__init__.py +33 -0
- qiskit/compiler/assembler.py +681 -0
- qiskit/compiler/scheduler.py +109 -0
- qiskit/compiler/sequencer.py +71 -0
- qiskit/compiler/transpiler.py +533 -0
- qiskit/converters/__init__.py +74 -0
- qiskit/converters/circuit_to_dag.py +78 -0
- qiskit/converters/circuit_to_dagdependency.py +51 -0
- qiskit/converters/circuit_to_dagdependency_v2.py +47 -0
- qiskit/converters/circuit_to_gate.py +107 -0
- qiskit/converters/circuit_to_instruction.py +155 -0
- qiskit/converters/dag_to_circuit.py +79 -0
- qiskit/converters/dag_to_dagdependency.py +55 -0
- qiskit/converters/dag_to_dagdependency_v2.py +44 -0
- qiskit/converters/dagdependency_to_circuit.py +46 -0
- qiskit/converters/dagdependency_to_dag.py +54 -0
- qiskit/dagcircuit/__init__.py +44 -0
- qiskit/dagcircuit/collect_blocks.py +391 -0
- qiskit/dagcircuit/dagcircuit.py +24 -0
- qiskit/dagcircuit/dagdependency.py +646 -0
- qiskit/dagcircuit/dagdependency_v2.py +641 -0
- qiskit/dagcircuit/dagdepnode.py +160 -0
- qiskit/dagcircuit/dagnode.py +176 -0
- qiskit/dagcircuit/exceptions.py +42 -0
- qiskit/exceptions.py +153 -0
- qiskit/passmanager/__init__.py +240 -0
- qiskit/passmanager/base_tasks.py +230 -0
- qiskit/passmanager/compilation_status.py +74 -0
- qiskit/passmanager/exceptions.py +19 -0
- qiskit/passmanager/flow_controllers.py +116 -0
- qiskit/passmanager/passmanager.py +333 -0
- qiskit/primitives/__init__.py +481 -0
- qiskit/primitives/backend_estimator.py +486 -0
- qiskit/primitives/backend_estimator_v2.py +434 -0
- qiskit/primitives/backend_sampler.py +222 -0
- qiskit/primitives/backend_sampler_v2.py +339 -0
- qiskit/primitives/base/__init__.py +20 -0
- qiskit/primitives/base/base_estimator.py +252 -0
- qiskit/primitives/base/base_primitive.py +45 -0
- qiskit/primitives/base/base_primitive_job.py +78 -0
- qiskit/primitives/base/base_result.py +65 -0
- qiskit/primitives/base/base_sampler.py +204 -0
- qiskit/primitives/base/estimator_result.py +46 -0
- qiskit/primitives/base/sampler_result.py +45 -0
- qiskit/primitives/base/validation.py +231 -0
- qiskit/primitives/containers/__init__.py +26 -0
- qiskit/primitives/containers/bindings_array.py +389 -0
- qiskit/primitives/containers/bit_array.py +741 -0
- qiskit/primitives/containers/data_bin.py +173 -0
- qiskit/primitives/containers/estimator_pub.py +222 -0
- qiskit/primitives/containers/object_array.py +94 -0
- qiskit/primitives/containers/observables_array.py +279 -0
- qiskit/primitives/containers/primitive_result.py +53 -0
- qiskit/primitives/containers/pub_result.py +51 -0
- qiskit/primitives/containers/sampler_pub.py +193 -0
- qiskit/primitives/containers/sampler_pub_result.py +74 -0
- qiskit/primitives/containers/shape.py +129 -0
- qiskit/primitives/estimator.py +172 -0
- qiskit/primitives/primitive_job.py +81 -0
- qiskit/primitives/sampler.py +162 -0
- qiskit/primitives/statevector_estimator.py +174 -0
- qiskit/primitives/statevector_sampler.py +292 -0
- qiskit/primitives/utils.py +247 -0
- qiskit/providers/__init__.py +803 -0
- qiskit/providers/backend.py +667 -0
- qiskit/providers/backend_compat.py +472 -0
- qiskit/providers/basic_provider/__init__.py +45 -0
- qiskit/providers/basic_provider/basic_provider.py +101 -0
- qiskit/providers/basic_provider/basic_provider_job.py +65 -0
- qiskit/providers/basic_provider/basic_provider_tools.py +218 -0
- qiskit/providers/basic_provider/basic_simulator.py +821 -0
- qiskit/providers/basic_provider/exceptions.py +30 -0
- qiskit/providers/exceptions.py +45 -0
- qiskit/providers/fake_provider/__init__.py +105 -0
- qiskit/providers/fake_provider/backends_v1/__init__.py +22 -0
- qiskit/providers/fake_provider/backends_v1/fake_127q_pulse/__init__.py +18 -0
- qiskit/providers/fake_provider/backends_v1/fake_127q_pulse/conf_washington.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_127q_pulse/defs_washington.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_127q_pulse/fake_127q_pulse_v1.py +37 -0
- qiskit/providers/fake_provider/backends_v1/fake_127q_pulse/props_washington.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_20q/__init__.py +18 -0
- qiskit/providers/fake_provider/backends_v1/fake_20q/conf_singapore.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_20q/fake_20q.py +43 -0
- qiskit/providers/fake_provider/backends_v1/fake_20q/props_singapore.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_27q_pulse/__init__.py +18 -0
- qiskit/providers/fake_provider/backends_v1/fake_27q_pulse/conf_hanoi.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_27q_pulse/defs_hanoi.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_27q_pulse/fake_27q_pulse_v1.py +50 -0
- qiskit/providers/fake_provider/backends_v1/fake_27q_pulse/props_hanoi.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_5q/__init__.py +18 -0
- qiskit/providers/fake_provider/backends_v1/fake_5q/conf_yorktown.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_5q/fake_5q_v1.py +41 -0
- qiskit/providers/fake_provider/backends_v1/fake_5q/props_yorktown.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_7q_pulse/__init__.py +18 -0
- qiskit/providers/fake_provider/backends_v1/fake_7q_pulse/conf_nairobi.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_7q_pulse/defs_nairobi.json +1 -0
- qiskit/providers/fake_provider/backends_v1/fake_7q_pulse/fake_7q_pulse_v1.py +44 -0
- qiskit/providers/fake_provider/backends_v1/fake_7q_pulse/props_nairobi.json +1 -0
- qiskit/providers/fake_provider/fake_1q.py +91 -0
- qiskit/providers/fake_provider/fake_backend.py +165 -0
- qiskit/providers/fake_provider/fake_openpulse_2q.py +391 -0
- qiskit/providers/fake_provider/fake_openpulse_3q.py +340 -0
- qiskit/providers/fake_provider/fake_pulse_backend.py +49 -0
- qiskit/providers/fake_provider/fake_qasm_backend.py +77 -0
- qiskit/providers/fake_provider/generic_backend_v2.py +1035 -0
- qiskit/providers/fake_provider/utils/__init__.py +15 -0
- qiskit/providers/fake_provider/utils/backend_converter.py +150 -0
- qiskit/providers/fake_provider/utils/json_decoder.py +109 -0
- qiskit/providers/job.py +147 -0
- qiskit/providers/jobstatus.py +30 -0
- qiskit/providers/models/__init__.py +89 -0
- qiskit/providers/models/backendconfiguration.py +1040 -0
- qiskit/providers/models/backendproperties.py +517 -0
- qiskit/providers/models/backendstatus.py +94 -0
- qiskit/providers/models/jobstatus.py +66 -0
- qiskit/providers/models/pulsedefaults.py +305 -0
- qiskit/providers/options.py +273 -0
- qiskit/providers/provider.py +95 -0
- qiskit/providers/providerutils.py +110 -0
- qiskit/pulse/__init__.py +158 -0
- qiskit/pulse/builder.py +2254 -0
- qiskit/pulse/calibration_entries.py +381 -0
- qiskit/pulse/channels.py +227 -0
- qiskit/pulse/configuration.py +245 -0
- qiskit/pulse/exceptions.py +45 -0
- qiskit/pulse/filters.py +309 -0
- qiskit/pulse/instruction_schedule_map.py +424 -0
- qiskit/pulse/instructions/__init__.py +67 -0
- qiskit/pulse/instructions/acquire.py +150 -0
- qiskit/pulse/instructions/delay.py +71 -0
- qiskit/pulse/instructions/directives.py +154 -0
- qiskit/pulse/instructions/frequency.py +135 -0
- qiskit/pulse/instructions/instruction.py +270 -0
- qiskit/pulse/instructions/phase.py +152 -0
- qiskit/pulse/instructions/play.py +99 -0
- qiskit/pulse/instructions/reference.py +100 -0
- qiskit/pulse/instructions/snapshot.py +82 -0
- qiskit/pulse/library/__init__.py +97 -0
- qiskit/pulse/library/continuous.py +430 -0
- qiskit/pulse/library/pulse.py +148 -0
- qiskit/pulse/library/samplers/__init__.py +15 -0
- qiskit/pulse/library/samplers/decorators.py +295 -0
- qiskit/pulse/library/samplers/strategies.py +71 -0
- qiskit/pulse/library/symbolic_pulses.py +1988 -0
- qiskit/pulse/library/waveform.py +136 -0
- qiskit/pulse/macros.py +262 -0
- qiskit/pulse/parameter_manager.py +445 -0
- qiskit/pulse/parser.py +314 -0
- qiskit/pulse/reference_manager.py +58 -0
- qiskit/pulse/schedule.py +1854 -0
- qiskit/pulse/transforms/__init__.py +106 -0
- qiskit/pulse/transforms/alignments.py +406 -0
- qiskit/pulse/transforms/base_transforms.py +71 -0
- qiskit/pulse/transforms/canonicalization.py +498 -0
- qiskit/pulse/transforms/dag.py +122 -0
- qiskit/pulse/utils.py +149 -0
- qiskit/qasm/libs/dummy/stdgates.inc +75 -0
- qiskit/qasm/libs/qelib1.inc +266 -0
- qiskit/qasm/libs/stdgates.inc +82 -0
- qiskit/qasm2/__init__.py +654 -0
- qiskit/qasm2/exceptions.py +27 -0
- qiskit/qasm2/export.py +372 -0
- qiskit/qasm2/parse.py +452 -0
- qiskit/qasm3/__init__.py +367 -0
- qiskit/qasm3/ast.py +738 -0
- qiskit/qasm3/exceptions.py +27 -0
- qiskit/qasm3/experimental.py +70 -0
- qiskit/qasm3/exporter.py +1299 -0
- qiskit/qasm3/printer.py +577 -0
- qiskit/qobj/__init__.py +75 -0
- qiskit/qobj/common.py +81 -0
- qiskit/qobj/converters/__init__.py +18 -0
- qiskit/qobj/converters/lo_config.py +177 -0
- qiskit/qobj/converters/pulse_instruction.py +897 -0
- qiskit/qobj/pulse_qobj.py +709 -0
- qiskit/qobj/qasm_qobj.py +708 -0
- qiskit/qobj/utils.py +46 -0
- qiskit/qpy/__init__.py +1822 -0
- qiskit/qpy/binary_io/__init__.py +36 -0
- qiskit/qpy/binary_io/circuits.py +1475 -0
- qiskit/qpy/binary_io/schedules.py +635 -0
- qiskit/qpy/binary_io/value.py +1025 -0
- qiskit/qpy/common.py +350 -0
- qiskit/qpy/exceptions.py +53 -0
- qiskit/qpy/formats.py +401 -0
- qiskit/qpy/interface.py +377 -0
- qiskit/qpy/type_keys.py +572 -0
- qiskit/quantum_info/__init__.py +162 -0
- qiskit/quantum_info/analysis/__init__.py +17 -0
- qiskit/quantum_info/analysis/average.py +47 -0
- qiskit/quantum_info/analysis/distance.py +102 -0
- qiskit/quantum_info/analysis/make_observable.py +44 -0
- qiskit/quantum_info/analysis/z2_symmetries.py +484 -0
- qiskit/quantum_info/operators/__init__.py +28 -0
- qiskit/quantum_info/operators/base_operator.py +145 -0
- qiskit/quantum_info/operators/channel/__init__.py +29 -0
- qiskit/quantum_info/operators/channel/chi.py +191 -0
- qiskit/quantum_info/operators/channel/choi.py +218 -0
- qiskit/quantum_info/operators/channel/kraus.py +337 -0
- qiskit/quantum_info/operators/channel/ptm.py +204 -0
- qiskit/quantum_info/operators/channel/quantum_channel.py +348 -0
- qiskit/quantum_info/operators/channel/stinespring.py +296 -0
- qiskit/quantum_info/operators/channel/superop.py +377 -0
- qiskit/quantum_info/operators/channel/transformations.py +475 -0
- qiskit/quantum_info/operators/custom_iterator.py +48 -0
- qiskit/quantum_info/operators/dihedral/__init__.py +18 -0
- qiskit/quantum_info/operators/dihedral/dihedral.py +509 -0
- qiskit/quantum_info/operators/dihedral/dihedral_circuits.py +216 -0
- qiskit/quantum_info/operators/dihedral/polynomial.py +313 -0
- qiskit/quantum_info/operators/dihedral/random.py +64 -0
- qiskit/quantum_info/operators/linear_op.py +25 -0
- qiskit/quantum_info/operators/measures.py +418 -0
- qiskit/quantum_info/operators/mixins/__init__.py +52 -0
- qiskit/quantum_info/operators/mixins/adjoint.py +52 -0
- qiskit/quantum_info/operators/mixins/group.py +171 -0
- qiskit/quantum_info/operators/mixins/linear.py +84 -0
- qiskit/quantum_info/operators/mixins/multiply.py +62 -0
- qiskit/quantum_info/operators/mixins/tolerances.py +72 -0
- qiskit/quantum_info/operators/op_shape.py +525 -0
- qiskit/quantum_info/operators/operator.py +865 -0
- qiskit/quantum_info/operators/operator_utils.py +76 -0
- qiskit/quantum_info/operators/predicates.py +183 -0
- qiskit/quantum_info/operators/random.py +154 -0
- qiskit/quantum_info/operators/scalar_op.py +254 -0
- qiskit/quantum_info/operators/symplectic/__init__.py +23 -0
- qiskit/quantum_info/operators/symplectic/base_pauli.py +719 -0
- qiskit/quantum_info/operators/symplectic/clifford.py +1030 -0
- qiskit/quantum_info/operators/symplectic/clifford_circuits.py +558 -0
- qiskit/quantum_info/operators/symplectic/pauli.py +753 -0
- qiskit/quantum_info/operators/symplectic/pauli_list.py +1230 -0
- qiskit/quantum_info/operators/symplectic/pauli_utils.py +40 -0
- qiskit/quantum_info/operators/symplectic/random.py +117 -0
- qiskit/quantum_info/operators/symplectic/sparse_pauli_op.py +1196 -0
- qiskit/quantum_info/operators/utils/__init__.py +20 -0
- qiskit/quantum_info/operators/utils/anti_commutator.py +36 -0
- qiskit/quantum_info/operators/utils/commutator.py +36 -0
- qiskit/quantum_info/operators/utils/double_commutator.py +76 -0
- qiskit/quantum_info/quaternion.py +156 -0
- qiskit/quantum_info/random.py +26 -0
- qiskit/quantum_info/states/__init__.py +28 -0
- qiskit/quantum_info/states/densitymatrix.py +845 -0
- qiskit/quantum_info/states/measures.py +288 -0
- qiskit/quantum_info/states/quantum_state.py +503 -0
- qiskit/quantum_info/states/random.py +157 -0
- qiskit/quantum_info/states/stabilizerstate.py +773 -0
- qiskit/quantum_info/states/statevector.py +958 -0
- qiskit/quantum_info/states/utils.py +247 -0
- qiskit/result/__init__.py +73 -0
- qiskit/result/counts.py +189 -0
- qiskit/result/distributions/__init__.py +17 -0
- qiskit/result/distributions/probability.py +100 -0
- qiskit/result/distributions/quasi.py +154 -0
- qiskit/result/exceptions.py +40 -0
- qiskit/result/mitigation/__init__.py +13 -0
- qiskit/result/mitigation/base_readout_mitigator.py +79 -0
- qiskit/result/mitigation/correlated_readout_mitigator.py +277 -0
- qiskit/result/mitigation/local_readout_mitigator.py +328 -0
- qiskit/result/mitigation/utils.py +217 -0
- qiskit/result/models.py +234 -0
- qiskit/result/postprocess.py +239 -0
- qiskit/result/result.py +392 -0
- qiskit/result/sampled_expval.py +75 -0
- qiskit/result/utils.py +295 -0
- qiskit/scheduler/__init__.py +40 -0
- qiskit/scheduler/config.py +37 -0
- qiskit/scheduler/lowering.py +187 -0
- qiskit/scheduler/methods/__init__.py +15 -0
- qiskit/scheduler/methods/basic.py +140 -0
- qiskit/scheduler/schedule_circuit.py +69 -0
- qiskit/scheduler/sequence.py +104 -0
- qiskit/synthesis/__init__.py +220 -0
- qiskit/synthesis/arithmetic/__init__.py +16 -0
- qiskit/synthesis/arithmetic/adders/__init__.py +17 -0
- qiskit/synthesis/arithmetic/adders/cdkm_ripple_carry_adder.py +154 -0
- qiskit/synthesis/arithmetic/adders/draper_qft_adder.py +103 -0
- qiskit/synthesis/arithmetic/adders/vbe_ripple_carry_adder.py +161 -0
- qiskit/synthesis/arithmetic/multipliers/__init__.py +16 -0
- qiskit/synthesis/arithmetic/multipliers/hrs_cumulative_multiplier.py +102 -0
- qiskit/synthesis/arithmetic/multipliers/rg_qft_multiplier.py +99 -0
- qiskit/synthesis/clifford/__init__.py +19 -0
- qiskit/synthesis/clifford/clifford_decompose_ag.py +178 -0
- qiskit/synthesis/clifford/clifford_decompose_bm.py +46 -0
- qiskit/synthesis/clifford/clifford_decompose_full.py +64 -0
- qiskit/synthesis/clifford/clifford_decompose_greedy.py +58 -0
- qiskit/synthesis/clifford/clifford_decompose_layers.py +447 -0
- qiskit/synthesis/cnotdihedral/__init__.py +17 -0
- qiskit/synthesis/cnotdihedral/cnotdihedral_decompose_full.py +52 -0
- qiskit/synthesis/cnotdihedral/cnotdihedral_decompose_general.py +141 -0
- qiskit/synthesis/cnotdihedral/cnotdihedral_decompose_two_qubits.py +266 -0
- qiskit/synthesis/discrete_basis/__init__.py +16 -0
- qiskit/synthesis/discrete_basis/commutator_decompose.py +241 -0
- qiskit/synthesis/discrete_basis/gate_sequence.py +415 -0
- qiskit/synthesis/discrete_basis/generate_basis_approximations.py +163 -0
- qiskit/synthesis/discrete_basis/solovay_kitaev.py +217 -0
- qiskit/synthesis/evolution/__init__.py +21 -0
- qiskit/synthesis/evolution/evolution_synthesis.py +48 -0
- qiskit/synthesis/evolution/lie_trotter.py +117 -0
- qiskit/synthesis/evolution/matrix_synthesis.py +47 -0
- qiskit/synthesis/evolution/pauli_network.py +80 -0
- qiskit/synthesis/evolution/product_formula.py +311 -0
- qiskit/synthesis/evolution/qdrift.py +138 -0
- qiskit/synthesis/evolution/suzuki_trotter.py +215 -0
- qiskit/synthesis/linear/__init__.py +26 -0
- qiskit/synthesis/linear/cnot_synth.py +69 -0
- qiskit/synthesis/linear/linear_circuits_utils.py +128 -0
- qiskit/synthesis/linear/linear_depth_lnn.py +276 -0
- qiskit/synthesis/linear/linear_matrix_utils.py +27 -0
- qiskit/synthesis/linear_phase/__init__.py +17 -0
- qiskit/synthesis/linear_phase/cnot_phase_synth.py +206 -0
- qiskit/synthesis/linear_phase/cx_cz_depth_lnn.py +262 -0
- qiskit/synthesis/linear_phase/cz_depth_lnn.py +58 -0
- qiskit/synthesis/multi_controlled/__init__.py +24 -0
- qiskit/synthesis/multi_controlled/mcmt_vchain.py +52 -0
- qiskit/synthesis/multi_controlled/mcx_synthesis.py +356 -0
- qiskit/synthesis/one_qubit/__init__.py +15 -0
- qiskit/synthesis/one_qubit/one_qubit_decompose.py +288 -0
- qiskit/synthesis/permutation/__init__.py +18 -0
- qiskit/synthesis/permutation/permutation_full.py +78 -0
- qiskit/synthesis/permutation/permutation_lnn.py +54 -0
- qiskit/synthesis/permutation/permutation_reverse_lnn.py +93 -0
- qiskit/synthesis/permutation/permutation_utils.py +16 -0
- qiskit/synthesis/qft/__init__.py +16 -0
- qiskit/synthesis/qft/qft_decompose_full.py +97 -0
- qiskit/synthesis/qft/qft_decompose_lnn.py +79 -0
- qiskit/synthesis/stabilizer/__init__.py +16 -0
- qiskit/synthesis/stabilizer/stabilizer_circuit.py +149 -0
- qiskit/synthesis/stabilizer/stabilizer_decompose.py +194 -0
- qiskit/synthesis/two_qubit/__init__.py +19 -0
- qiskit/synthesis/two_qubit/local_invariance.py +63 -0
- qiskit/synthesis/two_qubit/two_qubit_decompose.py +700 -0
- qiskit/synthesis/two_qubit/xx_decompose/__init__.py +19 -0
- qiskit/synthesis/two_qubit/xx_decompose/circuits.py +300 -0
- qiskit/synthesis/two_qubit/xx_decompose/decomposer.py +324 -0
- qiskit/synthesis/two_qubit/xx_decompose/embodiments.py +163 -0
- qiskit/synthesis/two_qubit/xx_decompose/paths.py +412 -0
- qiskit/synthesis/two_qubit/xx_decompose/polytopes.py +262 -0
- qiskit/synthesis/two_qubit/xx_decompose/utilities.py +40 -0
- qiskit/synthesis/two_qubit/xx_decompose/weyl.py +133 -0
- qiskit/synthesis/unitary/__init__.py +13 -0
- qiskit/synthesis/unitary/aqc/__init__.py +177 -0
- qiskit/synthesis/unitary/aqc/approximate.py +116 -0
- qiskit/synthesis/unitary/aqc/aqc.py +175 -0
- qiskit/synthesis/unitary/aqc/cnot_structures.py +300 -0
- qiskit/synthesis/unitary/aqc/cnot_unit_circuit.py +103 -0
- qiskit/synthesis/unitary/aqc/cnot_unit_objective.py +299 -0
- qiskit/synthesis/unitary/aqc/elementary_operations.py +108 -0
- qiskit/synthesis/unitary/aqc/fast_gradient/__init__.py +164 -0
- qiskit/synthesis/unitary/aqc/fast_gradient/fast_grad_utils.py +237 -0
- qiskit/synthesis/unitary/aqc/fast_gradient/fast_gradient.py +226 -0
- qiskit/synthesis/unitary/aqc/fast_gradient/layer.py +370 -0
- qiskit/synthesis/unitary/aqc/fast_gradient/pmatrix.py +312 -0
- qiskit/synthesis/unitary/qsd.py +288 -0
- qiskit/transpiler/__init__.py +1290 -0
- qiskit/transpiler/basepasses.py +221 -0
- qiskit/transpiler/coupling.py +500 -0
- qiskit/transpiler/exceptions.py +59 -0
- qiskit/transpiler/instruction_durations.py +281 -0
- qiskit/transpiler/layout.py +737 -0
- qiskit/transpiler/passes/__init__.py +312 -0
- qiskit/transpiler/passes/analysis/__init__.py +23 -0
- qiskit/transpiler/passes/analysis/count_ops.py +30 -0
- qiskit/transpiler/passes/analysis/count_ops_longest_path.py +26 -0
- qiskit/transpiler/passes/analysis/dag_longest_path.py +24 -0
- qiskit/transpiler/passes/analysis/depth.py +33 -0
- qiskit/transpiler/passes/analysis/num_qubits.py +26 -0
- qiskit/transpiler/passes/analysis/num_tensor_factors.py +26 -0
- qiskit/transpiler/passes/analysis/resource_estimation.py +41 -0
- qiskit/transpiler/passes/analysis/size.py +36 -0
- qiskit/transpiler/passes/analysis/width.py +27 -0
- qiskit/transpiler/passes/basis/__init__.py +19 -0
- qiskit/transpiler/passes/basis/basis_translator.py +137 -0
- qiskit/transpiler/passes/basis/decompose.py +131 -0
- qiskit/transpiler/passes/basis/translate_parameterized.py +175 -0
- qiskit/transpiler/passes/basis/unroll_3q_or_more.py +88 -0
- qiskit/transpiler/passes/basis/unroll_custom_definitions.py +109 -0
- qiskit/transpiler/passes/calibration/__init__.py +17 -0
- qiskit/transpiler/passes/calibration/base_builder.py +79 -0
- qiskit/transpiler/passes/calibration/builders.py +20 -0
- qiskit/transpiler/passes/calibration/exceptions.py +22 -0
- qiskit/transpiler/passes/calibration/pulse_gate.py +100 -0
- qiskit/transpiler/passes/calibration/rx_builder.py +164 -0
- qiskit/transpiler/passes/calibration/rzx_builder.py +411 -0
- qiskit/transpiler/passes/calibration/rzx_templates.py +51 -0
- qiskit/transpiler/passes/layout/__init__.py +26 -0
- qiskit/transpiler/passes/layout/_csp_custom_solver.py +65 -0
- qiskit/transpiler/passes/layout/apply_layout.py +123 -0
- qiskit/transpiler/passes/layout/csp_layout.py +132 -0
- qiskit/transpiler/passes/layout/dense_layout.py +202 -0
- qiskit/transpiler/passes/layout/disjoint_utils.py +219 -0
- qiskit/transpiler/passes/layout/enlarge_with_ancilla.py +49 -0
- qiskit/transpiler/passes/layout/full_ancilla_allocation.py +117 -0
- qiskit/transpiler/passes/layout/layout_2q_distance.py +77 -0
- qiskit/transpiler/passes/layout/sabre_layout.py +487 -0
- qiskit/transpiler/passes/layout/sabre_pre_layout.py +225 -0
- qiskit/transpiler/passes/layout/set_layout.py +69 -0
- qiskit/transpiler/passes/layout/trivial_layout.py +66 -0
- qiskit/transpiler/passes/layout/vf2_layout.py +263 -0
- qiskit/transpiler/passes/layout/vf2_post_layout.py +419 -0
- qiskit/transpiler/passes/layout/vf2_utils.py +260 -0
- qiskit/transpiler/passes/optimization/__init__.py +43 -0
- qiskit/transpiler/passes/optimization/_gate_extension.py +80 -0
- qiskit/transpiler/passes/optimization/collect_1q_runs.py +31 -0
- qiskit/transpiler/passes/optimization/collect_2q_blocks.py +35 -0
- qiskit/transpiler/passes/optimization/collect_and_collapse.py +115 -0
- qiskit/transpiler/passes/optimization/collect_cliffords.py +104 -0
- qiskit/transpiler/passes/optimization/collect_linear_functions.py +80 -0
- qiskit/transpiler/passes/optimization/collect_multiqubit_blocks.py +227 -0
- qiskit/transpiler/passes/optimization/commutation_analysis.py +44 -0
- qiskit/transpiler/passes/optimization/commutative_cancellation.py +82 -0
- qiskit/transpiler/passes/optimization/commutative_inverse_cancellation.py +140 -0
- qiskit/transpiler/passes/optimization/consolidate_blocks.py +149 -0
- qiskit/transpiler/passes/optimization/cx_cancellation.py +65 -0
- qiskit/transpiler/passes/optimization/echo_rzx_weyl_decomposition.py +162 -0
- qiskit/transpiler/passes/optimization/elide_permutations.py +91 -0
- qiskit/transpiler/passes/optimization/hoare_opt.py +420 -0
- qiskit/transpiler/passes/optimization/inverse_cancellation.py +95 -0
- qiskit/transpiler/passes/optimization/normalize_rx_angle.py +149 -0
- qiskit/transpiler/passes/optimization/optimize_1q_commutation.py +268 -0
- qiskit/transpiler/passes/optimization/optimize_1q_decomposition.py +254 -0
- qiskit/transpiler/passes/optimization/optimize_1q_gates.py +384 -0
- qiskit/transpiler/passes/optimization/optimize_annotated.py +448 -0
- qiskit/transpiler/passes/optimization/optimize_cliffords.py +89 -0
- qiskit/transpiler/passes/optimization/optimize_swap_before_measure.py +71 -0
- qiskit/transpiler/passes/optimization/remove_diagonal_gates_before_measure.py +41 -0
- qiskit/transpiler/passes/optimization/remove_final_reset.py +37 -0
- qiskit/transpiler/passes/optimization/remove_identity_equiv.py +69 -0
- qiskit/transpiler/passes/optimization/remove_reset_in_zero_state.py +37 -0
- qiskit/transpiler/passes/optimization/reset_after_measure_simplification.py +47 -0
- qiskit/transpiler/passes/optimization/split_2q_unitaries.py +40 -0
- qiskit/transpiler/passes/optimization/template_matching/__init__.py +19 -0
- qiskit/transpiler/passes/optimization/template_matching/backward_match.py +749 -0
- qiskit/transpiler/passes/optimization/template_matching/forward_match.py +452 -0
- qiskit/transpiler/passes/optimization/template_matching/maximal_matches.py +77 -0
- qiskit/transpiler/passes/optimization/template_matching/template_matching.py +370 -0
- qiskit/transpiler/passes/optimization/template_matching/template_substitution.py +638 -0
- qiskit/transpiler/passes/optimization/template_optimization.py +158 -0
- qiskit/transpiler/passes/routing/__init__.py +22 -0
- qiskit/transpiler/passes/routing/algorithms/__init__.py +33 -0
- qiskit/transpiler/passes/routing/algorithms/token_swapper.py +105 -0
- qiskit/transpiler/passes/routing/algorithms/types.py +46 -0
- qiskit/transpiler/passes/routing/algorithms/util.py +103 -0
- qiskit/transpiler/passes/routing/basic_swap.py +166 -0
- qiskit/transpiler/passes/routing/commuting_2q_gate_routing/__init__.py +25 -0
- qiskit/transpiler/passes/routing/commuting_2q_gate_routing/commuting_2q_block.py +60 -0
- qiskit/transpiler/passes/routing/commuting_2q_gate_routing/commuting_2q_gate_router.py +395 -0
- qiskit/transpiler/passes/routing/commuting_2q_gate_routing/pauli_2q_evolution_commutation.py +145 -0
- qiskit/transpiler/passes/routing/commuting_2q_gate_routing/swap_strategy.py +306 -0
- qiskit/transpiler/passes/routing/layout_transformation.py +119 -0
- qiskit/transpiler/passes/routing/lookahead_swap.py +390 -0
- qiskit/transpiler/passes/routing/sabre_swap.py +447 -0
- qiskit/transpiler/passes/routing/star_prerouting.py +392 -0
- qiskit/transpiler/passes/routing/stochastic_swap.py +532 -0
- qiskit/transpiler/passes/routing/utils.py +35 -0
- qiskit/transpiler/passes/scheduling/__init__.py +27 -0
- qiskit/transpiler/passes/scheduling/alap.py +153 -0
- qiskit/transpiler/passes/scheduling/alignments/__init__.py +81 -0
- qiskit/transpiler/passes/scheduling/alignments/align_measures.py +255 -0
- qiskit/transpiler/passes/scheduling/alignments/check_durations.py +78 -0
- qiskit/transpiler/passes/scheduling/alignments/pulse_gate_validation.py +107 -0
- qiskit/transpiler/passes/scheduling/alignments/reschedule.py +250 -0
- qiskit/transpiler/passes/scheduling/asap.py +175 -0
- qiskit/transpiler/passes/scheduling/base_scheduler.py +310 -0
- qiskit/transpiler/passes/scheduling/dynamical_decoupling.py +312 -0
- qiskit/transpiler/passes/scheduling/padding/__init__.py +16 -0
- qiskit/transpiler/passes/scheduling/padding/base_padding.py +256 -0
- qiskit/transpiler/passes/scheduling/padding/dynamical_decoupling.py +452 -0
- qiskit/transpiler/passes/scheduling/padding/pad_delay.py +82 -0
- qiskit/transpiler/passes/scheduling/scheduling/__init__.py +17 -0
- qiskit/transpiler/passes/scheduling/scheduling/alap.py +127 -0
- qiskit/transpiler/passes/scheduling/scheduling/asap.py +131 -0
- qiskit/transpiler/passes/scheduling/scheduling/base_scheduler.py +94 -0
- qiskit/transpiler/passes/scheduling/scheduling/set_io_latency.py +64 -0
- qiskit/transpiler/passes/scheduling/time_unit_conversion.py +165 -0
- qiskit/transpiler/passes/synthesis/__init__.py +20 -0
- qiskit/transpiler/passes/synthesis/aqc_plugin.py +153 -0
- qiskit/transpiler/passes/synthesis/high_level_synthesis.py +854 -0
- qiskit/transpiler/passes/synthesis/hls_plugins.py +1559 -0
- qiskit/transpiler/passes/synthesis/linear_functions_synthesis.py +41 -0
- qiskit/transpiler/passes/synthesis/plugin.py +734 -0
- qiskit/transpiler/passes/synthesis/solovay_kitaev_synthesis.py +297 -0
- qiskit/transpiler/passes/synthesis/unitary_synthesis.py +1076 -0
- qiskit/transpiler/passes/utils/__init__.py +33 -0
- qiskit/transpiler/passes/utils/barrier_before_final_measurements.py +41 -0
- qiskit/transpiler/passes/utils/check_gate_direction.py +52 -0
- qiskit/transpiler/passes/utils/check_map.py +78 -0
- qiskit/transpiler/passes/utils/contains_instruction.py +45 -0
- qiskit/transpiler/passes/utils/control_flow.py +65 -0
- qiskit/transpiler/passes/utils/convert_conditions_to_if_ops.py +93 -0
- qiskit/transpiler/passes/utils/dag_fixed_point.py +36 -0
- qiskit/transpiler/passes/utils/error.py +69 -0
- qiskit/transpiler/passes/utils/filter_op_nodes.py +65 -0
- qiskit/transpiler/passes/utils/fixed_point.py +48 -0
- qiskit/transpiler/passes/utils/gate_direction.py +86 -0
- qiskit/transpiler/passes/utils/gates_basis.py +51 -0
- qiskit/transpiler/passes/utils/merge_adjacent_barriers.py +163 -0
- qiskit/transpiler/passes/utils/minimum_point.py +118 -0
- qiskit/transpiler/passes/utils/remove_barriers.py +49 -0
- qiskit/transpiler/passes/utils/remove_final_measurements.py +114 -0
- qiskit/transpiler/passes/utils/unroll_forloops.py +81 -0
- qiskit/transpiler/passmanager.py +490 -0
- qiskit/transpiler/passmanager_config.py +216 -0
- qiskit/transpiler/preset_passmanagers/__init__.py +73 -0
- qiskit/transpiler/preset_passmanagers/builtin_plugins.py +1045 -0
- qiskit/transpiler/preset_passmanagers/common.py +649 -0
- qiskit/transpiler/preset_passmanagers/generate_preset_pass_manager.py +626 -0
- qiskit/transpiler/preset_passmanagers/level0.py +113 -0
- qiskit/transpiler/preset_passmanagers/level1.py +120 -0
- qiskit/transpiler/preset_passmanagers/level2.py +119 -0
- qiskit/transpiler/preset_passmanagers/level3.py +119 -0
- qiskit/transpiler/preset_passmanagers/plugin.py +353 -0
- qiskit/transpiler/target.py +1319 -0
- qiskit/transpiler/timing_constraints.py +59 -0
- qiskit/user_config.py +262 -0
- qiskit/utils/__init__.py +89 -0
- qiskit/utils/classtools.py +146 -0
- qiskit/utils/deprecate_pulse.py +119 -0
- qiskit/utils/deprecation.py +490 -0
- qiskit/utils/lazy_tester.py +363 -0
- qiskit/utils/multiprocessing.py +56 -0
- qiskit/utils/optionals.py +347 -0
- qiskit/utils/parallel.py +191 -0
- qiskit/utils/units.py +143 -0
- qiskit/version.py +84 -0
- qiskit/visualization/__init__.py +288 -0
- qiskit/visualization/array.py +204 -0
- qiskit/visualization/bloch.py +778 -0
- qiskit/visualization/circuit/__init__.py +15 -0
- qiskit/visualization/circuit/_utils.py +675 -0
- qiskit/visualization/circuit/circuit_visualization.py +727 -0
- qiskit/visualization/circuit/latex.py +661 -0
- qiskit/visualization/circuit/matplotlib.py +2029 -0
- qiskit/visualization/circuit/qcstyle.py +278 -0
- qiskit/visualization/circuit/styles/__init__.py +13 -0
- qiskit/visualization/circuit/styles/bw.json +202 -0
- qiskit/visualization/circuit/styles/clifford.json +202 -0
- qiskit/visualization/circuit/styles/iqp-dark.json +214 -0
- qiskit/visualization/circuit/styles/iqp.json +214 -0
- qiskit/visualization/circuit/styles/textbook.json +202 -0
- qiskit/visualization/circuit/text.py +1844 -0
- qiskit/visualization/circuit_visualization.py +19 -0
- qiskit/visualization/counts_visualization.py +481 -0
- qiskit/visualization/dag_visualization.py +316 -0
- qiskit/visualization/exceptions.py +21 -0
- qiskit/visualization/gate_map.py +1485 -0
- qiskit/visualization/library.py +37 -0
- qiskit/visualization/pass_manager_visualization.py +308 -0
- qiskit/visualization/pulse_v2/__init__.py +21 -0
- qiskit/visualization/pulse_v2/core.py +901 -0
- qiskit/visualization/pulse_v2/device_info.py +173 -0
- qiskit/visualization/pulse_v2/drawings.py +253 -0
- qiskit/visualization/pulse_v2/events.py +254 -0
- qiskit/visualization/pulse_v2/generators/__init__.py +40 -0
- qiskit/visualization/pulse_v2/generators/barrier.py +76 -0
- qiskit/visualization/pulse_v2/generators/chart.py +208 -0
- qiskit/visualization/pulse_v2/generators/frame.py +436 -0
- qiskit/visualization/pulse_v2/generators/snapshot.py +133 -0
- qiskit/visualization/pulse_v2/generators/waveform.py +645 -0
- qiskit/visualization/pulse_v2/interface.py +458 -0
- qiskit/visualization/pulse_v2/layouts.py +387 -0
- qiskit/visualization/pulse_v2/plotters/__init__.py +17 -0
- qiskit/visualization/pulse_v2/plotters/base_plotter.py +53 -0
- qiskit/visualization/pulse_v2/plotters/matplotlib.py +201 -0
- qiskit/visualization/pulse_v2/stylesheet.py +312 -0
- qiskit/visualization/pulse_v2/types.py +242 -0
- qiskit/visualization/state_visualization.py +1518 -0
- qiskit/visualization/timeline/__init__.py +21 -0
- qiskit/visualization/timeline/core.py +480 -0
- qiskit/visualization/timeline/drawings.py +260 -0
- qiskit/visualization/timeline/generators.py +506 -0
- qiskit/visualization/timeline/interface.py +436 -0
- qiskit/visualization/timeline/layouts.py +115 -0
- qiskit/visualization/timeline/plotters/__init__.py +16 -0
- qiskit/visualization/timeline/plotters/base_plotter.py +58 -0
- qiskit/visualization/timeline/plotters/matplotlib.py +192 -0
- qiskit/visualization/timeline/stylesheet.py +301 -0
- qiskit/visualization/timeline/types.py +148 -0
- qiskit/visualization/transition_visualization.py +369 -0
- qiskit/visualization/utils.py +49 -0
- qiskit-1.3.0.dist-info/LICENSE.txt +203 -0
- qiskit-1.3.0.dist-info/METADATA +222 -0
- qiskit-1.3.0.dist-info/RECORD +836 -0
- qiskit-1.3.0.dist-info/WHEEL +5 -0
- qiskit-1.3.0.dist-info/entry_points.txt +76 -0
- qiskit-1.3.0.dist-info/top_level.txt +1 -0
@@ -0,0 +1,854 @@
|
|
1
|
+
# This code is part of Qiskit.
|
2
|
+
#
|
3
|
+
# (C) Copyright IBM 2022, 2023.
|
4
|
+
#
|
5
|
+
# This code is licensed under the Apache License, Version 2.0. You may
|
6
|
+
# obtain a copy of this license in the LICENSE.txt file in the root directory
|
7
|
+
# of this source tree or at http://www.apache.org/licenses/LICENSE-2.0.
|
8
|
+
#
|
9
|
+
# Any modifications or derivative works of this code must retain this
|
10
|
+
# copyright notice, and modified files need to carry a notice indicating
|
11
|
+
# that they have been altered from the originals.
|
12
|
+
|
13
|
+
"""
|
14
|
+
High-level-synthesis transpiler pass.
|
15
|
+
"""
|
16
|
+
|
17
|
+
from __future__ import annotations
|
18
|
+
|
19
|
+
import typing
|
20
|
+
from functools import partial
|
21
|
+
from collections.abc import Callable
|
22
|
+
|
23
|
+
import numpy as np
|
24
|
+
|
25
|
+
from qiskit.circuit.annotated_operation import Modifier
|
26
|
+
from qiskit.circuit.operation import Operation
|
27
|
+
from qiskit.circuit.instruction import Instruction
|
28
|
+
from qiskit.converters import circuit_to_dag, dag_to_circuit
|
29
|
+
from qiskit.transpiler.basepasses import TransformationPass
|
30
|
+
from qiskit.circuit.quantumcircuit import QuantumCircuit
|
31
|
+
from qiskit.circuit import ControlledGate, EquivalenceLibrary, equivalence, Qubit
|
32
|
+
from qiskit.transpiler.passes.utils import control_flow
|
33
|
+
from qiskit.transpiler.target import Target
|
34
|
+
from qiskit.transpiler.coupling import CouplingMap
|
35
|
+
from qiskit.dagcircuit.dagcircuit import DAGCircuit
|
36
|
+
from qiskit.transpiler.exceptions import TranspilerError
|
37
|
+
|
38
|
+
from qiskit.circuit.annotated_operation import (
|
39
|
+
AnnotatedOperation,
|
40
|
+
InverseModifier,
|
41
|
+
ControlModifier,
|
42
|
+
PowerModifier,
|
43
|
+
)
|
44
|
+
|
45
|
+
from qiskit._accelerate.high_level_synthesis import QubitTracker, QubitContext
|
46
|
+
from .plugin import HighLevelSynthesisPluginManager
|
47
|
+
|
48
|
+
if typing.TYPE_CHECKING:
|
49
|
+
from qiskit.dagcircuit import DAGOpNode
|
50
|
+
|
51
|
+
|
52
|
+
class HLSConfig:
|
53
|
+
"""The high-level-synthesis config allows to specify a list of "methods" used by
|
54
|
+
:class:`~.HighLevelSynthesis` transformation pass to synthesize different types
|
55
|
+
of higher-level objects.
|
56
|
+
|
57
|
+
A higher-level object is an object of type :class:`~.Operation` (e.g., :class:`.Clifford` or
|
58
|
+
:class:`.LinearFunction`). Each object is referred to by its :attr:`~.Operation.name` field
|
59
|
+
(e.g., ``"clifford"`` for :class:`.Clifford` objects), and the applicable synthesis methods are
|
60
|
+
tied to this name.
|
61
|
+
|
62
|
+
In the config, each method is specified in one of several ways:
|
63
|
+
|
64
|
+
1. a tuple consisting of the name of a known synthesis plugin and a dictionary providing
|
65
|
+
additional arguments for the algorithm.
|
66
|
+
2. a tuple consisting of an instance of :class:`.HighLevelSynthesisPlugin` and additional
|
67
|
+
arguments for the algorithm.
|
68
|
+
3. a single string of a known synthesis plugin
|
69
|
+
4. a single instance of :class:`.HighLevelSynthesisPlugin`.
|
70
|
+
|
71
|
+
The following example illustrates different ways how a config file can be created::
|
72
|
+
|
73
|
+
from qiskit.transpiler.passes.synthesis.high_level_synthesis import HLSConfig
|
74
|
+
from qiskit.transpiler.passes.synthesis.high_level_synthesis import ACGSynthesisPermutation
|
75
|
+
|
76
|
+
# All the ways to specify hls_config are equivalent
|
77
|
+
hls_config = HLSConfig(permutation=[("acg", {})])
|
78
|
+
hls_config = HLSConfig(permutation=["acg"])
|
79
|
+
hls_config = HLSConfig(permutation=[(ACGSynthesisPermutation(), {})])
|
80
|
+
hls_config = HLSConfig(permutation=[ACGSynthesisPermutation()])
|
81
|
+
|
82
|
+
The names of the synthesis plugins should be declared in ``entry-points`` table for
|
83
|
+
``qiskit.synthesis`` in ``pyproject.toml``, in the form
|
84
|
+
<higher-level-object-name>.<synthesis-method-name>.
|
85
|
+
|
86
|
+
The standard higher-level-objects are recommended to have a synthesis method
|
87
|
+
called "default", which would be called automatically when synthesizing these objects,
|
88
|
+
without having to explicitly set these methods in the config.
|
89
|
+
|
90
|
+
To avoid synthesizing a given higher-level-object, one can give it an empty list of methods.
|
91
|
+
|
92
|
+
For an explicit example of using such config files, refer to the documentation for
|
93
|
+
:class:`~.HighLevelSynthesis`.
|
94
|
+
|
95
|
+
For an overview of the complete process of using high-level synthesis, see
|
96
|
+
:ref:`using-high-level-synthesis-plugins`.
|
97
|
+
"""
|
98
|
+
|
99
|
+
def __init__(
|
100
|
+
self,
|
101
|
+
use_default_on_unspecified: bool = True,
|
102
|
+
plugin_selection: str = "sequential",
|
103
|
+
plugin_evaluation_fn: Callable[[QuantumCircuit], int] | None = None,
|
104
|
+
**kwargs,
|
105
|
+
):
|
106
|
+
"""Creates a high-level-synthesis config.
|
107
|
+
|
108
|
+
Args:
|
109
|
+
use_default_on_unspecified: if True, every higher-level-object without an
|
110
|
+
explicitly specified list of methods will be synthesized using the "default"
|
111
|
+
algorithm if it exists.
|
112
|
+
plugin_selection: if set to ``"sequential"`` (default), for every higher-level-object
|
113
|
+
the synthesis pass will consider the specified methods sequentially, stopping
|
114
|
+
at the first method that is able to synthesize the object. If set to ``"all"``,
|
115
|
+
all the specified methods will be considered, and the best synthesized circuit,
|
116
|
+
according to ``plugin_evaluation_fn`` will be chosen.
|
117
|
+
plugin_evaluation_fn: a callable that evaluates the quality of the synthesized
|
118
|
+
quantum circuit; a smaller value means a better circuit. If ``None``, the
|
119
|
+
quality of the circuit its size (i.e. the number of gates that it contains).
|
120
|
+
kwargs: a dictionary mapping higher-level-objects to lists of synthesis methods.
|
121
|
+
"""
|
122
|
+
self.use_default_on_unspecified = use_default_on_unspecified
|
123
|
+
self.plugin_selection = plugin_selection
|
124
|
+
self.plugin_evaluation_fn = (
|
125
|
+
plugin_evaluation_fn if plugin_evaluation_fn is not None else lambda qc: qc.size()
|
126
|
+
)
|
127
|
+
self.methods = {}
|
128
|
+
|
129
|
+
for key, value in kwargs.items():
|
130
|
+
self.set_methods(key, value)
|
131
|
+
|
132
|
+
def set_methods(self, hls_name, hls_methods):
|
133
|
+
"""Sets the list of synthesis methods for a given higher-level-object. This overwrites
|
134
|
+
the lists of methods if also set previously."""
|
135
|
+
self.methods[hls_name] = hls_methods
|
136
|
+
|
137
|
+
|
138
|
+
class HighLevelSynthesis(TransformationPass):
|
139
|
+
r"""Synthesize higher-level objects and unroll custom definitions.
|
140
|
+
|
141
|
+
The input to this pass is a DAG that may contain higher-level objects,
|
142
|
+
including abstract mathematical objects (e.g., objects of type :class:`.LinearFunction`),
|
143
|
+
annotated operations (objects of type :class:`.AnnotatedOperation`), and
|
144
|
+
custom gates.
|
145
|
+
|
146
|
+
In the most common use-case when either ``basis_gates`` or ``target`` is specified,
|
147
|
+
all higher-level objects are synthesized, so the output is a :class:`.DAGCircuit`
|
148
|
+
without such objects.
|
149
|
+
More precisely, every gate in the output DAG is either directly supported by the target,
|
150
|
+
or is in ``equivalence_library``.
|
151
|
+
|
152
|
+
The abstract mathematical objects are synthesized using synthesis plugins, applying
|
153
|
+
synthesis methods specified in the high-level-synthesis config (refer to the documentation
|
154
|
+
for :class:`~.HLSConfig`).
|
155
|
+
|
156
|
+
As an example, let us assume that ``op_a`` and ``op_b`` are names of two higher-level objects,
|
157
|
+
that ``op_a``-objects have two synthesis methods ``default`` which does require any additional
|
158
|
+
parameters and ``other`` with two optional integer parameters ``option_1`` and ``option_2``,
|
159
|
+
that ``op_b``-objects have a single synthesis method ``default``, and ``qc`` is a quantum
|
160
|
+
circuit containing ``op_a`` and ``op_b`` objects. The following code snippet::
|
161
|
+
|
162
|
+
hls_config = HLSConfig(op_b=[("other", {"option_1": 7, "option_2": 4})])
|
163
|
+
pm = PassManager([HighLevelSynthesis(hls_config=hls_config)])
|
164
|
+
transpiled_qc = pm.run(qc)
|
165
|
+
|
166
|
+
shows how to run the alternative synthesis method ``other`` for ``op_b``-objects, while using the
|
167
|
+
``default`` methods for all other high-level objects, including ``op_a``-objects.
|
168
|
+
|
169
|
+
The annotated operations (consisting of a base operation and a list of inverse, control and power
|
170
|
+
modifiers) are synthesizing recursively, first synthesizing the base operation, and then applying
|
171
|
+
synthesis methods for creating inverted, controlled, or powered versions of that).
|
172
|
+
|
173
|
+
The custom gates are synthesized by recursively unrolling their definitions, until every gate
|
174
|
+
is either supported by the target or is in the equivalence library.
|
175
|
+
|
176
|
+
When neither ``basis_gates`` nor ``target`` is specified, the pass synthesizes only the top-level
|
177
|
+
abstract mathematical objects and annotated operations, without descending into the gate
|
178
|
+
``definitions``. This is consistent with the older behavior of the pass, allowing to synthesize
|
179
|
+
some higher-level objects using plugins and leaving the other gates untouched.
|
180
|
+
|
181
|
+
The high-level-synthesis passes information about available auxiliary qubits, and whether their
|
182
|
+
state is clean (defined as :math:`|0\rangle`) or dirty (unknown state) to the synthesis routine
|
183
|
+
via the respective arguments ``"num_clean_ancillas"`` and ``"num_dirty_ancillas"``.
|
184
|
+
If ``qubits_initially_zero`` is ``True`` (default), the qubits are assumed to be in the
|
185
|
+
:math:`|0\rangle` state. When appending a synthesized block using auxiliary qubits onto the
|
186
|
+
circuit, we first use the clean auxiliary qubits.
|
187
|
+
|
188
|
+
.. note::
|
189
|
+
|
190
|
+
Synthesis methods are assumed to maintain the state of the auxiliary qubits.
|
191
|
+
Concretely this means that clean auxiliary qubits must still be in the :math:`|0\rangle`
|
192
|
+
state after the synthesized block, while dirty auxiliary qubits are re-used only
|
193
|
+
as dirty qubits.
|
194
|
+
|
195
|
+
"""
|
196
|
+
|
197
|
+
def __init__(
|
198
|
+
self,
|
199
|
+
hls_config: HLSConfig | None = None,
|
200
|
+
coupling_map: CouplingMap | None = None,
|
201
|
+
target: Target | None = None,
|
202
|
+
use_qubit_indices: bool = False,
|
203
|
+
equivalence_library: EquivalenceLibrary | None = None,
|
204
|
+
basis_gates: list[str] | None = None,
|
205
|
+
min_qubits: int = 0,
|
206
|
+
qubits_initially_zero: bool = True,
|
207
|
+
):
|
208
|
+
r"""
|
209
|
+
HighLevelSynthesis initializer.
|
210
|
+
|
211
|
+
Args:
|
212
|
+
hls_config: Optional, the high-level-synthesis config that specifies synthesis methods
|
213
|
+
and parameters for various high-level-objects in the circuit. If it is not specified,
|
214
|
+
the default synthesis methods and parameters will be used.
|
215
|
+
coupling_map: Optional, directed graph represented as a coupling map.
|
216
|
+
target: Optional, the backend target to use for this pass. If it is specified,
|
217
|
+
it will be used instead of the coupling map.
|
218
|
+
use_qubit_indices: a flag indicating whether this synthesis pass is running before or after
|
219
|
+
the layout is set, that is, whether the qubit indices of higher-level-objects correspond
|
220
|
+
to qubit indices on the target backend.
|
221
|
+
equivalence_library: The equivalence library used (instructions in this library will not
|
222
|
+
be unrolled by this pass).
|
223
|
+
basis_gates: Optional, target basis names to unroll to, e.g. `['u3', 'cx']`.
|
224
|
+
Ignored if ``target`` is also specified.
|
225
|
+
min_qubits: The minimum number of qubits for operations in the input
|
226
|
+
dag to translate.
|
227
|
+
qubits_initially_zero: Indicates whether the qubits are initially in the state
|
228
|
+
:math:`|0\rangle`. This allows the high-level-synthesis to use clean auxiliary qubits
|
229
|
+
(i.e. in the zero state) to synthesize an operation.
|
230
|
+
"""
|
231
|
+
super().__init__()
|
232
|
+
|
233
|
+
if hls_config is not None:
|
234
|
+
self.hls_config = hls_config
|
235
|
+
else:
|
236
|
+
# When the config file is not provided, we will use the "default" method
|
237
|
+
# to synthesize Operations (when available).
|
238
|
+
self.hls_config = HLSConfig(True)
|
239
|
+
|
240
|
+
self.hls_plugin_manager = HighLevelSynthesisPluginManager()
|
241
|
+
self._coupling_map = coupling_map
|
242
|
+
self._target = target
|
243
|
+
self._use_qubit_indices = use_qubit_indices
|
244
|
+
self.qubits_initially_zero = qubits_initially_zero
|
245
|
+
if target is not None:
|
246
|
+
self._coupling_map = self._target.build_coupling_map()
|
247
|
+
self._equiv_lib = equivalence_library
|
248
|
+
self._basis_gates = basis_gates
|
249
|
+
self._min_qubits = min_qubits
|
250
|
+
|
251
|
+
self._top_level_only = self._basis_gates is None and self._target is None
|
252
|
+
|
253
|
+
# include path for when target exists but target.num_qubits is None (BasicSimulator)
|
254
|
+
if not self._top_level_only and (self._target is None or self._target.num_qubits is None):
|
255
|
+
basic_insts = {"measure", "reset", "barrier", "snapshot", "delay", "store"}
|
256
|
+
self._device_insts = basic_insts | set(self._basis_gates)
|
257
|
+
else:
|
258
|
+
self._device_insts = set()
|
259
|
+
|
260
|
+
def run(self, dag: DAGCircuit) -> DAGCircuit:
|
261
|
+
"""Run the HighLevelSynthesis pass on `dag`.
|
262
|
+
|
263
|
+
Args:
|
264
|
+
dag: input dag.
|
265
|
+
|
266
|
+
Returns:
|
267
|
+
Output dag with higher-level operations synthesized.
|
268
|
+
|
269
|
+
Raises:
|
270
|
+
TranspilerError: when the transpiler is unable to synthesize the given DAG
|
271
|
+
(for instance, when the specified synthesis method is not available).
|
272
|
+
"""
|
273
|
+
qubits = tuple(dag.find_bit(q).index for q in dag.qubits)
|
274
|
+
context = QubitContext(list(range(len(dag.qubits))))
|
275
|
+
tracker = QubitTracker(num_qubits=dag.num_qubits())
|
276
|
+
if self.qubits_initially_zero:
|
277
|
+
tracker.set_clean(context.to_globals(qubits))
|
278
|
+
|
279
|
+
out_dag = self._run(dag, tracker, context, use_ancillas=True, top_level=True)
|
280
|
+
return out_dag
|
281
|
+
|
282
|
+
def _run(
|
283
|
+
self,
|
284
|
+
dag: DAGCircuit,
|
285
|
+
tracker: QubitTracker,
|
286
|
+
context: QubitContext,
|
287
|
+
use_ancillas: bool,
|
288
|
+
top_level: bool,
|
289
|
+
) -> DAGCircuit:
|
290
|
+
"""
|
291
|
+
The main recursive function that synthesizes a DAGCircuit.
|
292
|
+
|
293
|
+
Input:
|
294
|
+
dag: the DAG to be synthesized.
|
295
|
+
tracker: the global tracker, tracking the state of original qubits.
|
296
|
+
context: the correspondence between the dag's qubits and the global qubits.
|
297
|
+
use_ancillas: if True, synthesis algorithms are allowed to use ancillas.
|
298
|
+
top_level: specifies if this is the top-level of the recursion.
|
299
|
+
|
300
|
+
The function returns the synthesized DAG.
|
301
|
+
|
302
|
+
Note that by using the auxiliary qubits to synthesize operations present in the input DAG,
|
303
|
+
the synthesized DAG may be defined over more qubits than the input DAG. In this case,
|
304
|
+
the function update in-place the global qubits tracker and extends the local-to-global
|
305
|
+
context.
|
306
|
+
"""
|
307
|
+
|
308
|
+
if dag.num_qubits() != context.num_qubits():
|
309
|
+
raise TranspilerError("HighLevelSynthesis internal error.")
|
310
|
+
|
311
|
+
# STEP 1: Check if HighLevelSynthesis can be skipped altogether. This is only
|
312
|
+
# done at the top-level since this does not update the global qubits tracker.
|
313
|
+
if top_level:
|
314
|
+
for node in dag.op_nodes():
|
315
|
+
qubits = tuple(dag.find_bit(q).index for q in node.qargs)
|
316
|
+
if not self._definitely_skip_node(node, qubits, dag):
|
317
|
+
break
|
318
|
+
else:
|
319
|
+
# The for-loop terminates without reaching the break statement
|
320
|
+
if dag.num_qubits() != context.num_qubits():
|
321
|
+
raise TranspilerError("HighLevelSynthesis internal error.")
|
322
|
+
return dag
|
323
|
+
|
324
|
+
# STEP 2: Analyze the nodes in the DAG. For each node in the DAG that needs
|
325
|
+
# to be synthesized, we recursively synthesize it and store the result. For
|
326
|
+
# instance, the result of synthesizing a custom gate is a DAGCircuit corresponding
|
327
|
+
# to the (recursively synthesized) gate's definition. When the result is a
|
328
|
+
# DAG, we also store its context (the mapping of its qubits to global qubits).
|
329
|
+
# In addition, we keep track of the qubit states using the (global) qubits tracker.
|
330
|
+
#
|
331
|
+
# Note: This is a first version of a potentially more elaborate approach to find
|
332
|
+
# good operation/ancilla allocations. The current approach is greedy and just gives
|
333
|
+
# all available ancilla qubits to the current operation ("the-first-takes-all" approach).
|
334
|
+
# It does not distribute ancilla qubits between different operations present in the DAG.
|
335
|
+
synthesized_nodes = {}
|
336
|
+
|
337
|
+
for node in dag.topological_op_nodes():
|
338
|
+
qubits = tuple(dag.find_bit(q).index for q in node.qargs)
|
339
|
+
processed = False
|
340
|
+
synthesized = None
|
341
|
+
synthesized_context = None
|
342
|
+
|
343
|
+
# Start by handling special operations. Other cases can also be
|
344
|
+
# considered: swaps, automatically simplifying control gate (e.g. if
|
345
|
+
# a control is 0).
|
346
|
+
if node.op.name in ["id", "delay", "barrier"]:
|
347
|
+
# tracker not updated, these are no-ops
|
348
|
+
processed = True
|
349
|
+
|
350
|
+
elif node.op.name == "reset":
|
351
|
+
# reset qubits to 0
|
352
|
+
tracker.set_clean(context.to_globals(qubits))
|
353
|
+
processed = True
|
354
|
+
|
355
|
+
# check if synthesis for the operation can be skipped
|
356
|
+
elif self._definitely_skip_node(node, qubits, dag):
|
357
|
+
tracker.set_dirty(context.to_globals(qubits))
|
358
|
+
|
359
|
+
# next check control flow
|
360
|
+
elif node.is_control_flow():
|
361
|
+
inner_context = context.restrict(qubits)
|
362
|
+
synthesized = control_flow.map_blocks(
|
363
|
+
partial(
|
364
|
+
self._run,
|
365
|
+
tracker=tracker,
|
366
|
+
context=inner_context,
|
367
|
+
use_ancillas=False,
|
368
|
+
top_level=False,
|
369
|
+
),
|
370
|
+
node.op,
|
371
|
+
)
|
372
|
+
|
373
|
+
# now we are free to synthesize
|
374
|
+
else:
|
375
|
+
# This returns the synthesized operation and its context (when the result is
|
376
|
+
# a DAG, it's the correspondence between its qubits and the global qubits).
|
377
|
+
# Also note that the DAG may use auxiliary qubits. The qubits tracker and the
|
378
|
+
# current DAG's context are updated in-place.
|
379
|
+
synthesized, synthesized_context = self._synthesize_operation(
|
380
|
+
node.op, qubits, tracker, context, use_ancillas=use_ancillas
|
381
|
+
)
|
382
|
+
|
383
|
+
# If the synthesis changed the operation (i.e. it is not None), store the result.
|
384
|
+
if synthesized is not None:
|
385
|
+
synthesized_nodes[node._node_id] = (synthesized, synthesized_context)
|
386
|
+
|
387
|
+
# If the synthesis did not change anything, just update the qubit tracker.
|
388
|
+
elif not processed:
|
389
|
+
tracker.set_dirty(context.to_globals(qubits))
|
390
|
+
|
391
|
+
# We did not change anything just return the input.
|
392
|
+
if len(synthesized_nodes) == 0:
|
393
|
+
if dag.num_qubits() != context.num_qubits():
|
394
|
+
raise TranspilerError("HighLevelSynthesis internal error.")
|
395
|
+
return dag
|
396
|
+
|
397
|
+
# STEP 3. We rebuild the DAG with new operations. Note that we could also
|
398
|
+
# check if no operation changed in size and substitute in-place, but rebuilding is
|
399
|
+
# generally as fast or faster, unless very few operations are changed.
|
400
|
+
out = dag.copy_empty_like()
|
401
|
+
num_additional_qubits = context.num_qubits() - out.num_qubits()
|
402
|
+
|
403
|
+
if num_additional_qubits > 0:
|
404
|
+
out.add_qubits([Qubit() for _ in range(num_additional_qubits)])
|
405
|
+
|
406
|
+
index_to_qubit = dict(enumerate(out.qubits))
|
407
|
+
outer_to_local = context.to_local_mapping()
|
408
|
+
|
409
|
+
for node in dag.topological_op_nodes():
|
410
|
+
|
411
|
+
if op_tuple := synthesized_nodes.get(node._node_id, None):
|
412
|
+
op, op_context = op_tuple
|
413
|
+
|
414
|
+
if isinstance(op, Operation):
|
415
|
+
out.apply_operation_back(op, node.qargs, node.cargs)
|
416
|
+
continue
|
417
|
+
|
418
|
+
if isinstance(op, QuantumCircuit):
|
419
|
+
op = circuit_to_dag(op, copy_operations=False)
|
420
|
+
|
421
|
+
inner_to_global = op_context.to_global_mapping()
|
422
|
+
if isinstance(op, DAGCircuit):
|
423
|
+
qubit_map = {
|
424
|
+
q: index_to_qubit[outer_to_local[inner_to_global[i]]]
|
425
|
+
for (i, q) in enumerate(op.qubits)
|
426
|
+
}
|
427
|
+
clbit_map = dict(zip(op.clbits, node.cargs))
|
428
|
+
|
429
|
+
for sub_node in op.op_nodes():
|
430
|
+
out.apply_operation_back(
|
431
|
+
sub_node.op,
|
432
|
+
tuple(qubit_map[qarg] for qarg in sub_node.qargs),
|
433
|
+
tuple(clbit_map[carg] for carg in sub_node.cargs),
|
434
|
+
)
|
435
|
+
out.global_phase += op.global_phase
|
436
|
+
|
437
|
+
else:
|
438
|
+
raise TranspilerError(f"Unexpected synthesized type: {type(op)}")
|
439
|
+
else:
|
440
|
+
out.apply_operation_back(node.op, node.qargs, node.cargs, check=False)
|
441
|
+
|
442
|
+
if out.num_qubits() != context.num_qubits():
|
443
|
+
raise TranspilerError("HighLevelSynthesis internal error.")
|
444
|
+
|
445
|
+
return out
|
446
|
+
|
447
|
+
def _synthesize_operation(
|
448
|
+
self,
|
449
|
+
operation: Operation,
|
450
|
+
qubits: tuple[int],
|
451
|
+
tracker: QubitTracker,
|
452
|
+
context: QubitContext,
|
453
|
+
use_ancillas: bool,
|
454
|
+
) -> tuple[QuantumCircuit | Operation | DAGCircuit | None, QubitContext | None]:
|
455
|
+
"""
|
456
|
+
Synthesizes an operation. The function receives the qubits on which the operation
|
457
|
+
is defined in the current DAG, the correspondence between the qubits of the current
|
458
|
+
DAG and the global qubits and the global qubits tracker. The function returns the
|
459
|
+
result of synthesizing the operation. The value of `None` means that the operation
|
460
|
+
should remain as it is. When it's a circuit, we also return the context, i.e. the
|
461
|
+
correspondence of its local qubits and the global qubits. The function changes
|
462
|
+
in-place the tracker (state of the global qubits), the qubits (when the synthesized
|
463
|
+
operation is defined over additional ancilla qubits), and the context (to keep track
|
464
|
+
of where these ancilla qubits maps to).
|
465
|
+
"""
|
466
|
+
|
467
|
+
synthesized_context = None
|
468
|
+
|
469
|
+
# Try to synthesize the operation. We'll go through the following options:
|
470
|
+
# (1) Annotations: if the operator is annotated, synthesize the base operation
|
471
|
+
# and then apply the modifiers. Returns a circuit (e.g. applying a power)
|
472
|
+
# or operation (e.g adding control on an X gate).
|
473
|
+
# (2) High-level objects: try running the battery of high-level synthesis plugins (e.g.
|
474
|
+
# if the operation is a Clifford). Returns a circuit.
|
475
|
+
# (3) Unrolling custom definitions: try defining the operation if it is not yet
|
476
|
+
# in the set of supported instructions. Returns a circuit.
|
477
|
+
#
|
478
|
+
# If any of the above were triggered, we will recurse and go again through these steps
|
479
|
+
# until no further change occurred. At this point, we convert circuits to DAGs (the final
|
480
|
+
# possible return type). If there was no change, we just return ``None``.
|
481
|
+
num_original_qubits = len(qubits)
|
482
|
+
qubits = list(qubits)
|
483
|
+
|
484
|
+
synthesized = None
|
485
|
+
|
486
|
+
# Try synthesizing via AnnotatedOperation. This is faster than an isinstance check
|
487
|
+
# but a bit less safe since someone could create operations with a ``modifiers`` attribute.
|
488
|
+
if len(modifiers := getattr(operation, "modifiers", [])) > 0:
|
489
|
+
# Note: the base operation must be synthesized without using potential control qubits
|
490
|
+
# used in the modifiers.
|
491
|
+
num_ctrl = sum(
|
492
|
+
mod.num_ctrl_qubits for mod in modifiers if isinstance(mod, ControlModifier)
|
493
|
+
)
|
494
|
+
baseop_qubits = qubits[num_ctrl:] # reminder: control qubits are the first ones
|
495
|
+
|
496
|
+
# get qubits of base operation
|
497
|
+
control_qubits = qubits[0:num_ctrl]
|
498
|
+
|
499
|
+
# Do not allow access to control qubits
|
500
|
+
tracker.disable(context.to_globals(control_qubits))
|
501
|
+
synthesized_base_op, _ = self._synthesize_operation(
|
502
|
+
operation.base_op,
|
503
|
+
baseop_qubits,
|
504
|
+
tracker,
|
505
|
+
context,
|
506
|
+
use_ancillas=use_ancillas,
|
507
|
+
)
|
508
|
+
|
509
|
+
if synthesized_base_op is None:
|
510
|
+
synthesized_base_op = operation.base_op
|
511
|
+
elif isinstance(synthesized_base_op, DAGCircuit):
|
512
|
+
synthesized_base_op = dag_to_circuit(synthesized_base_op)
|
513
|
+
|
514
|
+
# Handle the case that synthesizing the base operation introduced
|
515
|
+
# additional qubits (e.g. the base operation is a circuit that includes
|
516
|
+
# an MCX gate).
|
517
|
+
if synthesized_base_op.num_qubits > len(baseop_qubits):
|
518
|
+
global_aux_qubits = tracker.borrow(
|
519
|
+
synthesized_base_op.num_qubits - len(baseop_qubits),
|
520
|
+
context.to_globals(baseop_qubits),
|
521
|
+
)
|
522
|
+
global_to_local = context.to_local_mapping()
|
523
|
+
for aq in global_aux_qubits:
|
524
|
+
if aq in global_to_local:
|
525
|
+
qubits.append(global_to_local[aq])
|
526
|
+
else:
|
527
|
+
new_local_qubit = context.add_qubit(aq)
|
528
|
+
qubits.append(new_local_qubit)
|
529
|
+
# Restore access to control qubits.
|
530
|
+
tracker.enable(context.to_globals(control_qubits))
|
531
|
+
|
532
|
+
# This step currently does not introduce ancilla qubits.
|
533
|
+
synthesized = self._apply_annotations(synthesized_base_op, operation.modifiers)
|
534
|
+
|
535
|
+
# If it was no AnnotatedOperation, try synthesizing via HLS or by unrolling.
|
536
|
+
else:
|
537
|
+
# Try synthesis via HLS -- which will return ``None`` if unsuccessful.
|
538
|
+
indices = qubits if self._use_qubit_indices else None
|
539
|
+
if len(hls_methods := self._methods_to_try(operation.name)) > 0:
|
540
|
+
if use_ancillas:
|
541
|
+
num_clean_available = tracker.num_clean(context.to_globals(qubits))
|
542
|
+
num_dirty_available = tracker.num_dirty(context.to_globals(qubits))
|
543
|
+
else:
|
544
|
+
num_clean_available = 0
|
545
|
+
num_dirty_available = 0
|
546
|
+
synthesized = self._synthesize_op_using_plugins(
|
547
|
+
hls_methods,
|
548
|
+
operation,
|
549
|
+
indices,
|
550
|
+
num_clean_available,
|
551
|
+
num_dirty_available,
|
552
|
+
)
|
553
|
+
|
554
|
+
# It may happen that the plugin synthesis method uses clean/dirty ancilla qubits
|
555
|
+
if (synthesized is not None) and (synthesized.num_qubits > len(qubits)):
|
556
|
+
# need to borrow more qubits from tracker
|
557
|
+
global_aux_qubits = tracker.borrow(
|
558
|
+
synthesized.num_qubits - len(qubits), context.to_globals(qubits)
|
559
|
+
)
|
560
|
+
global_to_local = context.to_local_mapping()
|
561
|
+
|
562
|
+
for aq in global_aux_qubits:
|
563
|
+
if aq in global_to_local:
|
564
|
+
qubits.append(global_to_local[aq])
|
565
|
+
else:
|
566
|
+
new_local_qubit = context.add_qubit(aq)
|
567
|
+
qubits.append(new_local_qubit)
|
568
|
+
|
569
|
+
# If HLS did not apply, or was unsuccessful, try unrolling custom definitions.
|
570
|
+
if synthesized is None and not self._top_level_only:
|
571
|
+
synthesized = self._get_custom_definition(operation, indices)
|
572
|
+
|
573
|
+
if synthesized is None:
|
574
|
+
# if we didn't synthesize, there was nothing to unroll
|
575
|
+
# updating the tracker will be handled upstream
|
576
|
+
pass
|
577
|
+
|
578
|
+
# if it has been synthesized, recurse and finally store the decomposition
|
579
|
+
elif isinstance(synthesized, Operation):
|
580
|
+
resynthesized, resynthesized_context = self._synthesize_operation(
|
581
|
+
synthesized, qubits, tracker, context, use_ancillas=use_ancillas
|
582
|
+
)
|
583
|
+
|
584
|
+
if resynthesized is not None:
|
585
|
+
synthesized = resynthesized
|
586
|
+
else:
|
587
|
+
tracker.set_dirty(context.to_globals(qubits))
|
588
|
+
if isinstance(resynthesized, DAGCircuit):
|
589
|
+
synthesized_context = resynthesized_context
|
590
|
+
|
591
|
+
elif isinstance(synthesized, QuantumCircuit):
|
592
|
+
# Synthesized is a quantum circuit which we want to process recursively.
|
593
|
+
# For example, it's the definition circuit of a custom gate
|
594
|
+
# or a circuit obtained by calling a synthesis method on a high-level-object.
|
595
|
+
# In the second case, synthesized may have more qubits than the original node.
|
596
|
+
|
597
|
+
as_dag = circuit_to_dag(synthesized, copy_operations=False)
|
598
|
+
inner_context = context.restrict(qubits)
|
599
|
+
|
600
|
+
if as_dag.num_qubits() != inner_context.num_qubits():
|
601
|
+
raise TranspilerError("HighLevelSynthesis internal error.")
|
602
|
+
|
603
|
+
# We save the current state of the tracker to be able to return the ancilla
|
604
|
+
# qubits to the current positions. Note that at this point we do not know
|
605
|
+
# which ancilla qubits will be allocated.
|
606
|
+
saved_tracker = tracker.copy()
|
607
|
+
synthesized = self._run(
|
608
|
+
as_dag, tracker, inner_context, use_ancillas=use_ancillas, top_level=False
|
609
|
+
)
|
610
|
+
synthesized_context = inner_context
|
611
|
+
|
612
|
+
if (synthesized is not None) and (synthesized.num_qubits() > len(qubits)):
|
613
|
+
# need to borrow more qubits from tracker
|
614
|
+
global_aux_qubits = tracker.borrow(
|
615
|
+
synthesized.num_qubits() - len(qubits), context.to_globals(qubits)
|
616
|
+
)
|
617
|
+
global_to_local = context.to_local_mapping()
|
618
|
+
|
619
|
+
for aq in global_aux_qubits:
|
620
|
+
if aq in global_to_local:
|
621
|
+
qubits.append(global_to_local[aq])
|
622
|
+
else:
|
623
|
+
new_local_qubit = context.add_qubit(aq)
|
624
|
+
qubits.append(new_local_qubit)
|
625
|
+
|
626
|
+
if len(qubits) > num_original_qubits:
|
627
|
+
tracker.replace_state(
|
628
|
+
saved_tracker, context.to_globals(qubits[num_original_qubits:])
|
629
|
+
)
|
630
|
+
|
631
|
+
else:
|
632
|
+
raise TranspilerError(f"Unexpected synthesized type: {type(synthesized)}")
|
633
|
+
|
634
|
+
if isinstance(synthesized, DAGCircuit) and synthesized_context is None:
|
635
|
+
raise TranspilerError("HighLevelSynthesis internal error.")
|
636
|
+
|
637
|
+
return synthesized, synthesized_context
|
638
|
+
|
639
|
+
def _get_custom_definition(
|
640
|
+
self, inst: Instruction, qubits: list[int] | None
|
641
|
+
) -> QuantumCircuit | None:
|
642
|
+
# check if the operation is already supported natively
|
643
|
+
if not (isinstance(inst, ControlledGate) and inst._open_ctrl):
|
644
|
+
# include path for when target exists but target.num_qubits is None (BasicSimulator)
|
645
|
+
inst_supported = self._instruction_supported(inst.name, qubits)
|
646
|
+
if inst_supported or (self._equiv_lib is not None and self._equiv_lib.has_entry(inst)):
|
647
|
+
return None # we support this operation already
|
648
|
+
|
649
|
+
# if not, try to get the definition
|
650
|
+
try:
|
651
|
+
definition = inst.definition
|
652
|
+
except (TypeError, AttributeError) as err:
|
653
|
+
raise TranspilerError(f"HighLevelSynthesis was unable to define {inst.name}.") from err
|
654
|
+
|
655
|
+
if definition is None:
|
656
|
+
raise TranspilerError(f"HighLevelSynthesis was unable to synthesize {inst}.")
|
657
|
+
|
658
|
+
return definition
|
659
|
+
|
660
|
+
def _methods_to_try(self, name: str):
|
661
|
+
"""Get a sequence of methods to try for a given op name."""
|
662
|
+
if (methods := self.hls_config.methods.get(name)) is not None:
|
663
|
+
# the operation's name appears in the user-provided config,
|
664
|
+
# we use the list of methods provided by the user
|
665
|
+
return methods
|
666
|
+
if (
|
667
|
+
self.hls_config.use_default_on_unspecified
|
668
|
+
and "default" in self.hls_plugin_manager.method_names(name)
|
669
|
+
):
|
670
|
+
# the operation's name does not appear in the user-specified config,
|
671
|
+
# we use the "default" method when instructed to do so and the "default"
|
672
|
+
# method is available
|
673
|
+
return ["default"]
|
674
|
+
return []
|
675
|
+
|
676
|
+
def _synthesize_op_using_plugins(
|
677
|
+
self,
|
678
|
+
hls_methods: list,
|
679
|
+
op: Operation,
|
680
|
+
qubits: list[int] | None,
|
681
|
+
num_clean_ancillas: int = 0,
|
682
|
+
num_dirty_ancillas: int = 0,
|
683
|
+
) -> QuantumCircuit | None:
|
684
|
+
"""
|
685
|
+
Attempts to synthesize op using plugin mechanism.
|
686
|
+
|
687
|
+
The arguments ``num_clean_ancillas`` and ``num_dirty_ancillas`` specify
|
688
|
+
the number of clean and dirty qubits available to synthesize the given
|
689
|
+
operation. A synthesis method does not need to use these additional qubits.
|
690
|
+
|
691
|
+
Returns either the synthesized circuit or None (which may occur
|
692
|
+
when no synthesis methods is available or specified, or when there is
|
693
|
+
an insufficient number of auxiliary qubits).
|
694
|
+
"""
|
695
|
+
hls_plugin_manager = self.hls_plugin_manager
|
696
|
+
|
697
|
+
best_decomposition = None
|
698
|
+
best_score = np.inf
|
699
|
+
|
700
|
+
for method in hls_methods:
|
701
|
+
# There are two ways to specify a synthesis method. The more explicit
|
702
|
+
# way is to specify it as a tuple consisting of a synthesis algorithm and a
|
703
|
+
# list of additional arguments, e.g.,
|
704
|
+
# ("kms", {"all_mats": 1, "max_paths": 100, "orig_circuit": 0}), or
|
705
|
+
# ("pmh", {}).
|
706
|
+
# When the list of additional arguments is empty, one can also specify
|
707
|
+
# just the synthesis algorithm, e.g.,
|
708
|
+
# "pmh".
|
709
|
+
if isinstance(method, tuple):
|
710
|
+
plugin_specifier, plugin_args = method
|
711
|
+
else:
|
712
|
+
plugin_specifier = method
|
713
|
+
plugin_args = {}
|
714
|
+
|
715
|
+
# There are two ways to specify a synthesis algorithm being run,
|
716
|
+
# either by name, e.g. "kms" (which then should be specified in entry_points),
|
717
|
+
# or directly as a class inherited from HighLevelSynthesisPlugin (which then
|
718
|
+
# does not need to be specified in entry_points).
|
719
|
+
if isinstance(plugin_specifier, str):
|
720
|
+
if plugin_specifier not in hls_plugin_manager.method_names(op.name):
|
721
|
+
raise TranspilerError(
|
722
|
+
f"Specified method: {plugin_specifier} not found in available "
|
723
|
+
f"plugins for {op.name}"
|
724
|
+
)
|
725
|
+
plugin_method = hls_plugin_manager.method(op.name, plugin_specifier)
|
726
|
+
else:
|
727
|
+
plugin_method = plugin_specifier
|
728
|
+
|
729
|
+
# Set the number of available clean and dirty auxiliary qubits via plugin args.
|
730
|
+
plugin_args["num_clean_ancillas"] = num_clean_ancillas
|
731
|
+
plugin_args["num_dirty_ancillas"] = num_dirty_ancillas
|
732
|
+
|
733
|
+
decomposition = plugin_method.run(
|
734
|
+
op,
|
735
|
+
coupling_map=self._coupling_map,
|
736
|
+
target=self._target,
|
737
|
+
qubits=qubits,
|
738
|
+
**plugin_args,
|
739
|
+
)
|
740
|
+
|
741
|
+
# The synthesis methods that are not suited for the given higher-level-object
|
742
|
+
# will return None.
|
743
|
+
if decomposition is not None:
|
744
|
+
if self.hls_config.plugin_selection == "sequential":
|
745
|
+
# In the "sequential" mode the first successful decomposition is
|
746
|
+
# returned.
|
747
|
+
best_decomposition = decomposition
|
748
|
+
break
|
749
|
+
|
750
|
+
# In the "run everything" mode we update the best decomposition
|
751
|
+
# discovered
|
752
|
+
current_score = self.hls_config.plugin_evaluation_fn(decomposition)
|
753
|
+
if current_score < best_score:
|
754
|
+
best_decomposition = decomposition
|
755
|
+
best_score = current_score
|
756
|
+
|
757
|
+
return best_decomposition
|
758
|
+
|
759
|
+
def _apply_annotations(
|
760
|
+
self, synthesized: Operation | QuantumCircuit, modifiers: list[Modifier]
|
761
|
+
) -> QuantumCircuit:
|
762
|
+
"""
|
763
|
+
Recursively synthesizes annotated operations.
|
764
|
+
Returns either the synthesized operation or None (which occurs when the operation
|
765
|
+
is not an annotated operation).
|
766
|
+
"""
|
767
|
+
for modifier in modifiers:
|
768
|
+
if isinstance(modifier, InverseModifier):
|
769
|
+
# Both QuantumCircuit and Gate have inverse method
|
770
|
+
synthesized = synthesized.inverse()
|
771
|
+
|
772
|
+
elif isinstance(modifier, ControlModifier):
|
773
|
+
# Both QuantumCircuit and Gate have control method, however for circuits
|
774
|
+
# it is more efficient to avoid constructing the controlled quantum circuit.
|
775
|
+
if isinstance(synthesized, QuantumCircuit):
|
776
|
+
synthesized = synthesized.to_gate()
|
777
|
+
|
778
|
+
synthesized = synthesized.control(
|
779
|
+
num_ctrl_qubits=modifier.num_ctrl_qubits,
|
780
|
+
label=None,
|
781
|
+
ctrl_state=modifier.ctrl_state,
|
782
|
+
annotated=False,
|
783
|
+
)
|
784
|
+
|
785
|
+
if isinstance(synthesized, AnnotatedOperation):
|
786
|
+
raise TranspilerError(
|
787
|
+
"HighLevelSynthesis failed to synthesize the control modifier."
|
788
|
+
)
|
789
|
+
|
790
|
+
elif isinstance(modifier, PowerModifier):
|
791
|
+
# QuantumCircuit has power method, and Gate needs to be converted
|
792
|
+
# to a quantum circuit.
|
793
|
+
if not isinstance(synthesized, QuantumCircuit):
|
794
|
+
synthesized = _instruction_to_circuit(synthesized)
|
795
|
+
|
796
|
+
synthesized = synthesized.power(modifier.power)
|
797
|
+
|
798
|
+
else:
|
799
|
+
raise TranspilerError(f"Unknown modifier {modifier}.")
|
800
|
+
|
801
|
+
return synthesized
|
802
|
+
|
803
|
+
def _definitely_skip_node(
|
804
|
+
self, node: DAGOpNode, qubits: tuple[int] | None, dag: DAGCircuit
|
805
|
+
) -> bool:
|
806
|
+
"""Fast-path determination of whether a node can certainly be skipped (i.e. nothing will
|
807
|
+
attempt to synthesise it) without accessing its Python-space `Operation`.
|
808
|
+
|
809
|
+
This is tightly coupled to `_recursively_handle_op`; it exists as a temporary measure to
|
810
|
+
avoid Python-space `Operation` creation from a `DAGOpNode` if we wouldn't do anything to the
|
811
|
+
node (which is _most_ nodes)."""
|
812
|
+
|
813
|
+
if (
|
814
|
+
dag._has_calibration_for(node)
|
815
|
+
or len(node.qargs) < self._min_qubits
|
816
|
+
or node.is_directive()
|
817
|
+
or (self._instruction_supported(node.name, qubits) and not node.is_control_flow())
|
818
|
+
):
|
819
|
+
return True
|
820
|
+
|
821
|
+
return (
|
822
|
+
# The fast path is just for Rust-space standard gates (which excludes
|
823
|
+
# `AnnotatedOperation`).
|
824
|
+
node.is_standard_gate()
|
825
|
+
# We don't have the fast-path for controlled gates over 3 or more qubits.
|
826
|
+
# However, we most probably want the fast-path for controlled 2-qubit gates
|
827
|
+
# (such as CX, CZ, CY, CH, CRX, and so on), so "_definitely_skip_node" should
|
828
|
+
# not immediately return False when encountering a controlled gate over 2 qubits.
|
829
|
+
and not (node.is_controlled_gate() and node.num_qubits >= 3)
|
830
|
+
# If there are plugins to try, they need to be tried.
|
831
|
+
and not self._methods_to_try(node.name)
|
832
|
+
# If all the above constraints hold, and it's already supported or the basis translator
|
833
|
+
# can handle it, we'll leave it be.
|
834
|
+
and (
|
835
|
+
# This uses unfortunately private details of `EquivalenceLibrary`, but so does the
|
836
|
+
# `BasisTranslator`, and this is supposed to just be temporary til this is moved
|
837
|
+
# into Rust space.
|
838
|
+
self._equiv_lib is not None
|
839
|
+
and equivalence.Key(name=node.name, num_qubits=node.num_qubits)
|
840
|
+
in self._equiv_lib.keys()
|
841
|
+
)
|
842
|
+
)
|
843
|
+
|
844
|
+
def _instruction_supported(self, name: str, qubits: tuple[int] | None) -> bool:
|
845
|
+
# include path for when target exists but target.num_qubits is None (BasicSimulator)
|
846
|
+
if self._target is None or self._target.num_qubits is None:
|
847
|
+
return name in self._device_insts
|
848
|
+
return self._target.instruction_supported(operation_name=name, qargs=qubits)
|
849
|
+
|
850
|
+
|
851
|
+
def _instruction_to_circuit(inst: Instruction) -> QuantumCircuit:
|
852
|
+
circuit = QuantumCircuit(inst.num_qubits, inst.num_clbits)
|
853
|
+
circuit.append(inst, circuit.qubits, circuit.clbits)
|
854
|
+
return circuit
|