qiskit 2.1.0rc1__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 +159 -0
- qiskit/_accelerate.abi3.so +0 -0
- qiskit/_numpy_compat.py +73 -0
- qiskit/circuit/__init__.py +1335 -0
- qiskit/circuit/_add_control.py +338 -0
- qiskit/circuit/_classical_resource_map.py +154 -0
- qiskit/circuit/_standard_gates_commutations.py +3849 -0
- qiskit/circuit/_utils.py +167 -0
- qiskit/circuit/annotated_operation.py +279 -0
- qiskit/circuit/annotation.py +404 -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 +156 -0
- qiskit/circuit/classical/expr/visitors.py +381 -0
- qiskit/circuit/classical/types/__init__.py +113 -0
- qiskit/circuit/classical/types/ordering.py +229 -0
- qiskit/circuit/classical/types/types.py +30 -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 +188 -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 +159 -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 +506 -0
- qiskit/circuit/library/arithmetic/piecewise_linear_pauli_rotations.py +395 -0
- qiskit/circuit/library/arithmetic/piecewise_polynomial_pauli_rotations.py +501 -0
- qiskit/circuit/library/arithmetic/polynomial_pauli_rotations.py +389 -0
- qiskit/circuit/library/arithmetic/quadratic_form.py +370 -0
- qiskit/circuit/library/arithmetic/weighted_adder.py +428 -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 +322 -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 +163 -0
- qiskit/circuit/library/generalized_gates/gms.py +179 -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 +202 -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 +236 -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 +301 -0
- qiskit/circuit/library/n_local/n_local.py +1478 -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 +202 -0
- qiskit/circuit/library/phase_estimation.py +177 -0
- qiskit/circuit/library/phase_oracle.py +239 -0
- qiskit/circuit/library/quantum_volume.py +179 -0
- qiskit/circuit/library/standard_gates/__init__.py +141 -0
- qiskit/circuit/library/standard_gates/dcx.py +76 -0
- qiskit/circuit/library/standard_gates/ecr.py +126 -0
- qiskit/circuit/library/standard_gates/equivalence_library.py +1936 -0
- qiskit/circuit/library/standard_gates/global_phase.py +83 -0
- qiskit/circuit/library/standard_gates/h.py +230 -0
- qiskit/circuit/library/standard_gates/i.py +76 -0
- qiskit/circuit/library/standard_gates/iswap.py +115 -0
- qiskit/circuit/library/standard_gates/p.py +415 -0
- qiskit/circuit/library/standard_gates/r.py +108 -0
- qiskit/circuit/library/standard_gates/rx.py +269 -0
- qiskit/circuit/library/standard_gates/rxx.py +165 -0
- qiskit/circuit/library/standard_gates/ry.py +268 -0
- qiskit/circuit/library/standard_gates/ryy.py +165 -0
- qiskit/circuit/library/standard_gates/rz.py +290 -0
- qiskit/circuit/library/standard_gates/rzx.py +211 -0
- qiskit/circuit/library/standard_gates/rzz.py +181 -0
- qiskit/circuit/library/standard_gates/s.py +424 -0
- qiskit/circuit/library/standard_gates/swap.py +268 -0
- qiskit/circuit/library/standard_gates/sx.py +303 -0
- qiskit/circuit/library/standard_gates/t.py +169 -0
- qiskit/circuit/library/standard_gates/u.py +379 -0
- qiskit/circuit/library/standard_gates/u1.py +466 -0
- qiskit/circuit/library/standard_gates/u2.py +145 -0
- qiskit/circuit/library/standard_gates/u3.py +412 -0
- qiskit/circuit/library/standard_gates/x.py +1335 -0
- qiskit/circuit/library/standard_gates/xx_minus_yy.py +164 -0
- qiskit/circuit/library/standard_gates/xx_plus_yy.py +197 -0
- qiskit/circuit/library/standard_gates/y.py +253 -0
- qiskit/circuit/library/standard_gates/z.py +331 -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 +188 -0
- qiskit/circuit/parameterexpression.py +737 -0
- qiskit/circuit/parametertable.py +119 -0
- qiskit/circuit/parametervector.py +140 -0
- qiskit/circuit/quantumcircuit.py +7610 -0
- qiskit/circuit/quantumcircuitdata.py +137 -0
- qiskit/circuit/random/__init__.py +50 -0
- qiskit/circuit/random/utils.py +755 -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 +185 -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 +44 -0
- qiskit/dagcircuit/collect_blocks.py +403 -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 +193 -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 +172 -0
- qiskit/primitives/containers/estimator_pub.py +222 -0
- qiskit/primitives/containers/object_array.py +94 -0
- qiskit/primitives/containers/observables_array.py +380 -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 +100 -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 +376 -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 +466 -0
- qiskit/qasm3/ast.py +796 -0
- qiskit/qasm3/exceptions.py +27 -0
- qiskit/qasm3/experimental.py +70 -0
- qiskit/qasm3/exporter.py +1363 -0
- qiskit/qasm3/printer.py +620 -0
- qiskit/qpy/__init__.py +2141 -0
- qiskit/qpy/binary_io/__init__.py +35 -0
- qiskit/qpy/binary_io/circuits.py +1687 -0
- qiskit/qpy/binary_io/parse_sympy_repr.py +126 -0
- qiskit/qpy/binary_io/schedules.py +288 -0
- qiskit/qpy/binary_io/value.py +1183 -0
- qiskit/qpy/common.py +361 -0
- qiskit/qpy/exceptions.py +53 -0
- qiskit/qpy/formats.py +458 -0
- qiskit/qpy/interface.py +384 -0
- qiskit/qpy/type_keys.py +415 -0
- qiskit/quantum_info/__init__.py +172 -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 +29 -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 +24 -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 +584 -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 +76 -0
- qiskit/result/utils.py +294 -0
- qiskit/synthesis/__init__.py +250 -0
- qiskit/synthesis/arithmetic/__init__.py +18 -0
- qiskit/synthesis/arithmetic/adders/__init__.py +18 -0
- qiskit/synthesis/arithmetic/adders/cdkm_ripple_carry_adder.py +154 -0
- qiskit/synthesis/arithmetic/adders/draper_qft_adder.py +107 -0
- qiskit/synthesis/arithmetic/adders/rv_ripple_carry_adder.py +156 -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/generate_basis_approximations.py +53 -0
- qiskit/synthesis/discrete_basis/solovay_kitaev.py +280 -0
- qiskit/synthesis/evolution/__init__.py +21 -0
- qiskit/synthesis/evolution/evolution_synthesis.py +48 -0
- qiskit/synthesis/evolution/lie_trotter.py +123 -0
- qiskit/synthesis/evolution/matrix_synthesis.py +47 -0
- qiskit/synthesis/evolution/pauli_network.py +80 -0
- qiskit/synthesis/evolution/product_formula.py +316 -0
- qiskit/synthesis/evolution/qdrift.py +133 -0
- qiskit/synthesis/evolution/suzuki_trotter.py +227 -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 +29 -0
- qiskit/synthesis/multi_controlled/mcmt_vchain.py +52 -0
- qiskit/synthesis/multi_controlled/mcx_synthesis.py +583 -0
- qiskit/synthesis/multi_controlled/multi_control_rotation_gates.py +205 -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 +61 -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 +359 -0
- qiskit/transpiler/__init__.py +1352 -0
- qiskit/transpiler/basepasses.py +190 -0
- qiskit/transpiler/coupling.py +500 -0
- qiskit/transpiler/exceptions.py +59 -0
- qiskit/transpiler/instruction_durations.py +263 -0
- qiskit/transpiler/layout.py +740 -0
- qiskit/transpiler/passes/__init__.py +278 -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 +197 -0
- qiskit/transpiler/passes/layout/disjoint_utils.py +54 -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 +525 -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 +292 -0
- qiskit/transpiler/passes/layout/vf2_post_layout.py +376 -0
- qiskit/transpiler/passes/layout/vf2_utils.py +245 -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 +251 -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_clifford_t.py +68 -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 +633 -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 +465 -0
- qiskit/transpiler/passes/routing/star_prerouting.py +433 -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 +17 -0
- qiskit/transpiler/passes/scheduling/padding/base_padding.py +284 -0
- qiskit/transpiler/passes/scheduling/padding/context_aware_dynamical_decoupling.py +876 -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 +21 -0
- qiskit/transpiler/passes/synthesis/aqc_plugin.py +153 -0
- qiskit/transpiler/passes/synthesis/clifford_unitary_synth_plugin.py +123 -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 +2338 -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 +318 -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 +154 -0
- qiskit/transpiler/preset_passmanagers/__init__.py +93 -0
- qiskit/transpiler/preset_passmanagers/builtin_plugins.py +1114 -0
- qiskit/transpiler/preset_passmanagers/common.py +773 -0
- qiskit/transpiler/preset_passmanagers/generate_preset_pass_manager.py +443 -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 +355 -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 +677 -0
- qiskit/visualization/circuit/circuit_visualization.py +735 -0
- qiskit/visualization/circuit/latex.py +668 -0
- qiskit/visualization/circuit/matplotlib.py +2041 -0
- qiskit/visualization/circuit/qcstyle.py +130 -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/__init__.py +13 -0
- qiskit/visualization/dag/dagstyle.py +103 -0
- qiskit/visualization/dag/styles/__init__.py +13 -0
- qiskit/visualization/dag/styles/color.json +10 -0
- qiskit/visualization/dag/styles/plain.json +5 -0
- qiskit/visualization/dag_visualization.py +389 -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/style.py +223 -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.1.0rc1.dist-info/METADATA +221 -0
- qiskit-2.1.0rc1.dist-info/RECORD +699 -0
- qiskit-2.1.0rc1.dist-info/WHEEL +6 -0
- qiskit-2.1.0rc1.dist-info/entry_points.txt +88 -0
- qiskit-2.1.0rc1.dist-info/licenses/LICENSE.txt +203 -0
- qiskit-2.1.0rc1.dist-info/top_level.txt +1 -0
@@ -0,0 +1,60 @@
|
|
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
|
+
"""A gate made of commuting two-qubit gates."""
|
14
|
+
from __future__ import annotations
|
15
|
+
|
16
|
+
from collections.abc import Iterable
|
17
|
+
|
18
|
+
from qiskit.exceptions import QiskitError
|
19
|
+
from qiskit.circuit import Gate, Qubit, Clbit
|
20
|
+
from qiskit.dagcircuit import DAGOpNode
|
21
|
+
|
22
|
+
|
23
|
+
class Commuting2qBlock(Gate):
|
24
|
+
"""A gate made of commuting two-qubit gates.
|
25
|
+
|
26
|
+
This gate is intended for use with commuting swap strategies to make it convenient
|
27
|
+
for the swap strategy router to identify which blocks of operations commute.
|
28
|
+
"""
|
29
|
+
|
30
|
+
def __init__(self, node_block: Iterable[DAGOpNode]) -> None:
|
31
|
+
"""
|
32
|
+
Args:
|
33
|
+
node_block: A block of nodes that commute.
|
34
|
+
|
35
|
+
Raises:
|
36
|
+
QiskitError: If the nodes in the node block do not apply to two-qubits.
|
37
|
+
"""
|
38
|
+
qubits: set[Qubit] = set()
|
39
|
+
cbits: set[Clbit] = set()
|
40
|
+
for node in node_block:
|
41
|
+
if len(node.qargs) != 2:
|
42
|
+
raise QiskitError(f"Node {node.name} does not apply to two-qubits.")
|
43
|
+
|
44
|
+
qubits.update(node.qargs)
|
45
|
+
cbits.update(node.cargs)
|
46
|
+
|
47
|
+
if cbits:
|
48
|
+
raise QiskitError(
|
49
|
+
f"{self.__class__.__name__} does not accept nodes with classical bits."
|
50
|
+
)
|
51
|
+
|
52
|
+
super().__init__(
|
53
|
+
"commuting_2q_block", num_qubits=len(qubits), params=[], label="Commuting 2q gates"
|
54
|
+
)
|
55
|
+
self.node_block = node_block
|
56
|
+
self.qubits = qubits
|
57
|
+
|
58
|
+
def __iter__(self):
|
59
|
+
"""Iterate through the nodes in the block."""
|
60
|
+
return iter(self.node_block)
|
@@ -0,0 +1,397 @@
|
|
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
|
+
"""A swap strategy pass for blocks of commuting gates."""
|
14
|
+
from __future__ import annotations
|
15
|
+
from collections import defaultdict
|
16
|
+
|
17
|
+
from qiskit.circuit import Gate, QuantumCircuit, Qubit
|
18
|
+
from qiskit.converters import circuit_to_dag
|
19
|
+
from qiskit.dagcircuit import DAGCircuit, DAGOpNode
|
20
|
+
from qiskit.transpiler.basepasses import TransformationPass
|
21
|
+
from qiskit.transpiler.exceptions import TranspilerError
|
22
|
+
from qiskit.transpiler.layout import Layout
|
23
|
+
from qiskit.transpiler.passes.routing.commuting_2q_gate_routing.swap_strategy import SwapStrategy
|
24
|
+
from qiskit.transpiler.passes.routing.commuting_2q_gate_routing.commuting_2q_block import (
|
25
|
+
Commuting2qBlock,
|
26
|
+
)
|
27
|
+
|
28
|
+
|
29
|
+
class Commuting2qGateRouter(TransformationPass):
|
30
|
+
"""A class to swap route one or more commuting gates to the coupling map.
|
31
|
+
|
32
|
+
This pass routes blocks of commuting two-qubit gates encapsulated as
|
33
|
+
:class:`.Commuting2qBlock` instructions. This pass will not apply to other instructions.
|
34
|
+
The mapping to the coupling map is done using swap strategies, see :class:`.SwapStrategy`.
|
35
|
+
The swap strategy should suit the problem and the coupling map. This transpiler pass
|
36
|
+
should ideally be executed before the quantum circuit is enlarged with any idle ancilla
|
37
|
+
qubits. Otherwise, we may swap qubits outside the portion of the chip we want to use.
|
38
|
+
Therefore, the swap strategy and its associated coupling map do not represent physical
|
39
|
+
qubits. Instead, they represent an intermediate mapping that corresponds to the physical
|
40
|
+
qubits once the initial layout is applied. The example below shows how to map a four
|
41
|
+
qubit :class:`.PauliEvolutionGate` to qubits 0, 1, 3, and 4 of the five qubit device with
|
42
|
+
the coupling map
|
43
|
+
|
44
|
+
.. code-block:: text
|
45
|
+
|
46
|
+
0 -- 1 -- 2
|
47
|
+
|
|
48
|
+
3
|
49
|
+
|
|
50
|
+
4
|
51
|
+
|
52
|
+
To do this we use a line swap strategy for qubits 0, 1, 3, and 4 defined it in terms
|
53
|
+
of virtual qubits 0, 1, 2, and 3.
|
54
|
+
|
55
|
+
.. plot::
|
56
|
+
:include-source:
|
57
|
+
:nofigs:
|
58
|
+
|
59
|
+
from qiskit import QuantumCircuit
|
60
|
+
from qiskit.circuit.library import PauliEvolutionGate
|
61
|
+
from qiskit.quantum_info import SparsePauliOp
|
62
|
+
from qiskit.transpiler import Layout, CouplingMap, PassManager
|
63
|
+
from qiskit.transpiler.passes import FullAncillaAllocation
|
64
|
+
from qiskit.transpiler.passes import EnlargeWithAncilla
|
65
|
+
from qiskit.transpiler.passes import ApplyLayout
|
66
|
+
from qiskit.transpiler.passes import SetLayout
|
67
|
+
|
68
|
+
from qiskit.transpiler.passes.routing.commuting_2q_gate_routing import (
|
69
|
+
SwapStrategy,
|
70
|
+
FindCommutingPauliEvolutions,
|
71
|
+
Commuting2qGateRouter,
|
72
|
+
)
|
73
|
+
|
74
|
+
# Define the circuit on virtual qubits
|
75
|
+
op = SparsePauliOp.from_list([("IZZI", 1), ("ZIIZ", 2), ("ZIZI", 3)])
|
76
|
+
circ = QuantumCircuit(4)
|
77
|
+
circ.append(PauliEvolutionGate(op, 1), range(4))
|
78
|
+
|
79
|
+
# Define the swap strategy on qubits before the initial_layout is applied.
|
80
|
+
swap_strat = SwapStrategy.from_line([0, 1, 2, 3])
|
81
|
+
|
82
|
+
# Chose qubits 0, 1, 3, and 4 from the backend coupling map shown above.
|
83
|
+
backend_cmap = CouplingMap(couplinglist=[(0, 1), (1, 2), (1, 3), (3, 4)])
|
84
|
+
initial_layout = Layout.from_intlist([0, 1, 3, 4], *circ.qregs)
|
85
|
+
|
86
|
+
pm_pre = PassManager(
|
87
|
+
[
|
88
|
+
FindCommutingPauliEvolutions(),
|
89
|
+
Commuting2qGateRouter(swap_strat),
|
90
|
+
SetLayout(initial_layout),
|
91
|
+
FullAncillaAllocation(backend_cmap),
|
92
|
+
EnlargeWithAncilla(),
|
93
|
+
ApplyLayout(),
|
94
|
+
]
|
95
|
+
)
|
96
|
+
|
97
|
+
# Insert swap gates, map to initial_layout and finally enlarge with ancilla.
|
98
|
+
pm_pre.run(circ).draw("mpl")
|
99
|
+
|
100
|
+
This pass manager relies on the ``current_layout`` which corresponds to the qubit layout as
|
101
|
+
swap gates are applied. The pass will traverse all nodes in the dag. If a node should be
|
102
|
+
routed using a swap strategy then it will be decomposed into sub-instructions with swap
|
103
|
+
layers in between and the ``current_layout`` will be modified. Nodes that should not be
|
104
|
+
routed using swap strategies will be added back to the dag taking the ``current_layout``
|
105
|
+
into account.
|
106
|
+
"""
|
107
|
+
|
108
|
+
def __init__(
|
109
|
+
self,
|
110
|
+
swap_strategy: SwapStrategy | None = None,
|
111
|
+
edge_coloring: dict[tuple[int, int], int] | None = None,
|
112
|
+
) -> None:
|
113
|
+
r"""
|
114
|
+
Args:
|
115
|
+
swap_strategy: An instance of a :class:`.SwapStrategy` that holds the swap layers
|
116
|
+
that are used, and the order in which to apply them, to map the instruction to
|
117
|
+
the hardware. If this field is not given, it should be contained in the
|
118
|
+
property set of the pass. This allows other passes to determine the most
|
119
|
+
appropriate swap strategy at run-time.
|
120
|
+
edge_coloring: An optional edge coloring of the coupling map (I.e. no two edges that
|
121
|
+
share a node have the same color). If the edge coloring is given then the commuting
|
122
|
+
gates that can be simultaneously applied given the current qubit permutation are
|
123
|
+
grouped according to the edge coloring and applied according to this edge
|
124
|
+
coloring. Here, a color is an int which is used as the index to define and
|
125
|
+
access the groups of commuting gates that can be applied simultaneously.
|
126
|
+
If the edge coloring is not given then the sets will be built-up using a
|
127
|
+
greedy algorithm. The edge coloring is useful to position gates such as
|
128
|
+
``RZZGate``\s next to swap gates to exploit CX cancellations.
|
129
|
+
"""
|
130
|
+
super().__init__()
|
131
|
+
self._swap_strategy = swap_strategy
|
132
|
+
self._bit_indices: dict[Qubit, int] | None = None
|
133
|
+
self._edge_coloring = edge_coloring
|
134
|
+
|
135
|
+
def run(self, dag: DAGCircuit) -> DAGCircuit:
|
136
|
+
"""Run the pass by decomposing the nodes it applies on.
|
137
|
+
|
138
|
+
Args:
|
139
|
+
dag: The dag to which we will add swaps.
|
140
|
+
|
141
|
+
Returns:
|
142
|
+
A dag where swaps have been added for the intended gate type.
|
143
|
+
|
144
|
+
Raises:
|
145
|
+
TranspilerError: If the swap strategy was not given at init time and there is
|
146
|
+
no swap strategy in the property set.
|
147
|
+
TranspilerError: If the quantum circuit contains more than one qubit register.
|
148
|
+
TranspilerError: If there are qubits that are not contained in the quantum register.
|
149
|
+
"""
|
150
|
+
if self._swap_strategy is None:
|
151
|
+
swap_strategy = self.property_set["swap_strategy"]
|
152
|
+
|
153
|
+
if swap_strategy is None:
|
154
|
+
raise TranspilerError("No swap strategy given at init or in the property set.")
|
155
|
+
else:
|
156
|
+
swap_strategy = self._swap_strategy
|
157
|
+
|
158
|
+
if len(dag.qregs) != 1:
|
159
|
+
raise TranspilerError(
|
160
|
+
f"{self.__class__.__name__} runs on circuits with one quantum register."
|
161
|
+
)
|
162
|
+
|
163
|
+
if len(dag.qubits) != next(iter(dag.qregs.values())).size:
|
164
|
+
raise TranspilerError("Circuit has qubits not contained in the qubit register.")
|
165
|
+
|
166
|
+
# Fix output permutation -- copied from ElidePermutations
|
167
|
+
input_qubit_mapping = {qubit: index for index, qubit in enumerate(dag.qubits)}
|
168
|
+
self.property_set["original_layout"] = Layout(input_qubit_mapping)
|
169
|
+
if self.property_set["original_qubit_indices"] is None:
|
170
|
+
self.property_set["original_qubit_indices"] = input_qubit_mapping
|
171
|
+
|
172
|
+
new_dag = dag.copy_empty_like()
|
173
|
+
current_layout = Layout.generate_trivial_layout(*dag.qregs.values())
|
174
|
+
|
175
|
+
# Used to keep track of nodes that do not decompose using swap strategies.
|
176
|
+
accumulator = new_dag.copy_empty_like()
|
177
|
+
|
178
|
+
for node in dag.topological_op_nodes():
|
179
|
+
if isinstance(node.op, Commuting2qBlock):
|
180
|
+
|
181
|
+
# Check that the swap strategy creates enough connectivity for the node.
|
182
|
+
self._check_edges(dag, node, swap_strategy)
|
183
|
+
|
184
|
+
# Compose any accumulated non-swap strategy gates to the dag
|
185
|
+
accumulator = self._compose_non_swap_nodes(accumulator, current_layout, new_dag)
|
186
|
+
|
187
|
+
# Decompose the swap-strategy node and add to the dag.
|
188
|
+
new_dag.compose(self.swap_decompose(dag, node, current_layout, swap_strategy))
|
189
|
+
else:
|
190
|
+
accumulator.apply_operation_back(node.op, node.qargs, node.cargs)
|
191
|
+
|
192
|
+
self._compose_non_swap_nodes(accumulator, current_layout, new_dag)
|
193
|
+
|
194
|
+
self.property_set["virtual_permutation_layout"] = current_layout
|
195
|
+
|
196
|
+
return new_dag
|
197
|
+
|
198
|
+
def _compose_non_swap_nodes(
|
199
|
+
self, accumulator: DAGCircuit, layout: Layout, new_dag: DAGCircuit
|
200
|
+
) -> DAGCircuit:
|
201
|
+
"""Add all the non-swap strategy nodes that we have accumulated up to now.
|
202
|
+
|
203
|
+
This method also resets the node accumulator to an empty dag.
|
204
|
+
|
205
|
+
Args:
|
206
|
+
layout: The current layout that keeps track of the swaps.
|
207
|
+
new_dag: The new dag that we are building up.
|
208
|
+
accumulator: A DAG to keep track of nodes that do not decompose
|
209
|
+
using swap strategies.
|
210
|
+
|
211
|
+
Returns:
|
212
|
+
A new accumulator with the same registers as ``new_dag``.
|
213
|
+
"""
|
214
|
+
# Add all the non-swap strategy nodes that we have accumulated up to now.
|
215
|
+
order = layout.reorder_bits(new_dag.qubits)
|
216
|
+
order_bits: list[int | None] = [None] * len(layout)
|
217
|
+
for idx, val in enumerate(order):
|
218
|
+
order_bits[val] = idx
|
219
|
+
|
220
|
+
new_dag.compose(accumulator, qubits=order_bits)
|
221
|
+
|
222
|
+
# Re-initialize the node accumulator
|
223
|
+
return new_dag.copy_empty_like()
|
224
|
+
|
225
|
+
def _position_in_cmap(self, dag: DAGCircuit, j: int, k: int, layout: Layout) -> tuple[int, ...]:
|
226
|
+
"""A helper function to track the movement of virtual qubits through the swaps.
|
227
|
+
|
228
|
+
Args:
|
229
|
+
j: The index of decision variable j (i.e. virtual qubit).
|
230
|
+
k: The index of decision variable k (i.e. virtual qubit).
|
231
|
+
layout: The current layout that takes into account previous swap gates.
|
232
|
+
|
233
|
+
Returns:
|
234
|
+
The position in the coupling map of the virtual qubits j and k as a tuple.
|
235
|
+
"""
|
236
|
+
bit0 = dag.find_bit(layout.get_physical_bits()[j]).index
|
237
|
+
bit1 = dag.find_bit(layout.get_physical_bits()[k]).index
|
238
|
+
|
239
|
+
return bit0, bit1
|
240
|
+
|
241
|
+
def _build_sub_layers(
|
242
|
+
self, current_layer: dict[tuple[int, int], Gate]
|
243
|
+
) -> list[dict[tuple[int, int], Gate]]:
|
244
|
+
"""A helper method to build-up sets of gates to simultaneously apply.
|
245
|
+
|
246
|
+
This is done with an edge coloring if the ``edge_coloring`` init argument was given or with
|
247
|
+
a greedy algorithm if not. With an edge coloring all gates on edges with the same color
|
248
|
+
will be applied simultaneously. These sublayers are applied in the order of their color,
|
249
|
+
which is an int, in increasing color order.
|
250
|
+
|
251
|
+
Args:
|
252
|
+
current_layer: All gates in the current layer can be applied given the qubit ordering
|
253
|
+
of the current layout. However, not all gates in the current layer can be applied
|
254
|
+
simultaneously. This function creates sub-layers by building up sub-layers
|
255
|
+
of gates. All gates in a sub-layer can simultaneously be applied given the coupling
|
256
|
+
map and current qubit configuration.
|
257
|
+
|
258
|
+
Returns:
|
259
|
+
A list of gate dicts that can be applied. The gates a position 0 are applied first.
|
260
|
+
A gate dict has the qubit tuple as key and the gate to apply as value.
|
261
|
+
"""
|
262
|
+
if self._edge_coloring is not None:
|
263
|
+
return self._edge_coloring_build_sub_layers(current_layer)
|
264
|
+
else:
|
265
|
+
return self._greedy_build_sub_layers(current_layer)
|
266
|
+
|
267
|
+
def _edge_coloring_build_sub_layers(
|
268
|
+
self, current_layer: dict[tuple[int, int], Gate]
|
269
|
+
) -> list[dict[tuple[int, int], Gate]]:
|
270
|
+
"""The edge coloring method of building sub-layers of commuting gates."""
|
271
|
+
sub_layers: list[dict[tuple[int, int], Gate]] = [
|
272
|
+
{} for _ in set(self._edge_coloring.values())
|
273
|
+
]
|
274
|
+
for edge, gate in current_layer.items():
|
275
|
+
color = self._edge_coloring[edge]
|
276
|
+
sub_layers[color][edge] = gate
|
277
|
+
|
278
|
+
return sub_layers
|
279
|
+
|
280
|
+
@staticmethod
|
281
|
+
def _greedy_build_sub_layers(
|
282
|
+
current_layer: dict[tuple[int, int], Gate]
|
283
|
+
) -> list[dict[tuple[int, int], Gate]]:
|
284
|
+
"""The greedy method of building sub-layers of commuting gates."""
|
285
|
+
sub_layers = []
|
286
|
+
while len(current_layer) > 0:
|
287
|
+
current_sub_layer, remaining_gates = {}, {}
|
288
|
+
blocked_vertices: set[tuple] = set()
|
289
|
+
|
290
|
+
for edge, evo_gate in current_layer.items():
|
291
|
+
if blocked_vertices.isdisjoint(edge):
|
292
|
+
current_sub_layer[edge] = evo_gate
|
293
|
+
|
294
|
+
# A vertex becomes blocked once a gate is applied to it.
|
295
|
+
blocked_vertices = blocked_vertices.union(edge)
|
296
|
+
else:
|
297
|
+
remaining_gates[edge] = evo_gate
|
298
|
+
|
299
|
+
current_layer = remaining_gates
|
300
|
+
sub_layers.append(current_sub_layer)
|
301
|
+
|
302
|
+
return sub_layers
|
303
|
+
|
304
|
+
def swap_decompose(
|
305
|
+
self, dag: DAGCircuit, node: DAGOpNode, current_layout: Layout, swap_strategy: SwapStrategy
|
306
|
+
) -> DAGCircuit:
|
307
|
+
"""Take an instance of :class:`.Commuting2qBlock` and map it to the coupling map.
|
308
|
+
|
309
|
+
The mapping is done with the swap strategy.
|
310
|
+
|
311
|
+
Args:
|
312
|
+
dag: The dag which contains the :class:`.Commuting2qBlock` we route.
|
313
|
+
node: A node whose operation is a :class:`.Commuting2qBlock`.
|
314
|
+
current_layout: The layout before the swaps are applied. This function will
|
315
|
+
modify the layout so that subsequent gates can be properly composed on the dag.
|
316
|
+
swap_strategy: The swap strategy used to decompose the node.
|
317
|
+
|
318
|
+
Returns:
|
319
|
+
A dag that is compatible with the coupling map where swap gates have been added
|
320
|
+
to map the gates in the :class:`.Commuting2qBlock` to the hardware.
|
321
|
+
"""
|
322
|
+
trivial_layout = Layout.generate_trivial_layout(*dag.qregs.values())
|
323
|
+
gate_layers = self._make_op_layers(dag, node.op, current_layout, swap_strategy)
|
324
|
+
|
325
|
+
# Iterate over and apply gate layers
|
326
|
+
max_distance = max(gate_layers.keys())
|
327
|
+
|
328
|
+
circuit_with_swap = QuantumCircuit(len(dag.qubits))
|
329
|
+
|
330
|
+
for i in range(max_distance + 1):
|
331
|
+
# Get current layer and replace the problem indices j,k by the corresponding
|
332
|
+
# positions in the coupling map. The current layer corresponds
|
333
|
+
# to all the gates that can be applied at the ith swap layer.
|
334
|
+
current_layer = {}
|
335
|
+
for (j, k), local_gate in gate_layers.get(i, {}).items():
|
336
|
+
current_layer[self._position_in_cmap(dag, j, k, current_layout)] = local_gate
|
337
|
+
|
338
|
+
# Not all gates that are applied at the ith swap layer can be applied at the same
|
339
|
+
# time. We therefore greedily build sub-layers.
|
340
|
+
sub_layers = self._build_sub_layers(current_layer)
|
341
|
+
|
342
|
+
# Apply sub-layers
|
343
|
+
for sublayer in sub_layers:
|
344
|
+
for edge, local_gate in sublayer.items():
|
345
|
+
circuit_with_swap.append(local_gate, edge)
|
346
|
+
|
347
|
+
# Apply SWAP gates
|
348
|
+
if i < max_distance:
|
349
|
+
for swap in swap_strategy.swap_layer(i):
|
350
|
+
(j, k) = [trivial_layout.get_physical_bits()[vertex] for vertex in swap]
|
351
|
+
|
352
|
+
circuit_with_swap.swap(j, k)
|
353
|
+
current_layout.swap(j, k)
|
354
|
+
|
355
|
+
return circuit_to_dag(circuit_with_swap)
|
356
|
+
|
357
|
+
def _make_op_layers(
|
358
|
+
self, dag: DAGCircuit, op: Commuting2qBlock, layout: Layout, swap_strategy: SwapStrategy
|
359
|
+
) -> dict[int, dict[tuple, Gate]]:
|
360
|
+
"""Creates layers of two-qubit gates based on the distance in the swap strategy."""
|
361
|
+
|
362
|
+
gate_layers: dict[int, dict[tuple, Gate]] = defaultdict(dict)
|
363
|
+
|
364
|
+
for node in op.node_block:
|
365
|
+
edge = (dag.find_bit(node.qargs[0]).index, dag.find_bit(node.qargs[1]).index)
|
366
|
+
|
367
|
+
bit0 = layout.get_virtual_bits()[dag.qubits[edge[0]]]
|
368
|
+
bit1 = layout.get_virtual_bits()[dag.qubits[edge[1]]]
|
369
|
+
|
370
|
+
distance = swap_strategy.distance_matrix[bit0, bit1]
|
371
|
+
|
372
|
+
gate_layers[distance][edge] = node.op
|
373
|
+
|
374
|
+
return gate_layers
|
375
|
+
|
376
|
+
def _check_edges(self, dag: DAGCircuit, node: DAGOpNode, swap_strategy: SwapStrategy):
|
377
|
+
"""Check if the swap strategy can create the required connectivity.
|
378
|
+
|
379
|
+
Args:
|
380
|
+
node: The dag node for which to check if the swap strategy provides enough connectivity.
|
381
|
+
swap_strategy: The swap strategy that is being used.
|
382
|
+
|
383
|
+
Raises:
|
384
|
+
TranspilerError: If there is an edge that the swap strategy cannot accommodate
|
385
|
+
and if the pass has been configured to raise on such issues.
|
386
|
+
"""
|
387
|
+
required_edges = set()
|
388
|
+
|
389
|
+
for sub_node in node.op:
|
390
|
+
edge = (dag.find_bit(sub_node.qargs[0]).index, dag.find_bit(sub_node.qargs[1]).index)
|
391
|
+
required_edges.add(edge)
|
392
|
+
|
393
|
+
# Check that the swap strategy supports all required edges
|
394
|
+
if not required_edges.issubset(swap_strategy.possible_edges):
|
395
|
+
raise TranspilerError(
|
396
|
+
f"{swap_strategy} cannot implement all edges in {required_edges}."
|
397
|
+
)
|
@@ -0,0 +1,145 @@
|
|
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
|
+
"""An analysis pass to find evolution gates in which the Paulis commute."""
|
14
|
+
|
15
|
+
from typing import Tuple
|
16
|
+
|
17
|
+
import numpy as np
|
18
|
+
|
19
|
+
from qiskit.exceptions import QiskitError
|
20
|
+
from qiskit.circuit.library import PauliEvolutionGate
|
21
|
+
from qiskit.dagcircuit import DAGCircuit
|
22
|
+
from qiskit.transpiler import TransformationPass
|
23
|
+
from qiskit.quantum_info import SparsePauliOp, Pauli
|
24
|
+
from qiskit.transpiler.passes.routing.commuting_2q_gate_routing.commuting_2q_block import (
|
25
|
+
Commuting2qBlock,
|
26
|
+
)
|
27
|
+
|
28
|
+
|
29
|
+
class FindCommutingPauliEvolutions(TransformationPass):
|
30
|
+
"""Finds :class:`.PauliEvolutionGate`s where the operators, that are evolved, all commute."""
|
31
|
+
|
32
|
+
def run(self, dag: DAGCircuit) -> DAGCircuit:
|
33
|
+
"""Check for :class:`.PauliEvolutionGate`s where the summands all commute.
|
34
|
+
|
35
|
+
Args:
|
36
|
+
The DAG circuit in which to look for the commuting evolutions.
|
37
|
+
|
38
|
+
Returns:
|
39
|
+
The dag in which :class:`.PauliEvolutionGate`s made of commuting two-qubit Paulis
|
40
|
+
have been replaced with :class:`.Commuting2qBlocks`` gate instructions. These gates
|
41
|
+
contain nodes of two-qubit :class:`.PauliEvolutionGate`s.
|
42
|
+
"""
|
43
|
+
|
44
|
+
for node in dag.op_nodes():
|
45
|
+
if isinstance(node.op, PauliEvolutionGate):
|
46
|
+
operator = node.op.operator
|
47
|
+
if self.single_qubit_terms_only(operator):
|
48
|
+
continue
|
49
|
+
|
50
|
+
if self.summands_commute(node.op.operator):
|
51
|
+
sub_dag = self._decompose_to_2q(dag, node.op)
|
52
|
+
|
53
|
+
block_op = Commuting2qBlock(set(sub_dag.op_nodes()))
|
54
|
+
wire_order = {
|
55
|
+
wire: idx
|
56
|
+
for idx, wire in enumerate(sub_dag.qubits)
|
57
|
+
if wire not in sub_dag.idle_wires()
|
58
|
+
}
|
59
|
+
dag.replace_block_with_op([node], block_op, wire_order)
|
60
|
+
|
61
|
+
return dag
|
62
|
+
|
63
|
+
@staticmethod
|
64
|
+
def single_qubit_terms_only(operator: SparsePauliOp) -> bool:
|
65
|
+
"""Determine if the Paulis are made of single qubit terms only.
|
66
|
+
|
67
|
+
Args:
|
68
|
+
operator: The operator to check if it consists only of single qubit terms.
|
69
|
+
|
70
|
+
Returns:
|
71
|
+
True if the operator consists of only single qubit terms (like ``IIX + IZI``),
|
72
|
+
and False otherwise.
|
73
|
+
"""
|
74
|
+
|
75
|
+
for pauli in operator.paulis:
|
76
|
+
if sum(np.logical_or(pauli.x, pauli.z)) > 1:
|
77
|
+
return False
|
78
|
+
|
79
|
+
return True
|
80
|
+
|
81
|
+
@staticmethod
|
82
|
+
def summands_commute(operator: SparsePauliOp) -> bool:
|
83
|
+
"""Check if all summands in the evolved operator commute.
|
84
|
+
|
85
|
+
Args:
|
86
|
+
operator: The operator to check if all its summands commute.
|
87
|
+
|
88
|
+
Returns:
|
89
|
+
True if all summands commute, False otherwise.
|
90
|
+
"""
|
91
|
+
# get a list of summands that commute
|
92
|
+
commuting_subparts = operator.paulis.group_qubit_wise_commuting()
|
93
|
+
|
94
|
+
# if all commute we only have one summand!
|
95
|
+
return len(commuting_subparts) == 1
|
96
|
+
|
97
|
+
@staticmethod
|
98
|
+
def _pauli_to_edge(pauli: Pauli) -> Tuple[int, ...]:
|
99
|
+
"""Convert a pauli to an edge.
|
100
|
+
|
101
|
+
Args:
|
102
|
+
pauli: A pauli that is converted to a string to find out where non-identity
|
103
|
+
Paulis are.
|
104
|
+
|
105
|
+
Returns:
|
106
|
+
A tuple representing where the Paulis are. For example, the Pauli "IZIZ" will
|
107
|
+
return (0, 2) since virtual qubits 0 and 2 interact.
|
108
|
+
|
109
|
+
Raises:
|
110
|
+
QiskitError: If the pauli does not exactly have two non-identity terms.
|
111
|
+
"""
|
112
|
+
edge = tuple(np.logical_or(pauli.x, pauli.z).nonzero()[0])
|
113
|
+
|
114
|
+
if len(edge) != 2:
|
115
|
+
raise QiskitError(f"{pauli} does not have length two.")
|
116
|
+
|
117
|
+
return edge
|
118
|
+
|
119
|
+
def _decompose_to_2q(self, dag: DAGCircuit, op: PauliEvolutionGate) -> DAGCircuit:
|
120
|
+
"""Decompose the SparsePauliOp into two-qubit.
|
121
|
+
|
122
|
+
Args:
|
123
|
+
dag: The dag needed to get access to qubits.
|
124
|
+
op: The operator with all the Pauli terms we need to apply.
|
125
|
+
|
126
|
+
Returns:
|
127
|
+
A dag made of two-qubit :class:`.PauliEvolutionGate`.
|
128
|
+
"""
|
129
|
+
sub_dag = dag.copy_empty_like()
|
130
|
+
|
131
|
+
required_paulis = {
|
132
|
+
self._pauli_to_edge(pauli): (pauli, coeff)
|
133
|
+
for pauli, coeff in zip(op.operator.paulis, op.operator.coeffs)
|
134
|
+
}
|
135
|
+
|
136
|
+
for edge, (pauli, coeff) in required_paulis.items():
|
137
|
+
|
138
|
+
qubits = [dag.qubits[edge[0]], dag.qubits[edge[1]]]
|
139
|
+
|
140
|
+
simple_pauli = Pauli(pauli.to_label().replace("I", ""))
|
141
|
+
|
142
|
+
pauli_2q = PauliEvolutionGate(simple_pauli, op.time * np.real(coeff))
|
143
|
+
sub_dag.apply_operation_back(pauli_2q, qubits)
|
144
|
+
|
145
|
+
return sub_dag
|