qiskit 2.0.3__cp39-abi3-manylinux2014_aarch64.manylinux_2_17_aarch64.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 +141 -0
- qiskit/_accelerate.abi3.so +0 -0
- qiskit/_numpy_compat.py +73 -0
- qiskit/circuit/__init__.py +1343 -0
- qiskit/circuit/_add_control.py +312 -0
- qiskit/circuit/_classical_resource_map.py +150 -0
- qiskit/circuit/_standard_gates_commutations.py +3849 -0
- qiskit/circuit/_utils.py +167 -0
- qiskit/circuit/annotated_operation.py +279 -0
- qiskit/circuit/barrier.py +46 -0
- qiskit/circuit/classical/__init__.py +41 -0
- qiskit/circuit/classical/expr/__init__.py +266 -0
- qiskit/circuit/classical/expr/constructors.py +764 -0
- qiskit/circuit/classical/expr/expr.py +498 -0
- qiskit/circuit/classical/expr/visitors.py +375 -0
- qiskit/circuit/classical/types/__init__.py +113 -0
- qiskit/circuit/classical/types/ordering.py +229 -0
- qiskit/circuit/classical/types/types.py +153 -0
- qiskit/circuit/commutation_checker.py +133 -0
- qiskit/circuit/commutation_library.py +20 -0
- qiskit/circuit/controlflow/__init__.py +59 -0
- qiskit/circuit/controlflow/_builder_utils.py +211 -0
- qiskit/circuit/controlflow/box.py +163 -0
- qiskit/circuit/controlflow/break_loop.py +56 -0
- qiskit/circuit/controlflow/builder.py +791 -0
- qiskit/circuit/controlflow/continue_loop.py +56 -0
- qiskit/circuit/controlflow/control_flow.py +94 -0
- qiskit/circuit/controlflow/for_loop.py +218 -0
- qiskit/circuit/controlflow/if_else.py +498 -0
- qiskit/circuit/controlflow/switch_case.py +411 -0
- qiskit/circuit/controlflow/while_loop.py +166 -0
- qiskit/circuit/controlledgate.py +274 -0
- qiskit/circuit/delay.py +157 -0
- qiskit/circuit/duration.py +80 -0
- qiskit/circuit/equivalence.py +94 -0
- qiskit/circuit/equivalence_library.py +18 -0
- qiskit/circuit/exceptions.py +19 -0
- qiskit/circuit/gate.py +261 -0
- qiskit/circuit/instruction.py +564 -0
- qiskit/circuit/instructionset.py +132 -0
- qiskit/circuit/library/__init__.py +984 -0
- qiskit/circuit/library/arithmetic/__init__.py +40 -0
- qiskit/circuit/library/arithmetic/adders/__init__.py +18 -0
- qiskit/circuit/library/arithmetic/adders/adder.py +235 -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 +131 -0
- qiskit/circuit/library/arithmetic/functional_pauli_rotations.py +114 -0
- qiskit/circuit/library/arithmetic/integer_comparator.py +200 -0
- qiskit/circuit/library/arithmetic/linear_amplitude_function.py +363 -0
- qiskit/circuit/library/arithmetic/linear_pauli_rotations.py +243 -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 +201 -0
- qiskit/circuit/library/arithmetic/multipliers/rg_qft_multiplier.py +108 -0
- qiskit/circuit/library/arithmetic/piecewise_chebyshev.py +502 -0
- qiskit/circuit/library/arithmetic/piecewise_linear_pauli_rotations.py +387 -0
- qiskit/circuit/library/arithmetic/piecewise_polynomial_pauli_rotations.py +493 -0
- qiskit/circuit/library/arithmetic/polynomial_pauli_rotations.py +389 -0
- qiskit/circuit/library/arithmetic/quadratic_form.py +364 -0
- qiskit/circuit/library/arithmetic/weighted_adder.py +409 -0
- qiskit/circuit/library/basis_change/__init__.py +15 -0
- qiskit/circuit/library/basis_change/qft.py +316 -0
- qiskit/circuit/library/bit_flip_oracle.py +130 -0
- qiskit/circuit/library/blueprintcircuit.py +316 -0
- qiskit/circuit/library/boolean_logic/__init__.py +18 -0
- qiskit/circuit/library/boolean_logic/inner_product.py +157 -0
- qiskit/circuit/library/boolean_logic/quantum_and.py +204 -0
- qiskit/circuit/library/boolean_logic/quantum_or.py +206 -0
- qiskit/circuit/library/boolean_logic/quantum_xor.py +167 -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 +160 -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 +175 -0
- qiskit/circuit/library/generalized_gates/gr.py +219 -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 +84 -0
- qiskit/circuit/library/generalized_gates/permutation.py +198 -0
- qiskit/circuit/library/generalized_gates/rv.py +96 -0
- qiskit/circuit/library/generalized_gates/uc.py +303 -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 +217 -0
- qiskit/circuit/library/graph_state.py +172 -0
- qiskit/circuit/library/grover_operator.py +583 -0
- qiskit/circuit/library/hamiltonian_gate.py +142 -0
- qiskit/circuit/library/hidden_linear_function.py +163 -0
- qiskit/circuit/library/iqp.py +180 -0
- qiskit/circuit/library/n_local/__init__.py +45 -0
- qiskit/circuit/library/n_local/efficient_su2.py +282 -0
- qiskit/circuit/library/n_local/evolved_operator_ansatz.py +520 -0
- qiskit/circuit/library/n_local/excitation_preserving.py +303 -0
- qiskit/circuit/library/n_local/n_local.py +1477 -0
- qiskit/circuit/library/n_local/pauli_two_design.py +246 -0
- qiskit/circuit/library/n_local/qaoa_ansatz.py +367 -0
- qiskit/circuit/library/n_local/real_amplitudes.py +312 -0
- qiskit/circuit/library/n_local/two_local.py +289 -0
- qiskit/circuit/library/overlap.py +183 -0
- qiskit/circuit/library/pauli_evolution.py +201 -0
- qiskit/circuit/library/phase_estimation.py +177 -0
- qiskit/circuit/library/phase_oracle.py +239 -0
- qiskit/circuit/library/quantum_volume.py +180 -0
- qiskit/circuit/library/standard_gates/__init__.py +141 -0
- qiskit/circuit/library/standard_gates/dcx.py +77 -0
- qiskit/circuit/library/standard_gates/ecr.py +129 -0
- qiskit/circuit/library/standard_gates/equivalence_library.py +1800 -0
- qiskit/circuit/library/standard_gates/global_phase.py +84 -0
- qiskit/circuit/library/standard_gates/h.py +253 -0
- qiskit/circuit/library/standard_gates/i.py +76 -0
- qiskit/circuit/library/standard_gates/iswap.py +133 -0
- qiskit/circuit/library/standard_gates/p.py +422 -0
- qiskit/circuit/library/standard_gates/r.py +114 -0
- qiskit/circuit/library/standard_gates/rx.py +293 -0
- qiskit/circuit/library/standard_gates/rxx.py +180 -0
- qiskit/circuit/library/standard_gates/ry.py +286 -0
- qiskit/circuit/library/standard_gates/ryy.py +180 -0
- qiskit/circuit/library/standard_gates/rz.py +307 -0
- qiskit/circuit/library/standard_gates/rzx.py +226 -0
- qiskit/circuit/library/standard_gates/rzz.py +193 -0
- qiskit/circuit/library/standard_gates/s.py +419 -0
- qiskit/circuit/library/standard_gates/swap.py +281 -0
- qiskit/circuit/library/standard_gates/sx.py +310 -0
- qiskit/circuit/library/standard_gates/t.py +178 -0
- qiskit/circuit/library/standard_gates/u.py +395 -0
- qiskit/circuit/library/standard_gates/u1.py +490 -0
- qiskit/circuit/library/standard_gates/u2.py +145 -0
- qiskit/circuit/library/standard_gates/u3.py +428 -0
- qiskit/circuit/library/standard_gates/x.py +1481 -0
- qiskit/circuit/library/standard_gates/xx_minus_yy.py +202 -0
- qiskit/circuit/library/standard_gates/xx_plus_yy.py +236 -0
- qiskit/circuit/library/standard_gates/y.py +257 -0
- qiskit/circuit/library/standard_gates/z.py +338 -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 +53 -0
- qiskit/circuit/operation.py +68 -0
- qiskit/circuit/parameter.py +179 -0
- qiskit/circuit/parameterexpression.py +703 -0
- qiskit/circuit/parametertable.py +119 -0
- qiskit/circuit/parametervector.py +140 -0
- qiskit/circuit/quantumcircuit.py +7540 -0
- qiskit/circuit/quantumcircuitdata.py +136 -0
- qiskit/circuit/random/__init__.py +15 -0
- qiskit/circuit/random/utils.py +366 -0
- qiskit/circuit/reset.py +37 -0
- qiskit/circuit/singleton.py +600 -0
- qiskit/circuit/store.py +89 -0
- qiskit/circuit/tools/__init__.py +16 -0
- qiskit/circuit/tools/pi_check.py +193 -0
- qiskit/circuit/twirling.py +145 -0
- qiskit/compiler/__init__.py +27 -0
- qiskit/compiler/transpiler.py +375 -0
- qiskit/converters/__init__.py +74 -0
- qiskit/converters/circuit_to_dag.py +80 -0
- qiskit/converters/circuit_to_dagdependency.py +49 -0
- qiskit/converters/circuit_to_dagdependency_v2.py +46 -0
- qiskit/converters/circuit_to_gate.py +107 -0
- qiskit/converters/circuit_to_instruction.py +142 -0
- qiskit/converters/dag_to_circuit.py +79 -0
- qiskit/converters/dag_to_dagdependency.py +54 -0
- qiskit/converters/dag_to_dagdependency_v2.py +43 -0
- qiskit/converters/dagdependency_to_circuit.py +40 -0
- qiskit/converters/dagdependency_to_dag.py +48 -0
- qiskit/dagcircuit/__init__.py +55 -0
- qiskit/dagcircuit/collect_blocks.py +407 -0
- qiskit/dagcircuit/dagcircuit.py +24 -0
- qiskit/dagcircuit/dagdependency.py +612 -0
- qiskit/dagcircuit/dagdependency_v2.py +566 -0
- qiskit/dagcircuit/dagdepnode.py +160 -0
- qiskit/dagcircuit/dagnode.py +188 -0
- qiskit/dagcircuit/exceptions.py +42 -0
- qiskit/exceptions.py +153 -0
- qiskit/passmanager/__init__.py +258 -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 +353 -0
- qiskit/primitives/__init__.py +490 -0
- qiskit/primitives/backend_estimator_v2.py +530 -0
- qiskit/primitives/backend_sampler_v2.py +339 -0
- qiskit/primitives/base/__init__.py +20 -0
- qiskit/primitives/base/base_estimator.py +247 -0
- qiskit/primitives/base/base_primitive_job.py +78 -0
- qiskit/primitives/base/base_primitive_v1.py +45 -0
- qiskit/primitives/base/base_result_v1.py +65 -0
- qiskit/primitives/base/base_sampler.py +196 -0
- qiskit/primitives/base/estimator_result_v1.py +46 -0
- qiskit/primitives/base/sampler_result_v1.py +45 -0
- qiskit/primitives/base/validation_v1.py +250 -0
- qiskit/primitives/containers/__init__.py +26 -0
- qiskit/primitives/containers/bindings_array.py +391 -0
- qiskit/primitives/containers/bit_array.py +764 -0
- qiskit/primitives/containers/data_bin.py +175 -0
- qiskit/primitives/containers/estimator_pub.py +222 -0
- qiskit/primitives/containers/object_array.py +94 -0
- qiskit/primitives/containers/observables_array.py +296 -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/primitive_job.py +81 -0
- qiskit/primitives/statevector_estimator.py +175 -0
- qiskit/primitives/statevector_sampler.py +290 -0
- qiskit/primitives/utils.py +72 -0
- qiskit/providers/__init__.py +677 -0
- qiskit/providers/backend.py +364 -0
- qiskit/providers/basic_provider/__init__.py +47 -0
- qiskit/providers/basic_provider/basic_provider.py +121 -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 +693 -0
- qiskit/providers/basic_provider/exceptions.py +30 -0
- qiskit/providers/exceptions.py +33 -0
- qiskit/providers/fake_provider/__init__.py +69 -0
- qiskit/providers/fake_provider/generic_backend_v2.py +374 -0
- qiskit/providers/fake_provider/utils/__init__.py +15 -0
- qiskit/providers/job.py +147 -0
- qiskit/providers/jobstatus.py +30 -0
- qiskit/providers/options.py +273 -0
- qiskit/providers/providerutils.py +110 -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 +669 -0
- qiskit/qasm2/exceptions.py +27 -0
- qiskit/qasm2/export.py +364 -0
- qiskit/qasm2/parse.py +438 -0
- qiskit/qasm3/__init__.py +372 -0
- qiskit/qasm3/ast.py +782 -0
- qiskit/qasm3/exceptions.py +27 -0
- qiskit/qasm3/experimental.py +70 -0
- qiskit/qasm3/exporter.py +1340 -0
- qiskit/qasm3/printer.py +608 -0
- qiskit/qpy/__init__.py +1965 -0
- qiskit/qpy/binary_io/__init__.py +35 -0
- qiskit/qpy/binary_io/circuits.py +1455 -0
- qiskit/qpy/binary_io/parse_sympy_repr.py +121 -0
- qiskit/qpy/binary_io/schedules.py +308 -0
- qiskit/qpy/binary_io/value.py +1165 -0
- qiskit/qpy/common.py +353 -0
- qiskit/qpy/exceptions.py +53 -0
- qiskit/qpy/formats.py +442 -0
- qiskit/qpy/interface.py +344 -0
- qiskit/qpy/type_keys.py +409 -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 +104 -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 +373 -0
- qiskit/quantum_info/operators/channel/transformations.py +490 -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 +511 -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 +869 -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 +1032 -0
- qiskit/quantum_info/operators/symplectic/clifford_circuits.py +558 -0
- qiskit/quantum_info/operators/symplectic/pauli.py +755 -0
- qiskit/quantum_info/operators/symplectic/pauli_list.py +1242 -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 +1239 -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 +857 -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 +805 -0
- qiskit/quantum_info/states/statevector.py +977 -0
- qiskit/quantum_info/states/utils.py +247 -0
- qiskit/result/__init__.py +61 -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/models.py +241 -0
- qiskit/result/postprocess.py +239 -0
- qiskit/result/result.py +385 -0
- qiskit/result/sampled_expval.py +74 -0
- qiskit/result/utils.py +294 -0
- qiskit/synthesis/__init__.py +240 -0
- qiskit/synthesis/arithmetic/__init__.py +18 -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/comparators/__init__.py +16 -0
- qiskit/synthesis/arithmetic/comparators/compare_2s.py +112 -0
- qiskit/synthesis/arithmetic/comparators/compare_greedy.py +66 -0
- qiskit/synthesis/arithmetic/multipliers/__init__.py +16 -0
- qiskit/synthesis/arithmetic/multipliers/hrs_cumulative_multiplier.py +103 -0
- qiskit/synthesis/arithmetic/multipliers/rg_qft_multiplier.py +100 -0
- qiskit/synthesis/arithmetic/weighted_sum.py +155 -0
- qiskit/synthesis/boolean/__init__.py +13 -0
- qiskit/synthesis/boolean/boolean_expression.py +231 -0
- qiskit/synthesis/boolean/boolean_expression_synth.py +124 -0
- qiskit/synthesis/boolean/boolean_expression_visitor.py +96 -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 +265 -0
- qiskit/synthesis/discrete_basis/gate_sequence.py +421 -0
- qiskit/synthesis/discrete_basis/generate_basis_approximations.py +165 -0
- qiskit/synthesis/discrete_basis/solovay_kitaev.py +240 -0
- qiskit/synthesis/evolution/__init__.py +21 -0
- qiskit/synthesis/evolution/evolution_synthesis.py +48 -0
- qiskit/synthesis/evolution/lie_trotter.py +120 -0
- qiskit/synthesis/evolution/matrix_synthesis.py +47 -0
- qiskit/synthesis/evolution/pauli_network.py +80 -0
- qiskit/synthesis/evolution/product_formula.py +313 -0
- qiskit/synthesis/evolution/qdrift.py +130 -0
- qiskit/synthesis/evolution/suzuki_trotter.py +224 -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 +61 -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 +61 -0
- qiskit/synthesis/linear_phase/cz_depth_lnn.py +58 -0
- qiskit/synthesis/multi_controlled/__init__.py +25 -0
- qiskit/synthesis/multi_controlled/mcmt_vchain.py +52 -0
- qiskit/synthesis/multi_controlled/mcx_synthesis.py +359 -0
- qiskit/synthesis/multi_controlled/multi_control_rotation_gates.py +206 -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 +20 -0
- qiskit/synthesis/two_qubit/local_invariance.py +63 -0
- qiskit/synthesis/two_qubit/two_qubit_decompose.py +583 -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 +1345 -0
- qiskit/transpiler/basepasses.py +190 -0
- qiskit/transpiler/coupling.py +500 -0
- qiskit/transpiler/exceptions.py +59 -0
- qiskit/transpiler/instruction_durations.py +281 -0
- qiskit/transpiler/layout.py +740 -0
- qiskit/transpiler/passes/__init__.py +276 -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 +138 -0
- qiskit/transpiler/passes/basis/decompose.py +137 -0
- qiskit/transpiler/passes/basis/translate_parameterized.py +175 -0
- qiskit/transpiler/passes/basis/unroll_3q_or_more.py +84 -0
- qiskit/transpiler/passes/basis/unroll_custom_definitions.py +110 -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 +128 -0
- qiskit/transpiler/passes/layout/csp_layout.py +132 -0
- qiskit/transpiler/passes/layout/dense_layout.py +177 -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 +116 -0
- qiskit/transpiler/passes/layout/layout_2q_distance.py +77 -0
- qiskit/transpiler/passes/layout/sabre_layout.py +506 -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 +256 -0
- qiskit/transpiler/passes/layout/vf2_post_layout.py +376 -0
- qiskit/transpiler/passes/layout/vf2_utils.py +235 -0
- qiskit/transpiler/passes/optimization/__init__.py +42 -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 +117 -0
- qiskit/transpiler/passes/optimization/collect_cliffords.py +109 -0
- qiskit/transpiler/passes/optimization/collect_linear_functions.py +85 -0
- qiskit/transpiler/passes/optimization/collect_multiqubit_blocks.py +242 -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 +176 -0
- qiskit/transpiler/passes/optimization/contract_idle_wires_in_control_flow.py +104 -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/light_cone.py +135 -0
- qiskit/transpiler/passes/optimization/optimize_1q_commutation.py +267 -0
- qiskit/transpiler/passes/optimization/optimize_1q_decomposition.py +250 -0
- qiskit/transpiler/passes/optimization/optimize_1q_gates.py +384 -0
- qiskit/transpiler/passes/optimization/optimize_annotated.py +449 -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 +70 -0
- qiskit/transpiler/passes/optimization/remove_reset_in_zero_state.py +37 -0
- qiskit/transpiler/passes/optimization/reset_after_measure_simplification.py +50 -0
- qiskit/transpiler/passes/optimization/split_2q_unitaries.py +63 -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 +639 -0
- qiskit/transpiler/passes/optimization/template_optimization.py +158 -0
- qiskit/transpiler/passes/routing/__init__.py +21 -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 +397 -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 +463 -0
- qiskit/transpiler/passes/routing/star_prerouting.py +408 -0
- qiskit/transpiler/passes/routing/utils.py +35 -0
- qiskit/transpiler/passes/scheduling/__init__.py +21 -0
- qiskit/transpiler/passes/scheduling/alignments/__init__.py +79 -0
- qiskit/transpiler/passes/scheduling/alignments/check_durations.py +70 -0
- qiskit/transpiler/passes/scheduling/alignments/reschedule.py +251 -0
- qiskit/transpiler/passes/scheduling/padding/__init__.py +16 -0
- qiskit/transpiler/passes/scheduling/padding/base_padding.py +284 -0
- qiskit/transpiler/passes/scheduling/padding/dynamical_decoupling.py +415 -0
- qiskit/transpiler/passes/scheduling/padding/pad_delay.py +90 -0
- qiskit/transpiler/passes/scheduling/scheduling/__init__.py +17 -0
- qiskit/transpiler/passes/scheduling/scheduling/alap.py +93 -0
- qiskit/transpiler/passes/scheduling/scheduling/asap.py +100 -0
- qiskit/transpiler/passes/scheduling/scheduling/base_scheduler.py +88 -0
- qiskit/transpiler/passes/scheduling/scheduling/set_io_latency.py +64 -0
- qiskit/transpiler/passes/scheduling/time_unit_conversion.py +237 -0
- qiskit/transpiler/passes/synthesis/__init__.py +20 -0
- qiskit/transpiler/passes/synthesis/aqc_plugin.py +153 -0
- qiskit/transpiler/passes/synthesis/default_unitary_synth_plugin.py +653 -0
- qiskit/transpiler/passes/synthesis/high_level_synthesis.py +429 -0
- qiskit/transpiler/passes/synthesis/hls_plugins.py +1963 -0
- qiskit/transpiler/passes/synthesis/linear_functions_synthesis.py +41 -0
- qiskit/transpiler/passes/synthesis/plugin.py +738 -0
- qiskit/transpiler/passes/synthesis/solovay_kitaev_synthesis.py +313 -0
- qiskit/transpiler/passes/synthesis/unitary_synthesis.py +425 -0
- qiskit/transpiler/passes/utils/__init__.py +32 -0
- qiskit/transpiler/passes/utils/barrier_before_final_measurements.py +41 -0
- qiskit/transpiler/passes/utils/check_gate_direction.py +60 -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 +61 -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 +66 -0
- qiskit/transpiler/passes/utils/fixed_point.py +48 -0
- qiskit/transpiler/passes/utils/gate_direction.py +93 -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 +50 -0
- qiskit/transpiler/passes/utils/remove_final_measurements.py +121 -0
- qiskit/transpiler/passes/utils/unroll_forloops.py +81 -0
- qiskit/transpiler/passmanager.py +503 -0
- qiskit/transpiler/passmanager_config.py +151 -0
- qiskit/transpiler/preset_passmanagers/__init__.py +93 -0
- qiskit/transpiler/preset_passmanagers/builtin_plugins.py +993 -0
- qiskit/transpiler/preset_passmanagers/common.py +672 -0
- qiskit/transpiler/preset_passmanagers/generate_preset_pass_manager.py +437 -0
- qiskit/transpiler/preset_passmanagers/level0.py +104 -0
- qiskit/transpiler/preset_passmanagers/level1.py +108 -0
- qiskit/transpiler/preset_passmanagers/level2.py +109 -0
- qiskit/transpiler/preset_passmanagers/level3.py +110 -0
- qiskit/transpiler/preset_passmanagers/plugin.py +346 -0
- qiskit/transpiler/target.py +905 -0
- qiskit/transpiler/timing_constraints.py +59 -0
- qiskit/user_config.py +266 -0
- qiskit/utils/__init__.py +90 -0
- qiskit/utils/classtools.py +146 -0
- qiskit/utils/deprecation.py +382 -0
- qiskit/utils/lazy_tester.py +363 -0
- qiskit/utils/optionals.py +354 -0
- qiskit/utils/parallel.py +318 -0
- qiskit/utils/units.py +146 -0
- qiskit/version.py +84 -0
- qiskit/visualization/__init__.py +290 -0
- qiskit/visualization/array.py +207 -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 +735 -0
- qiskit/visualization/circuit/latex.py +661 -0
- qiskit/visualization/circuit/matplotlib.py +2019 -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 +1849 -0
- qiskit/visualization/circuit_visualization.py +19 -0
- qiskit/visualization/counts_visualization.py +487 -0
- qiskit/visualization/dag_visualization.py +318 -0
- qiskit/visualization/exceptions.py +21 -0
- qiskit/visualization/gate_map.py +1424 -0
- qiskit/visualization/library.py +40 -0
- qiskit/visualization/pass_manager_visualization.py +312 -0
- qiskit/visualization/state_visualization.py +1546 -0
- qiskit/visualization/timeline/__init__.py +21 -0
- qiskit/visualization/timeline/core.py +495 -0
- qiskit/visualization/timeline/drawings.py +260 -0
- qiskit/visualization/timeline/generators.py +506 -0
- qiskit/visualization/timeline/interface.py +444 -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 +195 -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-2.0.3.dist-info/METADATA +220 -0
- qiskit-2.0.3.dist-info/RECORD +690 -0
- qiskit-2.0.3.dist-info/WHEEL +6 -0
- qiskit-2.0.3.dist-info/entry_points.txt +82 -0
- qiskit-2.0.3.dist-info/licenses/LICENSE.txt +203 -0
- qiskit-2.0.3.dist-info/top_level.txt +1 -0
@@ -0,0 +1,376 @@
|
|
1
|
+
# This code is part of Qiskit.
|
2
|
+
#
|
3
|
+
# (C) Copyright IBM 2021.
|
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
|
+
"""VF2PostLayout pass to find a layout after transpile using subgraph isomorphism"""
|
15
|
+
from enum import Enum
|
16
|
+
import logging
|
17
|
+
import inspect
|
18
|
+
import itertools
|
19
|
+
import time
|
20
|
+
|
21
|
+
from rustworkx import PyDiGraph, vf2_mapping, PyGraph
|
22
|
+
|
23
|
+
from qiskit.transpiler.layout import Layout
|
24
|
+
from qiskit.transpiler.basepasses import AnalysisPass
|
25
|
+
from qiskit.transpiler.exceptions import TranspilerError
|
26
|
+
from qiskit.transpiler.passes.layout import vf2_utils
|
27
|
+
|
28
|
+
|
29
|
+
logger = logging.getLogger(__name__)
|
30
|
+
|
31
|
+
|
32
|
+
class VF2PostLayoutStopReason(Enum):
|
33
|
+
"""Stop reasons for VF2PostLayout pass."""
|
34
|
+
|
35
|
+
SOLUTION_FOUND = "solution found"
|
36
|
+
NO_BETTER_SOLUTION_FOUND = "no better solution found"
|
37
|
+
NO_SOLUTION_FOUND = "nonexistent solution"
|
38
|
+
MORE_THAN_2Q = ">2q gates in basis"
|
39
|
+
|
40
|
+
|
41
|
+
def _target_match(node_a, node_b):
|
42
|
+
# Node A is the set of operations in the target. Node B is the count dict
|
43
|
+
# of operations on the node or edge in the circuit.
|
44
|
+
if isinstance(node_a, set):
|
45
|
+
return node_a.issuperset(node_b.keys())
|
46
|
+
# Node A is the count dict of operations on the node or edge in the circuit
|
47
|
+
# Node B is the set of operations in the target on the same qubit(s).
|
48
|
+
else:
|
49
|
+
return set(node_a).issubset(node_b)
|
50
|
+
|
51
|
+
|
52
|
+
class VF2PostLayout(AnalysisPass):
|
53
|
+
"""A pass for improving an existing Layout after transpilation of a circuit onto a
|
54
|
+
Coupling graph, as a subgraph isomorphism problem, solved by VF2++.
|
55
|
+
|
56
|
+
Unlike the :class:`~.VF2Layout` transpiler pass which is designed to find an
|
57
|
+
initial layout for a circuit early in the transpilation pipeline this transpiler
|
58
|
+
pass is designed to try and find a better layout after transpilation is complete.
|
59
|
+
The initial layout phase of the transpiler doesn't have as much information available
|
60
|
+
as we do after transpilation. This pass is designed to be paired in a similar pipeline
|
61
|
+
as the layout passes. This pass will strip any idle wires from the circuit, use VF2
|
62
|
+
to find a subgraph in the coupling graph for the circuit to run on with better fidelity
|
63
|
+
and then update the circuit layout to use the new qubits. The algorithm used in this
|
64
|
+
pass is described in `arXiv:2209.15512 <https://arxiv.org/abs/2209.15512>`__.
|
65
|
+
|
66
|
+
If a solution is found that means there is a lower error layout available for the
|
67
|
+
circuit. If a solution is found the layout will be set in the property set as
|
68
|
+
``property_set['post_layout']``. However, if no solution or no better solution is found, no
|
69
|
+
``property_set['post_layout']`` is set. The stopping reason is
|
70
|
+
set in ``property_set['VF2PostLayout_stop_reason']`` in all the cases and will be
|
71
|
+
one of the values enumerated in ``VF2PostLayoutStopReason`` which has the
|
72
|
+
following values:
|
73
|
+
|
74
|
+
* ``"solution found"``: If a solution was found.
|
75
|
+
* ``"no better solution found"``: If the initial layout of the circuit is the best solution.
|
76
|
+
* ``"nonexistent solution"``: If no solution was found.
|
77
|
+
* ``">2q gates in basis"``: If VF2PostLayout can't work with the basis of the circuit.
|
78
|
+
|
79
|
+
By default, this pass will construct a heuristic scoring map based on
|
80
|
+
the error rates in the provided ``target``. However, analysis passes can be run prior to this pass
|
81
|
+
and set ``vf2_avg_error_map`` in the property set with a :class:`~.ErrorMap`
|
82
|
+
instance. If a value is ``NaN`` that is treated as an ideal edge
|
83
|
+
For example if an error map is created as::
|
84
|
+
|
85
|
+
from qiskit.transpiler.passes.layout.vf2_utils import ErrorMap
|
86
|
+
|
87
|
+
error_map = ErrorMap(3)
|
88
|
+
error_map.add_error((0, 0), 0.0024)
|
89
|
+
error_map.add_error((0, 1), 0.01)
|
90
|
+
error_map.add_error((1, 1), 0.0032)
|
91
|
+
|
92
|
+
that represents the error map for a 2 qubit target, where the avg 1q error
|
93
|
+
rate is ``0.0024`` on qubit 0 and ``0.0032`` on qubit 1. Then the avg 2q
|
94
|
+
error rate for gates that operate on (0, 1) is 0.01 and (1, 0) is not
|
95
|
+
supported by the target. This will be used for scoring if it's set as the
|
96
|
+
``vf2_avg_error_map`` key in the property set when :class:`~.VF2PostLayout`
|
97
|
+
is run.
|
98
|
+
"""
|
99
|
+
|
100
|
+
def __init__(
|
101
|
+
self,
|
102
|
+
target=None,
|
103
|
+
seed=None,
|
104
|
+
call_limit=None,
|
105
|
+
time_limit=None,
|
106
|
+
strict_direction=True,
|
107
|
+
max_trials=0,
|
108
|
+
):
|
109
|
+
"""Initialize a ``VF2PostLayout`` pass instance
|
110
|
+
|
111
|
+
Args:
|
112
|
+
target (Target): A target representing the backend device to run ``VF2PostLayout`` on.
|
113
|
+
seed (int): Sets the seed of the PRNG. -1 Means no node shuffling.
|
114
|
+
call_limit (int): The number of state visits to attempt in each execution of
|
115
|
+
VF2.
|
116
|
+
time_limit (float): The total time limit in seconds to run ``VF2PostLayout``
|
117
|
+
strict_direction (bool): Whether the pass is configured to follow
|
118
|
+
the strict direction in the coupling graph. If this is set to
|
119
|
+
false, the pass will treat any edge in the coupling graph as
|
120
|
+
a weak edge and the interaction graph will be undirected. For
|
121
|
+
the purposes of evaluating layouts the avg error rate for
|
122
|
+
each qubit and 2q link will be used. This enables the pass to be
|
123
|
+
run prior to basis translation and work with any 1q and 2q operations.
|
124
|
+
However, if ``strict_direction=True`` the pass expects the input
|
125
|
+
:class:`~.DAGCircuit` object to :meth:`~.VF2PostLayout.run` to be in
|
126
|
+
the target set of instructions.
|
127
|
+
max_trials (int): The maximum number of trials to run VF2 to find
|
128
|
+
a layout. A value of ``0`` (the default) means 'unlimited'.
|
129
|
+
|
130
|
+
Raises:
|
131
|
+
TypeError: At runtime, if ``target`` isn't provided.
|
132
|
+
"""
|
133
|
+
super().__init__()
|
134
|
+
self.target = target
|
135
|
+
self.call_limit = call_limit
|
136
|
+
self.time_limit = time_limit
|
137
|
+
self.max_trials = max_trials
|
138
|
+
self.seed = seed
|
139
|
+
self.strict_direction = strict_direction
|
140
|
+
self.avg_error_map = None
|
141
|
+
|
142
|
+
def run(self, dag):
|
143
|
+
"""run the layout method"""
|
144
|
+
if self.target is None:
|
145
|
+
raise TranspilerError("A target must be specified or a coupling map must be provided")
|
146
|
+
if not self.strict_direction:
|
147
|
+
self.avg_error_map = self.property_set["vf2_avg_error_map"]
|
148
|
+
if self.avg_error_map is None:
|
149
|
+
self.avg_error_map = vf2_utils.build_average_error_map(self.target, None)
|
150
|
+
|
151
|
+
result = vf2_utils.build_interaction_graph(dag, self.strict_direction)
|
152
|
+
if result is None:
|
153
|
+
self.property_set["VF2PostLayout_stop_reason"] = VF2PostLayoutStopReason.MORE_THAN_2Q
|
154
|
+
return
|
155
|
+
im_graph, im_graph_node_map, reverse_im_graph_node_map, free_nodes = result
|
156
|
+
scoring_bit_list = vf2_utils.build_bit_list(im_graph, im_graph_node_map)
|
157
|
+
scoring_edge_list = vf2_utils.build_edge_list(im_graph)
|
158
|
+
|
159
|
+
# If qargs is None then target is global and ideal so no
|
160
|
+
# scoring is needed
|
161
|
+
if self.target.qargs is None:
|
162
|
+
return
|
163
|
+
if self.strict_direction:
|
164
|
+
cm_graph = PyDiGraph(multigraph=False)
|
165
|
+
else:
|
166
|
+
cm_graph = PyGraph(multigraph=False)
|
167
|
+
# If None is present in qargs there are globally defined ideal operations
|
168
|
+
# we should add these to all entries based on the number of qubits, so we
|
169
|
+
# treat that as a valid operation even if there is no scoring for the
|
170
|
+
# strict direction case
|
171
|
+
global_ops = None
|
172
|
+
if None in self.target.qargs:
|
173
|
+
global_ops = {1: [], 2: []}
|
174
|
+
for op in self.target.operation_names_for_qargs(None):
|
175
|
+
operation = self.target.operation_for_name(op)
|
176
|
+
# If operation is a class this is a variable width ideal instruction
|
177
|
+
# so we treat it as available on both 1 and 2 qubits
|
178
|
+
if inspect.isclass(operation):
|
179
|
+
global_ops[1].append(op)
|
180
|
+
global_ops[2].append(op)
|
181
|
+
else:
|
182
|
+
num_qubits = operation.num_qubits
|
183
|
+
if num_qubits in global_ops:
|
184
|
+
global_ops[num_qubits].append(op)
|
185
|
+
op_names = []
|
186
|
+
for i in range(self.target.num_qubits):
|
187
|
+
try:
|
188
|
+
entry = set(self.target.operation_names_for_qargs((i,)))
|
189
|
+
except KeyError:
|
190
|
+
entry = set()
|
191
|
+
if global_ops is not None:
|
192
|
+
entry.update(global_ops[1])
|
193
|
+
op_names.append(entry)
|
194
|
+
cm_graph.add_nodes_from(op_names)
|
195
|
+
for qargs in self.target.qargs:
|
196
|
+
len_args = len(qargs)
|
197
|
+
# If qargs == 1 we already populated it and if qargs > 2 there are no instructions
|
198
|
+
# using those in the circuit because we'd have already returned by this point
|
199
|
+
if len_args == 2:
|
200
|
+
ops = set(self.target.operation_names_for_qargs(qargs))
|
201
|
+
if global_ops is not None:
|
202
|
+
ops.update(global_ops[2])
|
203
|
+
cm_graph.add_edge(qargs[0], qargs[1], ops)
|
204
|
+
cm_nodes = list(cm_graph.node_indexes())
|
205
|
+
# Filter qubits without any supported operations. If they
|
206
|
+
# don't support any operations, they're not valid for layout selection.
|
207
|
+
# This is only needed in the undirected case because in strict direction
|
208
|
+
# mode the node matcher will not match since none of the circuit ops
|
209
|
+
# will match the cmap ops.
|
210
|
+
if not self.strict_direction:
|
211
|
+
has_operations = set(itertools.chain.from_iterable(self.target.qargs))
|
212
|
+
to_remove = set(cm_graph.node_indices()).difference(has_operations)
|
213
|
+
if to_remove:
|
214
|
+
cm_graph.remove_nodes_from(list(to_remove))
|
215
|
+
|
216
|
+
logger.debug("Running VF2 to find post transpile mappings")
|
217
|
+
if self.target and self.strict_direction:
|
218
|
+
mappings = vf2_mapping(
|
219
|
+
cm_graph,
|
220
|
+
im_graph,
|
221
|
+
node_matcher=_target_match,
|
222
|
+
edge_matcher=_target_match,
|
223
|
+
subgraph=True,
|
224
|
+
id_order=False,
|
225
|
+
induced=False,
|
226
|
+
call_limit=self.call_limit,
|
227
|
+
)
|
228
|
+
else:
|
229
|
+
mappings = vf2_mapping(
|
230
|
+
cm_graph,
|
231
|
+
im_graph,
|
232
|
+
subgraph=True,
|
233
|
+
id_order=False,
|
234
|
+
induced=False,
|
235
|
+
call_limit=self.call_limit,
|
236
|
+
)
|
237
|
+
chosen_layout = None
|
238
|
+
try:
|
239
|
+
if self.strict_direction:
|
240
|
+
initial_layout = Layout({bit: index for index, bit in enumerate(dag.qubits)})
|
241
|
+
chosen_layout_score = self._score_layout(
|
242
|
+
initial_layout,
|
243
|
+
im_graph_node_map,
|
244
|
+
reverse_im_graph_node_map,
|
245
|
+
im_graph,
|
246
|
+
)
|
247
|
+
else:
|
248
|
+
initial_layout = {
|
249
|
+
im_graph_node_map[bit]: index
|
250
|
+
for index, bit in enumerate(dag.qubits)
|
251
|
+
if bit in im_graph_node_map
|
252
|
+
}
|
253
|
+
chosen_layout_score = vf2_utils.score_layout(
|
254
|
+
self.avg_error_map,
|
255
|
+
initial_layout,
|
256
|
+
im_graph_node_map,
|
257
|
+
reverse_im_graph_node_map,
|
258
|
+
im_graph,
|
259
|
+
self.strict_direction,
|
260
|
+
edge_list=scoring_edge_list,
|
261
|
+
bit_list=scoring_bit_list,
|
262
|
+
)
|
263
|
+
chosen_layout = initial_layout
|
264
|
+
stop_reason = VF2PostLayoutStopReason.NO_BETTER_SOLUTION_FOUND
|
265
|
+
# Circuit not in basis so we have nothing to compare against return here
|
266
|
+
except KeyError:
|
267
|
+
self.property_set["VF2PostLayout_stop_reason"] = (
|
268
|
+
VF2PostLayoutStopReason.NO_SOLUTION_FOUND
|
269
|
+
)
|
270
|
+
return
|
271
|
+
|
272
|
+
logger.debug("Initial layout has score %s", chosen_layout_score)
|
273
|
+
|
274
|
+
start_time = time.time()
|
275
|
+
trials = 0
|
276
|
+
for mapping in mappings:
|
277
|
+
trials += 1
|
278
|
+
logger.debug("Running trial: %s", trials)
|
279
|
+
layout_mapping = {im_i: cm_nodes[cm_i] for cm_i, im_i in mapping.items()}
|
280
|
+
if self.strict_direction:
|
281
|
+
layout = Layout(
|
282
|
+
{reverse_im_graph_node_map[k]: v for k, v in layout_mapping.items()}
|
283
|
+
)
|
284
|
+
layout_score = self._score_layout(
|
285
|
+
layout, im_graph_node_map, reverse_im_graph_node_map, im_graph
|
286
|
+
)
|
287
|
+
else:
|
288
|
+
layout_score = vf2_utils.score_layout(
|
289
|
+
self.avg_error_map,
|
290
|
+
layout_mapping,
|
291
|
+
im_graph_node_map,
|
292
|
+
reverse_im_graph_node_map,
|
293
|
+
im_graph,
|
294
|
+
self.strict_direction,
|
295
|
+
edge_list=scoring_edge_list,
|
296
|
+
bit_list=scoring_bit_list,
|
297
|
+
)
|
298
|
+
logger.debug("Trial %s has score %s", trials, layout_score)
|
299
|
+
if layout_score < chosen_layout_score:
|
300
|
+
layout = Layout(
|
301
|
+
{reverse_im_graph_node_map[k]: v for k, v in layout_mapping.items()}
|
302
|
+
)
|
303
|
+
logger.debug(
|
304
|
+
"Found layout %s has a lower score (%s) than previous best %s (%s)",
|
305
|
+
layout,
|
306
|
+
layout_score,
|
307
|
+
chosen_layout,
|
308
|
+
chosen_layout_score,
|
309
|
+
)
|
310
|
+
chosen_layout = layout
|
311
|
+
chosen_layout_score = layout_score
|
312
|
+
stop_reason = VF2PostLayoutStopReason.SOLUTION_FOUND
|
313
|
+
|
314
|
+
if self.max_trials and trials >= self.max_trials:
|
315
|
+
logger.debug("Trial %s is >= configured max trials %s", trials, self.max_trials)
|
316
|
+
break
|
317
|
+
|
318
|
+
elapsed_time = time.time() - start_time
|
319
|
+
if self.time_limit is not None and elapsed_time >= self.time_limit:
|
320
|
+
logger.debug(
|
321
|
+
"VFPostLayout has taken %s which exceeds configured max time: %s",
|
322
|
+
elapsed_time,
|
323
|
+
self.time_limit,
|
324
|
+
)
|
325
|
+
break
|
326
|
+
if stop_reason == VF2PostLayoutStopReason.SOLUTION_FOUND:
|
327
|
+
chosen_layout = vf2_utils.map_free_qubits(
|
328
|
+
free_nodes,
|
329
|
+
chosen_layout,
|
330
|
+
cm_graph.num_nodes(),
|
331
|
+
reverse_im_graph_node_map,
|
332
|
+
self.avg_error_map,
|
333
|
+
)
|
334
|
+
existing_layout = self.property_set["layout"]
|
335
|
+
# If any ancillas in initial layout map them back to the final layout output
|
336
|
+
if existing_layout is not None and len(existing_layout) > len(chosen_layout):
|
337
|
+
virtual_bits = chosen_layout.get_virtual_bits()
|
338
|
+
used_bits = set(virtual_bits.values())
|
339
|
+
num_qubits = len(cm_graph)
|
340
|
+
for bit in dag.qubits:
|
341
|
+
if len(chosen_layout) == len(existing_layout):
|
342
|
+
break
|
343
|
+
if bit not in virtual_bits:
|
344
|
+
for i in range(num_qubits):
|
345
|
+
if i not in used_bits:
|
346
|
+
used_bits.add(i)
|
347
|
+
chosen_layout.add(bit, i)
|
348
|
+
break
|
349
|
+
self.property_set["post_layout"] = chosen_layout
|
350
|
+
else:
|
351
|
+
if chosen_layout is None:
|
352
|
+
stop_reason = VF2PostLayoutStopReason.NO_SOLUTION_FOUND
|
353
|
+
# else the initial layout is optimal -> don't set post_layout, return 'no better solution'
|
354
|
+
self.property_set["VF2PostLayout_stop_reason"] = stop_reason
|
355
|
+
|
356
|
+
def _score_layout(self, layout, bit_map, reverse_bit_map, im_graph):
|
357
|
+
bits = layout.get_virtual_bits()
|
358
|
+
fidelity = 1
|
359
|
+
if self.target is not None:
|
360
|
+
for bit, node_index in bit_map.items():
|
361
|
+
gate_counts = im_graph[node_index]
|
362
|
+
for gate, count in gate_counts.items():
|
363
|
+
if self.target[gate] is not None and None not in self.target[gate]:
|
364
|
+
props = self.target[gate][(bits[bit],)]
|
365
|
+
if props is not None and props.error is not None:
|
366
|
+
fidelity *= (1 - props.error) ** count
|
367
|
+
|
368
|
+
for edge in im_graph.edge_index_map().values():
|
369
|
+
qargs = (bits[reverse_bit_map[edge[0]]], bits[reverse_bit_map[edge[1]]])
|
370
|
+
gate_counts = edge[2]
|
371
|
+
for gate, count in gate_counts.items():
|
372
|
+
if self.target[gate] is not None and None not in self.target[gate]:
|
373
|
+
props = self.target[gate][qargs]
|
374
|
+
if props is not None and props.error is not None:
|
375
|
+
fidelity *= (1 - props.error) ** count
|
376
|
+
return 1 - fidelity
|
@@ -0,0 +1,235 @@
|
|
1
|
+
# This code is part of Qiskit.
|
2
|
+
#
|
3
|
+
# (C) Copyright IBM 2022.
|
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
|
+
"""This module contains common utils for vf2 layout passes."""
|
14
|
+
|
15
|
+
from collections import defaultdict
|
16
|
+
import random
|
17
|
+
|
18
|
+
import numpy as np
|
19
|
+
from rustworkx import PyDiGraph, PyGraph, connected_components
|
20
|
+
|
21
|
+
from qiskit.circuit import ForLoopOp
|
22
|
+
from qiskit.converters import circuit_to_dag
|
23
|
+
from qiskit._accelerate import vf2_layout
|
24
|
+
from qiskit._accelerate.nlayout import NLayout
|
25
|
+
from qiskit._accelerate.error_map import ErrorMap
|
26
|
+
|
27
|
+
|
28
|
+
def build_interaction_graph(dag, strict_direction=True):
|
29
|
+
"""Build an interaction graph from a dag."""
|
30
|
+
im_graph = PyDiGraph(multigraph=False) if strict_direction else PyGraph(multigraph=False)
|
31
|
+
im_graph_node_map = {}
|
32
|
+
reverse_im_graph_node_map = {}
|
33
|
+
|
34
|
+
class MultiQEncountered(Exception):
|
35
|
+
"""Used to singal an error-status return from the DAG visitor."""
|
36
|
+
|
37
|
+
def _visit(dag, weight, wire_map):
|
38
|
+
for node in dag.op_nodes(include_directives=False):
|
39
|
+
if node.is_control_flow():
|
40
|
+
if isinstance(node.op, ForLoopOp):
|
41
|
+
inner_weight = len(node.op.params[0]) * weight
|
42
|
+
else:
|
43
|
+
inner_weight = weight
|
44
|
+
for block in node.op.blocks:
|
45
|
+
inner_wire_map = {
|
46
|
+
inner: wire_map[outer] for outer, inner in zip(node.qargs, block.qubits)
|
47
|
+
}
|
48
|
+
_visit(circuit_to_dag(block), inner_weight, inner_wire_map)
|
49
|
+
continue
|
50
|
+
len_args = len(node.qargs)
|
51
|
+
qargs = [wire_map[q] for q in node.qargs]
|
52
|
+
if len_args == 1:
|
53
|
+
if qargs[0] not in im_graph_node_map:
|
54
|
+
weights = defaultdict(int)
|
55
|
+
weights[node.name] += weight
|
56
|
+
im_graph_node_map[qargs[0]] = im_graph.add_node(weights)
|
57
|
+
reverse_im_graph_node_map[im_graph_node_map[qargs[0]]] = qargs[0]
|
58
|
+
else:
|
59
|
+
im_graph[im_graph_node_map[qargs[0]]][node.name] += weight
|
60
|
+
if len_args == 2:
|
61
|
+
if qargs[0] not in im_graph_node_map:
|
62
|
+
im_graph_node_map[qargs[0]] = im_graph.add_node(defaultdict(int))
|
63
|
+
reverse_im_graph_node_map[im_graph_node_map[qargs[0]]] = qargs[0]
|
64
|
+
if qargs[1] not in im_graph_node_map:
|
65
|
+
im_graph_node_map[qargs[1]] = im_graph.add_node(defaultdict(int))
|
66
|
+
reverse_im_graph_node_map[im_graph_node_map[qargs[1]]] = qargs[1]
|
67
|
+
edge = (im_graph_node_map[qargs[0]], im_graph_node_map[qargs[1]])
|
68
|
+
if im_graph.has_edge(*edge):
|
69
|
+
im_graph.get_edge_data(*edge)[node.name] += weight
|
70
|
+
else:
|
71
|
+
weights = defaultdict(int)
|
72
|
+
weights[node.name] += weight
|
73
|
+
im_graph.add_edge(*edge, weights)
|
74
|
+
if len_args > 2:
|
75
|
+
raise MultiQEncountered()
|
76
|
+
|
77
|
+
try:
|
78
|
+
_visit(dag, 1, {bit: bit for bit in dag.qubits})
|
79
|
+
except MultiQEncountered:
|
80
|
+
return None
|
81
|
+
# Remove components with no 2q interactions from interaction graph
|
82
|
+
# these will be evaluated separately independently of scoring isomorphic
|
83
|
+
# mappings. This is not done for strict direction because for post layout
|
84
|
+
# we need to factor in local operation constraints when evaluating a graph
|
85
|
+
free_nodes = {}
|
86
|
+
if not strict_direction:
|
87
|
+
conn_comp = connected_components(im_graph)
|
88
|
+
for comp in conn_comp:
|
89
|
+
if len(comp) == 1:
|
90
|
+
index = comp.pop()
|
91
|
+
free_nodes[index] = im_graph[index]
|
92
|
+
im_graph.remove_node(index)
|
93
|
+
|
94
|
+
return im_graph, im_graph_node_map, reverse_im_graph_node_map, free_nodes
|
95
|
+
|
96
|
+
|
97
|
+
def build_edge_list(im_graph):
|
98
|
+
"""Generate an edge list for scoring."""
|
99
|
+
return vf2_layout.EdgeList(
|
100
|
+
[((edge[0], edge[1]), sum(edge[2].values())) for edge in im_graph.edge_index_map().values()]
|
101
|
+
)
|
102
|
+
|
103
|
+
|
104
|
+
def build_bit_list(im_graph, bit_map):
|
105
|
+
"""Generate a bit list for scoring."""
|
106
|
+
bit_list = np.zeros(len(im_graph), dtype=np.int32)
|
107
|
+
for node_index in bit_map.values():
|
108
|
+
try:
|
109
|
+
bit_list[node_index] = sum(im_graph[node_index].values())
|
110
|
+
# If node_index not in im_graph that means there was a standalone
|
111
|
+
# node we will score/sort separately outside the vf2 mapping, so we
|
112
|
+
# can skip the hole
|
113
|
+
except IndexError:
|
114
|
+
pass
|
115
|
+
return bit_list
|
116
|
+
|
117
|
+
|
118
|
+
def score_layout(
|
119
|
+
avg_error_map,
|
120
|
+
layout_mapping,
|
121
|
+
bit_map,
|
122
|
+
_reverse_bit_map,
|
123
|
+
im_graph,
|
124
|
+
strict_direction=False,
|
125
|
+
run_in_parallel=False,
|
126
|
+
edge_list=None,
|
127
|
+
bit_list=None,
|
128
|
+
):
|
129
|
+
"""Score a layout given an average error map."""
|
130
|
+
if layout_mapping:
|
131
|
+
size = max(max(layout_mapping), max(layout_mapping.values()))
|
132
|
+
else:
|
133
|
+
size = 0
|
134
|
+
nlayout = NLayout(layout_mapping, size + 1, size + 1)
|
135
|
+
if bit_list is None:
|
136
|
+
bit_list = build_bit_list(im_graph, bit_map)
|
137
|
+
if edge_list is None:
|
138
|
+
edge_list = build_edge_list(im_graph)
|
139
|
+
return vf2_layout.score_layout(
|
140
|
+
bit_list, edge_list, avg_error_map, nlayout, strict_direction, run_in_parallel
|
141
|
+
)
|
142
|
+
|
143
|
+
|
144
|
+
def build_average_error_map(target, coupling_map):
|
145
|
+
"""Build an average error map used for scoring layouts pre-basis translation."""
|
146
|
+
num_qubits = 0
|
147
|
+
if target is not None and target.qargs is not None:
|
148
|
+
num_qubits = target.num_qubits
|
149
|
+
avg_map = ErrorMap(len(target.qargs))
|
150
|
+
elif coupling_map is not None:
|
151
|
+
num_qubits = coupling_map.size()
|
152
|
+
avg_map = ErrorMap(num_qubits + coupling_map.graph.num_edges())
|
153
|
+
else:
|
154
|
+
# If coupling map is not defined almost certainly we don't have any
|
155
|
+
# data to build an error map, but just in case initialize an empty
|
156
|
+
# object
|
157
|
+
avg_map = ErrorMap(0)
|
158
|
+
built = False
|
159
|
+
if target is not None and target.qargs is not None:
|
160
|
+
for qargs in target.qargs:
|
161
|
+
if qargs is None:
|
162
|
+
continue
|
163
|
+
qarg_error = 0.0
|
164
|
+
count = 0
|
165
|
+
for op in target.operation_names_for_qargs(qargs):
|
166
|
+
inst_props = target[op].get(qargs, None)
|
167
|
+
if inst_props is not None and inst_props.error is not None:
|
168
|
+
count += 1
|
169
|
+
qarg_error += inst_props.error
|
170
|
+
if count > 0:
|
171
|
+
if len(qargs) == 1:
|
172
|
+
qargs = (qargs[0], qargs[0])
|
173
|
+
avg_map.add_error(qargs, qarg_error / count)
|
174
|
+
built = True
|
175
|
+
# if there are no error rates in the target we should fallback to using the degree heuristic
|
176
|
+
# used for a coupling map. To do this we can build the coupling map from the target before
|
177
|
+
# running the fallback heuristic
|
178
|
+
if not built and target is not None and coupling_map is None:
|
179
|
+
coupling_map = target.build_coupling_map()
|
180
|
+
if not built and coupling_map is not None and num_qubits is not None:
|
181
|
+
for qubit in range(num_qubits):
|
182
|
+
neighbor_set = set(coupling_map.graph.successor_indices(qubit))
|
183
|
+
neighbor_set.update(coupling_map.graph.predecessor_indices(qubit))
|
184
|
+
degree = len(neighbor_set)
|
185
|
+
avg_map.add_error((qubit, qubit), degree / num_qubits)
|
186
|
+
for edge in coupling_map.graph.edge_list():
|
187
|
+
avg_map.add_error(edge, (avg_map[edge[0], edge[0]] + avg_map[edge[1], edge[1]]) / 2)
|
188
|
+
built = True
|
189
|
+
if built:
|
190
|
+
return avg_map
|
191
|
+
else:
|
192
|
+
return None
|
193
|
+
|
194
|
+
|
195
|
+
def shuffle_coupling_graph(coupling_map, seed, strict_direction=True):
|
196
|
+
"""Create a shuffled coupling graph from a coupling map."""
|
197
|
+
if strict_direction:
|
198
|
+
cm_graph = coupling_map.graph
|
199
|
+
else:
|
200
|
+
cm_graph = coupling_map.graph.to_undirected(multigraph=False)
|
201
|
+
cm_nodes = list(cm_graph.node_indexes())
|
202
|
+
if seed != -1:
|
203
|
+
random.Random(seed).shuffle(cm_nodes)
|
204
|
+
shuffled_cm_graph = type(cm_graph)()
|
205
|
+
shuffled_cm_graph.add_nodes_from(cm_nodes)
|
206
|
+
new_edges = [(cm_nodes[edge[0]], cm_nodes[edge[1]]) for edge in cm_graph.edge_list()]
|
207
|
+
shuffled_cm_graph.add_edges_from_no_data(new_edges)
|
208
|
+
cm_nodes = [k for k, v in sorted(enumerate(cm_nodes), key=lambda item: item[1])]
|
209
|
+
cm_graph = shuffled_cm_graph
|
210
|
+
return cm_graph, cm_nodes
|
211
|
+
|
212
|
+
|
213
|
+
def map_free_qubits(
|
214
|
+
free_nodes, partial_layout, num_physical_qubits, reverse_bit_map, avg_error_map
|
215
|
+
):
|
216
|
+
"""Add any free nodes to a layout."""
|
217
|
+
if not free_nodes:
|
218
|
+
return partial_layout
|
219
|
+
if avg_error_map is not None:
|
220
|
+
free_qubits = sorted(
|
221
|
+
set(range(num_physical_qubits)) - partial_layout.get_physical_bits().keys(),
|
222
|
+
key=lambda bit: avg_error_map.get((bit, bit), 1.0),
|
223
|
+
)
|
224
|
+
# If no error map is available this means there is no scoring heuristic available for this
|
225
|
+
# backend and we can just randomly pick a free qubit
|
226
|
+
else:
|
227
|
+
free_qubits = list(
|
228
|
+
set(range(num_physical_qubits)) - partial_layout.get_physical_bits().keys()
|
229
|
+
)
|
230
|
+
for im_index in sorted(free_nodes, key=lambda x: sum(free_nodes[x].values())):
|
231
|
+
if not free_qubits:
|
232
|
+
return None
|
233
|
+
selected_qubit = free_qubits.pop(0)
|
234
|
+
partial_layout.add(reverse_bit_map[im_index], selected_qubit)
|
235
|
+
return partial_layout
|
@@ -0,0 +1,42 @@
|
|
1
|
+
# This code is part of Qiskit.
|
2
|
+
#
|
3
|
+
# (C) Copyright IBM 2017, 2018.
|
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
|
+
"""Module containing transpiler optimization passes."""
|
14
|
+
|
15
|
+
from .optimize_1q_gates import Optimize1qGates
|
16
|
+
from .optimize_1q_decomposition import Optimize1qGatesDecomposition
|
17
|
+
from .collect_2q_blocks import Collect2qBlocks
|
18
|
+
from .collect_multiqubit_blocks import CollectMultiQBlocks
|
19
|
+
from .consolidate_blocks import ConsolidateBlocks
|
20
|
+
from .commutation_analysis import CommutationAnalysis
|
21
|
+
from .commutative_cancellation import CommutativeCancellation
|
22
|
+
from .commutative_inverse_cancellation import CommutativeInverseCancellation
|
23
|
+
from .optimize_1q_commutation import Optimize1qGatesSimpleCommutation
|
24
|
+
from .optimize_swap_before_measure import OptimizeSwapBeforeMeasure
|
25
|
+
from .remove_reset_in_zero_state import RemoveResetInZeroState
|
26
|
+
from .remove_final_reset import RemoveFinalReset
|
27
|
+
from .remove_diagonal_gates_before_measure import RemoveDiagonalGatesBeforeMeasure
|
28
|
+
from .hoare_opt import HoareOptimizer
|
29
|
+
from .template_optimization import TemplateOptimization
|
30
|
+
from .inverse_cancellation import InverseCancellation
|
31
|
+
from .collect_1q_runs import Collect1qRuns
|
32
|
+
from .collect_linear_functions import CollectLinearFunctions
|
33
|
+
from .reset_after_measure_simplification import ResetAfterMeasureSimplification
|
34
|
+
from .optimize_cliffords import OptimizeCliffords
|
35
|
+
from .collect_cliffords import CollectCliffords
|
36
|
+
from .elide_permutations import ElidePermutations
|
37
|
+
from .optimize_annotated import OptimizeAnnotated
|
38
|
+
from .remove_identity_equiv import RemoveIdentityEquivalent
|
39
|
+
from .split_2q_unitaries import Split2QUnitaries
|
40
|
+
from .collect_and_collapse import CollectAndCollapse
|
41
|
+
from .contract_idle_wires_in_control_flow import ContractIdleWiresInControlFlow
|
42
|
+
from .light_cone import LightCone
|