qiskit 1.0.0b1__cp38-abi3-manylinux_2_17_aarch64.manylinux2014_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.
Files changed (1095) hide show
  1. qiskit/VERSION.txt +1 -0
  2. qiskit/__init__.py +195 -0
  3. qiskit/_accelerate.abi3.so +0 -0
  4. qiskit/_qasm2.abi3.so +0 -0
  5. qiskit/assembler/__init__.py +47 -0
  6. qiskit/assembler/assemble_circuits.py +408 -0
  7. qiskit/assembler/assemble_schedules.py +372 -0
  8. qiskit/assembler/disassemble.py +318 -0
  9. qiskit/assembler/run_config.py +77 -0
  10. qiskit/circuit/__init__.py +425 -0
  11. qiskit/circuit/_classical_resource_map.py +144 -0
  12. qiskit/circuit/_utils.py +170 -0
  13. qiskit/circuit/add_control.py +274 -0
  14. qiskit/circuit/annotated_operation.py +188 -0
  15. qiskit/circuit/barrier.py +51 -0
  16. qiskit/circuit/bit.py +142 -0
  17. qiskit/circuit/classical/__init__.py +41 -0
  18. qiskit/circuit/classical/expr/__init__.py +218 -0
  19. qiskit/circuit/classical/expr/constructors.py +473 -0
  20. qiskit/circuit/classical/expr/expr.py +356 -0
  21. qiskit/circuit/classical/expr/visitors.py +280 -0
  22. qiskit/circuit/classical/types/__init__.py +108 -0
  23. qiskit/circuit/classical/types/ordering.py +222 -0
  24. qiskit/circuit/classical/types/types.py +117 -0
  25. qiskit/circuit/classicalfunction/__init__.py +131 -0
  26. qiskit/circuit/classicalfunction/boolean_expression.py +129 -0
  27. qiskit/circuit/classicalfunction/classical_element.py +54 -0
  28. qiskit/circuit/classicalfunction/classical_function_visitor.py +155 -0
  29. qiskit/circuit/classicalfunction/classicalfunction.py +173 -0
  30. qiskit/circuit/classicalfunction/exceptions.py +35 -0
  31. qiskit/circuit/classicalfunction/types.py +18 -0
  32. qiskit/circuit/classicalfunction/utils.py +91 -0
  33. qiskit/circuit/classicalregister.py +71 -0
  34. qiskit/circuit/commutation_checker.py +176 -0
  35. qiskit/circuit/controlflow/__init__.py +27 -0
  36. qiskit/circuit/controlflow/_builder_utils.py +199 -0
  37. qiskit/circuit/controlflow/break_loop.py +70 -0
  38. qiskit/circuit/controlflow/builder.py +651 -0
  39. qiskit/circuit/controlflow/continue_loop.py +72 -0
  40. qiskit/circuit/controlflow/control_flow.py +52 -0
  41. qiskit/circuit/controlflow/for_loop.py +232 -0
  42. qiskit/circuit/controlflow/if_else.py +517 -0
  43. qiskit/circuit/controlflow/switch_case.py +424 -0
  44. qiskit/circuit/controlflow/while_loop.py +177 -0
  45. qiskit/circuit/controlledgate.py +271 -0
  46. qiskit/circuit/delay.py +104 -0
  47. qiskit/circuit/duration.py +88 -0
  48. qiskit/circuit/equivalence.py +291 -0
  49. qiskit/circuit/equivalence_library.py +18 -0
  50. qiskit/circuit/exceptions.py +19 -0
  51. qiskit/circuit/gate.py +245 -0
  52. qiskit/circuit/instruction.py +655 -0
  53. qiskit/circuit/instructionset.py +191 -0
  54. qiskit/circuit/library/__init__.py +581 -0
  55. qiskit/circuit/library/arithmetic/__init__.py +27 -0
  56. qiskit/circuit/library/arithmetic/adders/__init__.py +17 -0
  57. qiskit/circuit/library/arithmetic/adders/adder.py +58 -0
  58. qiskit/circuit/library/arithmetic/adders/cdkm_ripple_carry_adder.py +159 -0
  59. qiskit/circuit/library/arithmetic/adders/draper_qft_adder.py +116 -0
  60. qiskit/circuit/library/arithmetic/adders/vbe_ripple_carry_adder.py +165 -0
  61. qiskit/circuit/library/arithmetic/exact_reciprocal.py +88 -0
  62. qiskit/circuit/library/arithmetic/functional_pauli_rotations.py +114 -0
  63. qiskit/circuit/library/arithmetic/integer_comparator.py +243 -0
  64. qiskit/circuit/library/arithmetic/linear_amplitude_function.py +196 -0
  65. qiskit/circuit/library/arithmetic/linear_pauli_rotations.py +189 -0
  66. qiskit/circuit/library/arithmetic/multipliers/__init__.py +16 -0
  67. qiskit/circuit/library/arithmetic/multipliers/hrs_cumulative_multiplier.py +138 -0
  68. qiskit/circuit/library/arithmetic/multipliers/multiplier.py +101 -0
  69. qiskit/circuit/library/arithmetic/multipliers/rg_qft_multiplier.py +101 -0
  70. qiskit/circuit/library/arithmetic/piecewise_chebyshev.py +353 -0
  71. qiskit/circuit/library/arithmetic/piecewise_linear_pauli_rotations.py +277 -0
  72. qiskit/circuit/library/arithmetic/piecewise_polynomial_pauli_rotations.py +317 -0
  73. qiskit/circuit/library/arithmetic/polynomial_pauli_rotations.py +335 -0
  74. qiskit/circuit/library/arithmetic/quadratic_form.py +197 -0
  75. qiskit/circuit/library/arithmetic/weighted_adder.py +337 -0
  76. qiskit/circuit/library/basis_change/__init__.py +15 -0
  77. qiskit/circuit/library/basis_change/qft.py +289 -0
  78. qiskit/circuit/library/blueprintcircuit.py +186 -0
  79. qiskit/circuit/library/boolean_logic/__init__.py +18 -0
  80. qiskit/circuit/library/boolean_logic/inner_product.py +78 -0
  81. qiskit/circuit/library/boolean_logic/quantum_and.py +97 -0
  82. qiskit/circuit/library/boolean_logic/quantum_or.py +98 -0
  83. qiskit/circuit/library/boolean_logic/quantum_xor.py +71 -0
  84. qiskit/circuit/library/data_preparation/__init__.py +47 -0
  85. qiskit/circuit/library/data_preparation/initializer.py +96 -0
  86. qiskit/circuit/library/data_preparation/pauli_feature_map.py +296 -0
  87. qiskit/circuit/library/data_preparation/state_preparation.py +526 -0
  88. qiskit/circuit/library/data_preparation/z_feature_map.py +104 -0
  89. qiskit/circuit/library/data_preparation/zz_feature_map.py +114 -0
  90. qiskit/circuit/library/evolved_operator_ansatz.py +245 -0
  91. qiskit/circuit/library/fourier_checking.py +97 -0
  92. qiskit/circuit/library/generalized_gates/__init__.py +30 -0
  93. qiskit/circuit/library/generalized_gates/diagonal.py +164 -0
  94. qiskit/circuit/library/generalized_gates/gms.py +121 -0
  95. qiskit/circuit/library/generalized_gates/gr.py +215 -0
  96. qiskit/circuit/library/generalized_gates/isometry.py +573 -0
  97. qiskit/circuit/library/generalized_gates/linear_function.py +302 -0
  98. qiskit/circuit/library/generalized_gates/mcg_up_to_diagonal.py +138 -0
  99. qiskit/circuit/library/generalized_gates/mcmt.py +254 -0
  100. qiskit/circuit/library/generalized_gates/pauli.py +85 -0
  101. qiskit/circuit/library/generalized_gates/permutation.py +190 -0
  102. qiskit/circuit/library/generalized_gates/rv.py +93 -0
  103. qiskit/circuit/library/generalized_gates/uc.py +304 -0
  104. qiskit/circuit/library/generalized_gates/uc_pauli_rot.py +164 -0
  105. qiskit/circuit/library/generalized_gates/ucrx.py +32 -0
  106. qiskit/circuit/library/generalized_gates/ucry.py +32 -0
  107. qiskit/circuit/library/generalized_gates/ucrz.py +32 -0
  108. qiskit/circuit/library/generalized_gates/unitary.py +198 -0
  109. qiskit/circuit/library/graph_state.py +86 -0
  110. qiskit/circuit/library/grover_operator.py +311 -0
  111. qiskit/circuit/library/hamiltonian_gate.py +146 -0
  112. qiskit/circuit/library/hidden_linear_function.py +98 -0
  113. qiskit/circuit/library/iqp.py +96 -0
  114. qiskit/circuit/library/n_local/__init__.py +31 -0
  115. qiskit/circuit/library/n_local/efficient_su2.py +162 -0
  116. qiskit/circuit/library/n_local/excitation_preserving.py +176 -0
  117. qiskit/circuit/library/n_local/n_local.py +1044 -0
  118. qiskit/circuit/library/n_local/pauli_two_design.py +131 -0
  119. qiskit/circuit/library/n_local/qaoa_ansatz.py +288 -0
  120. qiskit/circuit/library/n_local/real_amplitudes.py +189 -0
  121. qiskit/circuit/library/n_local/two_local.py +334 -0
  122. qiskit/circuit/library/overlap.py +111 -0
  123. qiskit/circuit/library/pauli_evolution.py +180 -0
  124. qiskit/circuit/library/phase_estimation.py +99 -0
  125. qiskit/circuit/library/phase_oracle.py +153 -0
  126. qiskit/circuit/library/quantum_volume.py +114 -0
  127. qiskit/circuit/library/standard_gates/__init__.py +116 -0
  128. qiskit/circuit/library/standard_gates/dcx.py +71 -0
  129. qiskit/circuit/library/standard_gates/ecr.py +114 -0
  130. qiskit/circuit/library/standard_gates/equivalence_library.py +1677 -0
  131. qiskit/circuit/library/standard_gates/global_phase.py +63 -0
  132. qiskit/circuit/library/standard_gates/h.py +224 -0
  133. qiskit/circuit/library/standard_gates/i.py +60 -0
  134. qiskit/circuit/library/standard_gates/iswap.py +129 -0
  135. qiskit/circuit/library/standard_gates/multi_control_rotation_gates.py +390 -0
  136. qiskit/circuit/library/standard_gates/p.py +364 -0
  137. qiskit/circuit/library/standard_gates/r.py +101 -0
  138. qiskit/circuit/library/standard_gates/rx.py +246 -0
  139. qiskit/circuit/library/standard_gates/rxx.py +128 -0
  140. qiskit/circuit/library/standard_gates/ry.py +241 -0
  141. qiskit/circuit/library/standard_gates/ryy.py +128 -0
  142. qiskit/circuit/library/standard_gates/rz.py +261 -0
  143. qiskit/circuit/library/standard_gates/rzx.py +174 -0
  144. qiskit/circuit/library/standard_gates/rzz.py +141 -0
  145. qiskit/circuit/library/standard_gates/s.py +303 -0
  146. qiskit/circuit/library/standard_gates/swap.py +246 -0
  147. qiskit/circuit/library/standard_gates/sx.py +268 -0
  148. qiskit/circuit/library/standard_gates/t.py +150 -0
  149. qiskit/circuit/library/standard_gates/u.py +338 -0
  150. qiskit/circuit/library/standard_gates/u1.py +383 -0
  151. qiskit/circuit/library/standard_gates/u2.py +132 -0
  152. qiskit/circuit/library/standard_gates/u3.py +358 -0
  153. qiskit/circuit/library/standard_gates/x.py +1370 -0
  154. qiskit/circuit/library/standard_gates/xx_minus_yy.py +179 -0
  155. qiskit/circuit/library/standard_gates/xx_plus_yy.py +180 -0
  156. qiskit/circuit/library/standard_gates/y.py +221 -0
  157. qiskit/circuit/library/standard_gates/z.py +294 -0
  158. qiskit/circuit/library/templates/__init__.py +92 -0
  159. qiskit/circuit/library/templates/clifford/__init__.py +33 -0
  160. qiskit/circuit/library/templates/clifford/clifford_2_1.py +33 -0
  161. qiskit/circuit/library/templates/clifford/clifford_2_2.py +34 -0
  162. qiskit/circuit/library/templates/clifford/clifford_2_3.py +32 -0
  163. qiskit/circuit/library/templates/clifford/clifford_2_4.py +33 -0
  164. qiskit/circuit/library/templates/clifford/clifford_3_1.py +34 -0
  165. qiskit/circuit/library/templates/clifford/clifford_4_1.py +37 -0
  166. qiskit/circuit/library/templates/clifford/clifford_4_2.py +36 -0
  167. qiskit/circuit/library/templates/clifford/clifford_4_3.py +37 -0
  168. qiskit/circuit/library/templates/clifford/clifford_4_4.py +36 -0
  169. qiskit/circuit/library/templates/clifford/clifford_5_1.py +39 -0
  170. qiskit/circuit/library/templates/clifford/clifford_6_1.py +39 -0
  171. qiskit/circuit/library/templates/clifford/clifford_6_2.py +39 -0
  172. qiskit/circuit/library/templates/clifford/clifford_6_3.py +39 -0
  173. qiskit/circuit/library/templates/clifford/clifford_6_4.py +37 -0
  174. qiskit/circuit/library/templates/clifford/clifford_6_5.py +39 -0
  175. qiskit/circuit/library/templates/clifford/clifford_8_1.py +41 -0
  176. qiskit/circuit/library/templates/clifford/clifford_8_2.py +41 -0
  177. qiskit/circuit/library/templates/clifford/clifford_8_3.py +40 -0
  178. qiskit/circuit/library/templates/nct/__init__.py +67 -0
  179. qiskit/circuit/library/templates/nct/template_nct_2a_1.py +32 -0
  180. qiskit/circuit/library/templates/nct/template_nct_2a_2.py +33 -0
  181. qiskit/circuit/library/templates/nct/template_nct_2a_3.py +35 -0
  182. qiskit/circuit/library/templates/nct/template_nct_4a_1.py +41 -0
  183. qiskit/circuit/library/templates/nct/template_nct_4a_2.py +39 -0
  184. qiskit/circuit/library/templates/nct/template_nct_4a_3.py +37 -0
  185. qiskit/circuit/library/templates/nct/template_nct_4b_1.py +39 -0
  186. qiskit/circuit/library/templates/nct/template_nct_4b_2.py +37 -0
  187. qiskit/circuit/library/templates/nct/template_nct_5a_1.py +38 -0
  188. qiskit/circuit/library/templates/nct/template_nct_5a_2.py +38 -0
  189. qiskit/circuit/library/templates/nct/template_nct_5a_3.py +38 -0
  190. qiskit/circuit/library/templates/nct/template_nct_5a_4.py +37 -0
  191. qiskit/circuit/library/templates/nct/template_nct_6a_1.py +38 -0
  192. qiskit/circuit/library/templates/nct/template_nct_6a_2.py +39 -0
  193. qiskit/circuit/library/templates/nct/template_nct_6a_3.py +39 -0
  194. qiskit/circuit/library/templates/nct/template_nct_6a_4.py +39 -0
  195. qiskit/circuit/library/templates/nct/template_nct_6b_1.py +39 -0
  196. qiskit/circuit/library/templates/nct/template_nct_6b_2.py +39 -0
  197. qiskit/circuit/library/templates/nct/template_nct_6c_1.py +39 -0
  198. qiskit/circuit/library/templates/nct/template_nct_7a_1.py +41 -0
  199. qiskit/circuit/library/templates/nct/template_nct_7b_1.py +41 -0
  200. qiskit/circuit/library/templates/nct/template_nct_7c_1.py +41 -0
  201. qiskit/circuit/library/templates/nct/template_nct_7d_1.py +41 -0
  202. qiskit/circuit/library/templates/nct/template_nct_7e_1.py +41 -0
  203. qiskit/circuit/library/templates/nct/template_nct_9a_1.py +43 -0
  204. qiskit/circuit/library/templates/nct/template_nct_9c_1.py +41 -0
  205. qiskit/circuit/library/templates/nct/template_nct_9c_10.py +42 -0
  206. qiskit/circuit/library/templates/nct/template_nct_9c_11.py +42 -0
  207. qiskit/circuit/library/templates/nct/template_nct_9c_12.py +42 -0
  208. qiskit/circuit/library/templates/nct/template_nct_9c_2.py +42 -0
  209. qiskit/circuit/library/templates/nct/template_nct_9c_3.py +42 -0
  210. qiskit/circuit/library/templates/nct/template_nct_9c_4.py +42 -0
  211. qiskit/circuit/library/templates/nct/template_nct_9c_5.py +42 -0
  212. qiskit/circuit/library/templates/nct/template_nct_9c_6.py +42 -0
  213. qiskit/circuit/library/templates/nct/template_nct_9c_7.py +42 -0
  214. qiskit/circuit/library/templates/nct/template_nct_9c_8.py +42 -0
  215. qiskit/circuit/library/templates/nct/template_nct_9c_9.py +42 -0
  216. qiskit/circuit/library/templates/nct/template_nct_9d_1.py +41 -0
  217. qiskit/circuit/library/templates/nct/template_nct_9d_10.py +42 -0
  218. qiskit/circuit/library/templates/nct/template_nct_9d_2.py +42 -0
  219. qiskit/circuit/library/templates/nct/template_nct_9d_3.py +42 -0
  220. qiskit/circuit/library/templates/nct/template_nct_9d_4.py +42 -0
  221. qiskit/circuit/library/templates/nct/template_nct_9d_5.py +42 -0
  222. qiskit/circuit/library/templates/nct/template_nct_9d_6.py +42 -0
  223. qiskit/circuit/library/templates/nct/template_nct_9d_7.py +42 -0
  224. qiskit/circuit/library/templates/nct/template_nct_9d_8.py +42 -0
  225. qiskit/circuit/library/templates/nct/template_nct_9d_9.py +42 -0
  226. qiskit/circuit/library/templates/rzx/__init__.py +25 -0
  227. qiskit/circuit/library/templates/rzx/rzx_cy.py +46 -0
  228. qiskit/circuit/library/templates/rzx/rzx_xz.py +53 -0
  229. qiskit/circuit/library/templates/rzx/rzx_yz.py +43 -0
  230. qiskit/circuit/library/templates/rzx/rzx_zz1.py +67 -0
  231. qiskit/circuit/library/templates/rzx/rzx_zz2.py +58 -0
  232. qiskit/circuit/library/templates/rzx/rzx_zz3.py +57 -0
  233. qiskit/circuit/measure.py +41 -0
  234. qiskit/circuit/operation.py +62 -0
  235. qiskit/circuit/parameter.py +160 -0
  236. qiskit/circuit/parameterexpression.py +515 -0
  237. qiskit/circuit/parametertable.py +263 -0
  238. qiskit/circuit/parametervector.py +114 -0
  239. qiskit/circuit/qpy_serialization.py +28 -0
  240. qiskit/circuit/quantumcircuit.py +6074 -0
  241. qiskit/circuit/quantumcircuitdata.py +138 -0
  242. qiskit/circuit/quantumregister.py +90 -0
  243. qiskit/circuit/random/__init__.py +15 -0
  244. qiskit/circuit/random/utils.py +209 -0
  245. qiskit/circuit/register.py +256 -0
  246. qiskit/circuit/reset.py +31 -0
  247. qiskit/circuit/singleton.py +604 -0
  248. qiskit/circuit/store.py +87 -0
  249. qiskit/circuit/tools/__init__.py +16 -0
  250. qiskit/circuit/tools/pi_check.py +190 -0
  251. qiskit/compiler/__init__.py +33 -0
  252. qiskit/compiler/assembler.py +597 -0
  253. qiskit/compiler/scheduler.py +107 -0
  254. qiskit/compiler/sequencer.py +69 -0
  255. qiskit/compiler/transpiler.py +613 -0
  256. qiskit/converters/__init__.py +59 -0
  257. qiskit/converters/circuit_to_dag.py +96 -0
  258. qiskit/converters/circuit_to_dagdependency.py +51 -0
  259. qiskit/converters/circuit_to_gate.py +109 -0
  260. qiskit/converters/circuit_to_instruction.py +131 -0
  261. qiskit/converters/dag_to_circuit.py +77 -0
  262. qiskit/converters/dag_to_dagdependency.py +55 -0
  263. qiskit/converters/dagdependency_to_circuit.py +42 -0
  264. qiskit/converters/dagdependency_to_dag.py +49 -0
  265. qiskit/dagcircuit/__init__.py +44 -0
  266. qiskit/dagcircuit/collect_blocks.py +386 -0
  267. qiskit/dagcircuit/dagcircuit.py +2105 -0
  268. qiskit/dagcircuit/dagdependency.py +626 -0
  269. qiskit/dagcircuit/dagdepnode.py +157 -0
  270. qiskit/dagcircuit/dagnode.py +322 -0
  271. qiskit/dagcircuit/exceptions.py +42 -0
  272. qiskit/exceptions.py +97 -0
  273. qiskit/execute_function.py +354 -0
  274. qiskit/extensions/__init__.py +70 -0
  275. qiskit/extensions/exceptions.py +31 -0
  276. qiskit/extensions/quantum_initializer/__init__.py +26 -0
  277. qiskit/extensions/quantum_initializer/squ.py +163 -0
  278. qiskit/extensions/simulator/__init__.py +15 -0
  279. qiskit/extensions/simulator/snapshot.py +70 -0
  280. qiskit/namespace.py +76 -0
  281. qiskit/passmanager/__init__.py +242 -0
  282. qiskit/passmanager/base_tasks.py +230 -0
  283. qiskit/passmanager/compilation_status.py +74 -0
  284. qiskit/passmanager/exceptions.py +19 -0
  285. qiskit/passmanager/flow_controllers.py +336 -0
  286. qiskit/passmanager/passmanager.py +317 -0
  287. qiskit/primitives/__init__.py +62 -0
  288. qiskit/primitives/backend_estimator.py +473 -0
  289. qiskit/primitives/backend_sampler.py +209 -0
  290. qiskit/primitives/base/__init__.py +20 -0
  291. qiskit/primitives/base/base_estimator.py +256 -0
  292. qiskit/primitives/base/base_primitive.py +74 -0
  293. qiskit/primitives/base/base_result.py +87 -0
  294. qiskit/primitives/base/base_sampler.py +202 -0
  295. qiskit/primitives/base/estimator_result.py +46 -0
  296. qiskit/primitives/base/sampler_result.py +45 -0
  297. qiskit/primitives/base/validation.py +231 -0
  298. qiskit/primitives/estimator.py +158 -0
  299. qiskit/primitives/primitive_job.py +73 -0
  300. qiskit/primitives/sampler.py +155 -0
  301. qiskit/primitives/utils.py +216 -0
  302. qiskit/providers/__init__.py +773 -0
  303. qiskit/providers/backend.py +653 -0
  304. qiskit/providers/backend_compat.py +347 -0
  305. qiskit/providers/basicaer/__init__.py +73 -0
  306. qiskit/providers/basicaer/basicaerjob.py +65 -0
  307. qiskit/providers/basicaer/basicaerprovider.py +127 -0
  308. qiskit/providers/basicaer/basicaertools.py +186 -0
  309. qiskit/providers/basicaer/exceptions.py +30 -0
  310. qiskit/providers/basicaer/qasm_simulator.py +678 -0
  311. qiskit/providers/basicaer/statevector_simulator.py +121 -0
  312. qiskit/providers/basicaer/unitary_simulator.py +395 -0
  313. qiskit/providers/exceptions.py +45 -0
  314. qiskit/providers/fake_provider/__init__.py +267 -0
  315. qiskit/providers/fake_provider/backends/__init__.py +110 -0
  316. qiskit/providers/fake_provider/backends/almaden/__init__.py +16 -0
  317. qiskit/providers/fake_provider/backends/almaden/conf_almaden.json +1 -0
  318. qiskit/providers/fake_provider/backends/almaden/fake_almaden.py +58 -0
  319. qiskit/providers/fake_provider/backends/almaden/props_almaden.json +1 -0
  320. qiskit/providers/fake_provider/backends/armonk/__init__.py +16 -0
  321. qiskit/providers/fake_provider/backends/armonk/conf_armonk.json +1 -0
  322. qiskit/providers/fake_provider/backends/armonk/defs_armonk.json +1 -0
  323. qiskit/providers/fake_provider/backends/armonk/fake_armonk.py +48 -0
  324. qiskit/providers/fake_provider/backends/armonk/props_armonk.json +1 -0
  325. qiskit/providers/fake_provider/backends/athens/__init__.py +16 -0
  326. qiskit/providers/fake_provider/backends/athens/conf_athens.json +1 -0
  327. qiskit/providers/fake_provider/backends/athens/defs_athens.json +1 -0
  328. qiskit/providers/fake_provider/backends/athens/fake_athens.py +38 -0
  329. qiskit/providers/fake_provider/backends/athens/props_athens.json +1 -0
  330. qiskit/providers/fake_provider/backends/auckland/__init__.py +15 -0
  331. qiskit/providers/fake_provider/backends/auckland/conf_auckland.json +1 -0
  332. qiskit/providers/fake_provider/backends/auckland/defs_auckland.json +1 -0
  333. qiskit/providers/fake_provider/backends/auckland/fake_auckland.py +29 -0
  334. qiskit/providers/fake_provider/backends/auckland/props_auckland.json +1 -0
  335. qiskit/providers/fake_provider/backends/belem/__init__.py +16 -0
  336. qiskit/providers/fake_provider/backends/belem/conf_belem.json +1 -0
  337. qiskit/providers/fake_provider/backends/belem/defs_belem.json +1 -0
  338. qiskit/providers/fake_provider/backends/belem/fake_belem.py +38 -0
  339. qiskit/providers/fake_provider/backends/belem/props_belem.json +1 -0
  340. qiskit/providers/fake_provider/backends/boeblingen/__init__.py +16 -0
  341. qiskit/providers/fake_provider/backends/boeblingen/conf_boeblingen.json +1 -0
  342. qiskit/providers/fake_provider/backends/boeblingen/defs_boeblingen.json +1 -0
  343. qiskit/providers/fake_provider/backends/boeblingen/fake_boeblingen.py +60 -0
  344. qiskit/providers/fake_provider/backends/boeblingen/props_boeblingen.json +1 -0
  345. qiskit/providers/fake_provider/backends/bogota/__init__.py +16 -0
  346. qiskit/providers/fake_provider/backends/bogota/conf_bogota.json +1 -0
  347. qiskit/providers/fake_provider/backends/bogota/defs_bogota.json +1 -0
  348. qiskit/providers/fake_provider/backends/bogota/fake_bogota.py +38 -0
  349. qiskit/providers/fake_provider/backends/bogota/props_bogota.json +1 -0
  350. qiskit/providers/fake_provider/backends/brooklyn/__init__.py +16 -0
  351. qiskit/providers/fake_provider/backends/brooklyn/conf_brooklyn.json +1 -0
  352. qiskit/providers/fake_provider/backends/brooklyn/defs_brooklyn.json +1 -0
  353. qiskit/providers/fake_provider/backends/brooklyn/fake_brooklyn.py +38 -0
  354. qiskit/providers/fake_provider/backends/brooklyn/props_brooklyn.json +1 -0
  355. qiskit/providers/fake_provider/backends/burlington/__init__.py +16 -0
  356. qiskit/providers/fake_provider/backends/burlington/conf_burlington.json +1 -0
  357. qiskit/providers/fake_provider/backends/burlington/fake_burlington.py +50 -0
  358. qiskit/providers/fake_provider/backends/burlington/props_burlington.json +1 -0
  359. qiskit/providers/fake_provider/backends/cairo/__init__.py +16 -0
  360. qiskit/providers/fake_provider/backends/cairo/conf_cairo.json +1 -0
  361. qiskit/providers/fake_provider/backends/cairo/defs_cairo.json +1 -0
  362. qiskit/providers/fake_provider/backends/cairo/fake_cairo.py +38 -0
  363. qiskit/providers/fake_provider/backends/cairo/props_cairo.json +1 -0
  364. qiskit/providers/fake_provider/backends/cambridge/__init__.py +17 -0
  365. qiskit/providers/fake_provider/backends/cambridge/conf_cambridge.json +1 -0
  366. qiskit/providers/fake_provider/backends/cambridge/fake_cambridge.py +72 -0
  367. qiskit/providers/fake_provider/backends/cambridge/props_cambridge.json +1 -0
  368. qiskit/providers/fake_provider/backends/cambridge/props_cambridge_alt.json +1 -0
  369. qiskit/providers/fake_provider/backends/casablanca/__init__.py +16 -0
  370. qiskit/providers/fake_provider/backends/casablanca/conf_casablanca.json +1 -0
  371. qiskit/providers/fake_provider/backends/casablanca/defs_casablanca.json +1 -0
  372. qiskit/providers/fake_provider/backends/casablanca/fake_casablanca.py +38 -0
  373. qiskit/providers/fake_provider/backends/casablanca/props_casablanca.json +1 -0
  374. qiskit/providers/fake_provider/backends/essex/__init__.py +16 -0
  375. qiskit/providers/fake_provider/backends/essex/conf_essex.json +1 -0
  376. qiskit/providers/fake_provider/backends/essex/fake_essex.py +54 -0
  377. qiskit/providers/fake_provider/backends/essex/props_essex.json +1 -0
  378. qiskit/providers/fake_provider/backends/geneva/__init__.py +15 -0
  379. qiskit/providers/fake_provider/backends/geneva/conf_geneva.json +1 -0
  380. qiskit/providers/fake_provider/backends/geneva/defs_geneva.json +1 -0
  381. qiskit/providers/fake_provider/backends/geneva/fake_geneva.py +29 -0
  382. qiskit/providers/fake_provider/backends/geneva/props_geneva.json +1 -0
  383. qiskit/providers/fake_provider/backends/guadalupe/__init__.py +16 -0
  384. qiskit/providers/fake_provider/backends/guadalupe/conf_guadalupe.json +1 -0
  385. qiskit/providers/fake_provider/backends/guadalupe/defs_guadalupe.json +1 -0
  386. qiskit/providers/fake_provider/backends/guadalupe/fake_guadalupe.py +39 -0
  387. qiskit/providers/fake_provider/backends/guadalupe/props_guadalupe.json +1 -0
  388. qiskit/providers/fake_provider/backends/hanoi/__init__.py +16 -0
  389. qiskit/providers/fake_provider/backends/hanoi/conf_hanoi.json +1 -0
  390. qiskit/providers/fake_provider/backends/hanoi/defs_hanoi.json +1 -0
  391. qiskit/providers/fake_provider/backends/hanoi/fake_hanoi.py +38 -0
  392. qiskit/providers/fake_provider/backends/hanoi/props_hanoi.json +1 -0
  393. qiskit/providers/fake_provider/backends/jakarta/__init__.py +16 -0
  394. qiskit/providers/fake_provider/backends/jakarta/conf_jakarta.json +1 -0
  395. qiskit/providers/fake_provider/backends/jakarta/defs_jakarta.json +1 -0
  396. qiskit/providers/fake_provider/backends/jakarta/fake_jakarta.py +38 -0
  397. qiskit/providers/fake_provider/backends/jakarta/props_jakarta.json +1 -0
  398. qiskit/providers/fake_provider/backends/johannesburg/__init__.py +16 -0
  399. qiskit/providers/fake_provider/backends/johannesburg/conf_johannesburg.json +1 -0
  400. qiskit/providers/fake_provider/backends/johannesburg/fake_johannesburg.py +58 -0
  401. qiskit/providers/fake_provider/backends/johannesburg/props_johannesburg.json +1 -0
  402. qiskit/providers/fake_provider/backends/kolkata/__init__.py +16 -0
  403. qiskit/providers/fake_provider/backends/kolkata/conf_kolkata.json +1 -0
  404. qiskit/providers/fake_provider/backends/kolkata/defs_kolkata.json +1 -0
  405. qiskit/providers/fake_provider/backends/kolkata/fake_kolkata.py +38 -0
  406. qiskit/providers/fake_provider/backends/kolkata/props_kolkata.json +1 -0
  407. qiskit/providers/fake_provider/backends/lagos/__init__.py +16 -0
  408. qiskit/providers/fake_provider/backends/lagos/conf_lagos.json +1 -0
  409. qiskit/providers/fake_provider/backends/lagos/defs_lagos.json +1 -0
  410. qiskit/providers/fake_provider/backends/lagos/fake_lagos.py +38 -0
  411. qiskit/providers/fake_provider/backends/lagos/props_lagos.json +1 -0
  412. qiskit/providers/fake_provider/backends/lima/__init__.py +16 -0
  413. qiskit/providers/fake_provider/backends/lima/conf_lima.json +1 -0
  414. qiskit/providers/fake_provider/backends/lima/defs_lima.json +1 -0
  415. qiskit/providers/fake_provider/backends/lima/fake_lima.py +38 -0
  416. qiskit/providers/fake_provider/backends/lima/props_lima.json +1 -0
  417. qiskit/providers/fake_provider/backends/london/__init__.py +16 -0
  418. qiskit/providers/fake_provider/backends/london/conf_london.json +1 -0
  419. qiskit/providers/fake_provider/backends/london/fake_london.py +54 -0
  420. qiskit/providers/fake_provider/backends/london/props_london.json +1 -0
  421. qiskit/providers/fake_provider/backends/manhattan/__init__.py +16 -0
  422. qiskit/providers/fake_provider/backends/manhattan/conf_manhattan.json +1 -0
  423. qiskit/providers/fake_provider/backends/manhattan/defs_manhattan.json +1 -0
  424. qiskit/providers/fake_provider/backends/manhattan/fake_manhattan.py +38 -0
  425. qiskit/providers/fake_provider/backends/manhattan/props_manhattan.json +1 -0
  426. qiskit/providers/fake_provider/backends/manila/__init__.py +16 -0
  427. qiskit/providers/fake_provider/backends/manila/conf_manila.json +1 -0
  428. qiskit/providers/fake_provider/backends/manila/defs_manila.json +1 -0
  429. qiskit/providers/fake_provider/backends/manila/fake_manila.py +38 -0
  430. qiskit/providers/fake_provider/backends/manila/props_manila.json +1 -0
  431. qiskit/providers/fake_provider/backends/melbourne/__init__.py +16 -0
  432. qiskit/providers/fake_provider/backends/melbourne/conf_melbourne.json +1 -0
  433. qiskit/providers/fake_provider/backends/melbourne/fake_melbourne.py +91 -0
  434. qiskit/providers/fake_provider/backends/melbourne/props_melbourne.json +1 -0
  435. qiskit/providers/fake_provider/backends/montreal/__init__.py +16 -0
  436. qiskit/providers/fake_provider/backends/montreal/conf_montreal.json +1 -0
  437. qiskit/providers/fake_provider/backends/montreal/defs_montreal.json +1 -0
  438. qiskit/providers/fake_provider/backends/montreal/fake_montreal.py +38 -0
  439. qiskit/providers/fake_provider/backends/montreal/props_montreal.json +1 -0
  440. qiskit/providers/fake_provider/backends/mumbai/__init__.py +16 -0
  441. qiskit/providers/fake_provider/backends/mumbai/conf_mumbai.json +1 -0
  442. qiskit/providers/fake_provider/backends/mumbai/defs_mumbai.json +1 -0
  443. qiskit/providers/fake_provider/backends/mumbai/fake_mumbai.py +38 -0
  444. qiskit/providers/fake_provider/backends/mumbai/props_mumbai.json +1 -0
  445. qiskit/providers/fake_provider/backends/nairobi/__init__.py +16 -0
  446. qiskit/providers/fake_provider/backends/nairobi/conf_nairobi.json +1 -0
  447. qiskit/providers/fake_provider/backends/nairobi/defs_nairobi.json +1 -0
  448. qiskit/providers/fake_provider/backends/nairobi/fake_nairobi.py +38 -0
  449. qiskit/providers/fake_provider/backends/nairobi/props_nairobi.json +1 -0
  450. qiskit/providers/fake_provider/backends/oslo/__init__.py +15 -0
  451. qiskit/providers/fake_provider/backends/oslo/conf_oslo.json +1 -0
  452. qiskit/providers/fake_provider/backends/oslo/defs_oslo.json +1 -0
  453. qiskit/providers/fake_provider/backends/oslo/fake_oslo.py +29 -0
  454. qiskit/providers/fake_provider/backends/oslo/props_oslo.json +1 -0
  455. qiskit/providers/fake_provider/backends/ourense/__init__.py +16 -0
  456. qiskit/providers/fake_provider/backends/ourense/conf_ourense.json +1 -0
  457. qiskit/providers/fake_provider/backends/ourense/fake_ourense.py +50 -0
  458. qiskit/providers/fake_provider/backends/ourense/props_ourense.json +1 -0
  459. qiskit/providers/fake_provider/backends/paris/__init__.py +16 -0
  460. qiskit/providers/fake_provider/backends/paris/conf_paris.json +1 -0
  461. qiskit/providers/fake_provider/backends/paris/defs_paris.json +1 -0
  462. qiskit/providers/fake_provider/backends/paris/fake_paris.py +64 -0
  463. qiskit/providers/fake_provider/backends/paris/props_paris.json +1 -0
  464. qiskit/providers/fake_provider/backends/perth/__init__.py +15 -0
  465. qiskit/providers/fake_provider/backends/perth/conf_perth.json +1 -0
  466. qiskit/providers/fake_provider/backends/perth/defs_perth.json +1 -0
  467. qiskit/providers/fake_provider/backends/perth/fake_perth.py +29 -0
  468. qiskit/providers/fake_provider/backends/perth/props_perth.json +1 -0
  469. qiskit/providers/fake_provider/backends/poughkeepsie/__init__.py +16 -0
  470. qiskit/providers/fake_provider/backends/poughkeepsie/conf_poughkeepsie.json +1 -0
  471. qiskit/providers/fake_provider/backends/poughkeepsie/defs_poughkeepsie.json +1 -0
  472. qiskit/providers/fake_provider/backends/poughkeepsie/fake_poughkeepsie.py +124 -0
  473. qiskit/providers/fake_provider/backends/poughkeepsie/props_poughkeepsie.json +1 -0
  474. qiskit/providers/fake_provider/backends/prague/__init__.py +15 -0
  475. qiskit/providers/fake_provider/backends/prague/conf_prague.json +1 -0
  476. qiskit/providers/fake_provider/backends/prague/fake_prague.py +28 -0
  477. qiskit/providers/fake_provider/backends/prague/props_prague.json +1 -0
  478. qiskit/providers/fake_provider/backends/quito/__init__.py +16 -0
  479. qiskit/providers/fake_provider/backends/quito/conf_quito.json +1 -0
  480. qiskit/providers/fake_provider/backends/quito/defs_quito.json +1 -0
  481. qiskit/providers/fake_provider/backends/quito/fake_quito.py +38 -0
  482. qiskit/providers/fake_provider/backends/quito/props_quito.json +1 -0
  483. qiskit/providers/fake_provider/backends/rochester/__init__.py +16 -0
  484. qiskit/providers/fake_provider/backends/rochester/conf_rochester.json +1 -0
  485. qiskit/providers/fake_provider/backends/rochester/fake_rochester.py +36 -0
  486. qiskit/providers/fake_provider/backends/rochester/props_rochester.json +1 -0
  487. qiskit/providers/fake_provider/backends/rome/__init__.py +16 -0
  488. qiskit/providers/fake_provider/backends/rome/conf_rome.json +1 -0
  489. qiskit/providers/fake_provider/backends/rome/defs_rome.json +1 -0
  490. qiskit/providers/fake_provider/backends/rome/fake_rome.py +38 -0
  491. qiskit/providers/fake_provider/backends/rome/props_rome.json +1 -0
  492. qiskit/providers/fake_provider/backends/rueschlikon/__init__.py +15 -0
  493. qiskit/providers/fake_provider/backends/rueschlikon/fake_rueschlikon.py +74 -0
  494. qiskit/providers/fake_provider/backends/santiago/__init__.py +16 -0
  495. qiskit/providers/fake_provider/backends/santiago/conf_santiago.json +1 -0
  496. qiskit/providers/fake_provider/backends/santiago/defs_santiago.json +1 -0
  497. qiskit/providers/fake_provider/backends/santiago/fake_santiago.py +38 -0
  498. qiskit/providers/fake_provider/backends/santiago/props_santiago.json +1 -0
  499. qiskit/providers/fake_provider/backends/sherbrooke/__init__.py +17 -0
  500. qiskit/providers/fake_provider/backends/sherbrooke/conf_sherbrooke.json +1 -0
  501. qiskit/providers/fake_provider/backends/sherbrooke/defs_sherbrooke.json +1 -0
  502. qiskit/providers/fake_provider/backends/sherbrooke/fake_sherbrooke.py +28 -0
  503. qiskit/providers/fake_provider/backends/sherbrooke/props_sherbrooke.json +1 -0
  504. qiskit/providers/fake_provider/backends/singapore/__init__.py +16 -0
  505. qiskit/providers/fake_provider/backends/singapore/conf_singapore.json +1 -0
  506. qiskit/providers/fake_provider/backends/singapore/fake_singapore.py +58 -0
  507. qiskit/providers/fake_provider/backends/singapore/props_singapore.json +1 -0
  508. qiskit/providers/fake_provider/backends/sydney/__init__.py +16 -0
  509. qiskit/providers/fake_provider/backends/sydney/conf_sydney.json +1 -0
  510. qiskit/providers/fake_provider/backends/sydney/defs_sydney.json +1 -0
  511. qiskit/providers/fake_provider/backends/sydney/fake_sydney.py +38 -0
  512. qiskit/providers/fake_provider/backends/sydney/props_sydney.json +1 -0
  513. qiskit/providers/fake_provider/backends/tenerife/__init__.py +15 -0
  514. qiskit/providers/fake_provider/backends/tenerife/fake_tenerife.py +64 -0
  515. qiskit/providers/fake_provider/backends/tenerife/props_tenerife.json +1 -0
  516. qiskit/providers/fake_provider/backends/tokyo/__init__.py +15 -0
  517. qiskit/providers/fake_provider/backends/tokyo/fake_tokyo.py +137 -0
  518. qiskit/providers/fake_provider/backends/tokyo/props_tokyo.json +1 -0
  519. qiskit/providers/fake_provider/backends/toronto/__init__.py +16 -0
  520. qiskit/providers/fake_provider/backends/toronto/conf_toronto.json +1 -0
  521. qiskit/providers/fake_provider/backends/toronto/defs_toronto.json +1 -0
  522. qiskit/providers/fake_provider/backends/toronto/fake_toronto.py +38 -0
  523. qiskit/providers/fake_provider/backends/toronto/props_toronto.json +1 -0
  524. qiskit/providers/fake_provider/backends/valencia/__init__.py +16 -0
  525. qiskit/providers/fake_provider/backends/valencia/conf_valencia.json +1 -0
  526. qiskit/providers/fake_provider/backends/valencia/defs_valencia.json +1 -0
  527. qiskit/providers/fake_provider/backends/valencia/fake_valencia.py +38 -0
  528. qiskit/providers/fake_provider/backends/valencia/props_valencia.json +1 -0
  529. qiskit/providers/fake_provider/backends/vigo/__init__.py +16 -0
  530. qiskit/providers/fake_provider/backends/vigo/conf_vigo.json +1 -0
  531. qiskit/providers/fake_provider/backends/vigo/fake_vigo.py +50 -0
  532. qiskit/providers/fake_provider/backends/vigo/props_vigo.json +1 -0
  533. qiskit/providers/fake_provider/backends/washington/__init__.py +18 -0
  534. qiskit/providers/fake_provider/backends/washington/conf_washington.json +1 -0
  535. qiskit/providers/fake_provider/backends/washington/defs_washington.json +1 -0
  536. qiskit/providers/fake_provider/backends/washington/fake_washington.py +38 -0
  537. qiskit/providers/fake_provider/backends/washington/props_washington.json +1 -0
  538. qiskit/providers/fake_provider/backends/yorktown/__init__.py +16 -0
  539. qiskit/providers/fake_provider/backends/yorktown/conf_yorktown.json +1 -0
  540. qiskit/providers/fake_provider/backends/yorktown/fake_yorktown.py +54 -0
  541. qiskit/providers/fake_provider/backends/yorktown/props_yorktown.json +1 -0
  542. qiskit/providers/fake_provider/fake_1q.py +91 -0
  543. qiskit/providers/fake_provider/fake_backend.py +572 -0
  544. qiskit/providers/fake_provider/fake_backend_v2.py +217 -0
  545. qiskit/providers/fake_provider/fake_job.py +81 -0
  546. qiskit/providers/fake_provider/fake_mumbai_v2.py +637 -0
  547. qiskit/providers/fake_provider/fake_openpulse_2q.py +342 -0
  548. qiskit/providers/fake_provider/fake_openpulse_3q.py +332 -0
  549. qiskit/providers/fake_provider/fake_provider.py +214 -0
  550. qiskit/providers/fake_provider/fake_pulse_backend.py +43 -0
  551. qiskit/providers/fake_provider/fake_qasm_backend.py +72 -0
  552. qiskit/providers/fake_provider/fake_qasm_simulator.py +48 -0
  553. qiskit/providers/fake_provider/fake_qobj.py +44 -0
  554. qiskit/providers/fake_provider/utils/__init__.py +15 -0
  555. qiskit/providers/fake_provider/utils/backend_converter.py +150 -0
  556. qiskit/providers/fake_provider/utils/configurable_backend.py +360 -0
  557. qiskit/providers/fake_provider/utils/json_decoder.py +109 -0
  558. qiskit/providers/job.py +142 -0
  559. qiskit/providers/jobstatus.py +30 -0
  560. qiskit/providers/models/__init__.py +52 -0
  561. qiskit/providers/models/backendconfiguration.py +994 -0
  562. qiskit/providers/models/backendproperties.py +490 -0
  563. qiskit/providers/models/backendstatus.py +94 -0
  564. qiskit/providers/models/jobstatus.py +66 -0
  565. qiskit/providers/models/pulsedefaults.py +304 -0
  566. qiskit/providers/options.py +273 -0
  567. qiskit/providers/provider.py +79 -0
  568. qiskit/providers/providerutils.py +99 -0
  569. qiskit/pulse/__init__.py +170 -0
  570. qiskit/pulse/builder.py +2733 -0
  571. qiskit/pulse/calibration_entries.py +357 -0
  572. qiskit/pulse/channels.py +221 -0
  573. qiskit/pulse/configuration.py +244 -0
  574. qiskit/pulse/exceptions.py +43 -0
  575. qiskit/pulse/filters.py +302 -0
  576. qiskit/pulse/instruction_schedule_map.py +406 -0
  577. qiskit/pulse/instructions/__init__.py +69 -0
  578. qiskit/pulse/instructions/acquire.py +150 -0
  579. qiskit/pulse/instructions/call.py +176 -0
  580. qiskit/pulse/instructions/delay.py +69 -0
  581. qiskit/pulse/instructions/directives.py +145 -0
  582. qiskit/pulse/instructions/frequency.py +132 -0
  583. qiskit/pulse/instructions/instruction.py +266 -0
  584. qiskit/pulse/instructions/phase.py +149 -0
  585. qiskit/pulse/instructions/play.py +96 -0
  586. qiskit/pulse/instructions/reference.py +99 -0
  587. qiskit/pulse/instructions/snapshot.py +80 -0
  588. qiskit/pulse/library/__init__.py +99 -0
  589. qiskit/pulse/library/continuous.py +430 -0
  590. qiskit/pulse/library/parametric_pulses.py +629 -0
  591. qiskit/pulse/library/pulse.py +137 -0
  592. qiskit/pulse/library/samplers/__init__.py +15 -0
  593. qiskit/pulse/library/samplers/decorators.py +299 -0
  594. qiskit/pulse/library/samplers/strategies.py +71 -0
  595. qiskit/pulse/library/symbolic_pulses.py +1962 -0
  596. qiskit/pulse/library/waveform.py +134 -0
  597. qiskit/pulse/macros.py +256 -0
  598. qiskit/pulse/parameter_manager.py +432 -0
  599. qiskit/pulse/parser.py +314 -0
  600. qiskit/pulse/reference_manager.py +58 -0
  601. qiskit/pulse/schedule.py +2002 -0
  602. qiskit/pulse/transforms/__init__.py +106 -0
  603. qiskit/pulse/transforms/alignments.py +406 -0
  604. qiskit/pulse/transforms/base_transforms.py +71 -0
  605. qiskit/pulse/transforms/canonicalization.py +514 -0
  606. qiskit/pulse/transforms/dag.py +107 -0
  607. qiskit/pulse/utils.py +109 -0
  608. qiskit/qasm/libs/qelib1.inc +266 -0
  609. qiskit/qasm/libs/stdgates.inc +75 -0
  610. qiskit/qasm2/__init__.py +658 -0
  611. qiskit/qasm2/exceptions.py +27 -0
  612. qiskit/qasm2/export.py +374 -0
  613. qiskit/qasm2/parse.py +403 -0
  614. qiskit/qasm3/__init__.py +255 -0
  615. qiskit/qasm3/ast.py +606 -0
  616. qiskit/qasm3/exceptions.py +27 -0
  617. qiskit/qasm3/experimental.py +30 -0
  618. qiskit/qasm3/exporter.py +1079 -0
  619. qiskit/qasm3/printer.py +545 -0
  620. qiskit/qobj/__init__.py +75 -0
  621. qiskit/qobj/common.py +71 -0
  622. qiskit/qobj/converters/__init__.py +18 -0
  623. qiskit/qobj/converters/lo_config.py +168 -0
  624. qiskit/qobj/converters/pulse_instruction.py +1070 -0
  625. qiskit/qobj/pulse_qobj.py +655 -0
  626. qiskit/qobj/qasm_qobj.py +656 -0
  627. qiskit/qobj/utils.py +37 -0
  628. qiskit/qpy/__init__.py +1348 -0
  629. qiskit/qpy/binary_io/__init__.py +36 -0
  630. qiskit/qpy/binary_io/circuits.py +1212 -0
  631. qiskit/qpy/binary_io/schedules.py +619 -0
  632. qiskit/qpy/binary_io/value.py +549 -0
  633. qiskit/qpy/common.py +305 -0
  634. qiskit/qpy/exceptions.py +28 -0
  635. qiskit/qpy/formats.py +360 -0
  636. qiskit/qpy/interface.py +308 -0
  637. qiskit/qpy/type_keys.py +544 -0
  638. qiskit/quantum_info/__init__.py +173 -0
  639. qiskit/quantum_info/analysis/__init__.py +17 -0
  640. qiskit/quantum_info/analysis/average.py +47 -0
  641. qiskit/quantum_info/analysis/distance.py +101 -0
  642. qiskit/quantum_info/analysis/make_observable.py +43 -0
  643. qiskit/quantum_info/analysis/z2_symmetries.py +483 -0
  644. qiskit/quantum_info/operators/__init__.py +28 -0
  645. qiskit/quantum_info/operators/base_operator.py +145 -0
  646. qiskit/quantum_info/operators/channel/__init__.py +29 -0
  647. qiskit/quantum_info/operators/channel/chi.py +190 -0
  648. qiskit/quantum_info/operators/channel/choi.py +217 -0
  649. qiskit/quantum_info/operators/channel/kraus.py +336 -0
  650. qiskit/quantum_info/operators/channel/ptm.py +203 -0
  651. qiskit/quantum_info/operators/channel/quantum_channel.py +350 -0
  652. qiskit/quantum_info/operators/channel/stinespring.py +295 -0
  653. qiskit/quantum_info/operators/channel/superop.py +376 -0
  654. qiskit/quantum_info/operators/channel/transformations.py +467 -0
  655. qiskit/quantum_info/operators/custom_iterator.py +48 -0
  656. qiskit/quantum_info/operators/dihedral/__init__.py +18 -0
  657. qiskit/quantum_info/operators/dihedral/dihedral.py +508 -0
  658. qiskit/quantum_info/operators/dihedral/dihedral_circuits.py +218 -0
  659. qiskit/quantum_info/operators/dihedral/polynomial.py +313 -0
  660. qiskit/quantum_info/operators/dihedral/random.py +61 -0
  661. qiskit/quantum_info/operators/linear_op.py +25 -0
  662. qiskit/quantum_info/operators/measures.py +423 -0
  663. qiskit/quantum_info/operators/mixins/__init__.py +52 -0
  664. qiskit/quantum_info/operators/mixins/adjoint.py +52 -0
  665. qiskit/quantum_info/operators/mixins/group.py +171 -0
  666. qiskit/quantum_info/operators/mixins/linear.py +84 -0
  667. qiskit/quantum_info/operators/mixins/multiply.py +62 -0
  668. qiskit/quantum_info/operators/mixins/tolerances.py +72 -0
  669. qiskit/quantum_info/operators/op_shape.py +533 -0
  670. qiskit/quantum_info/operators/operator.py +778 -0
  671. qiskit/quantum_info/operators/predicates.py +170 -0
  672. qiskit/quantum_info/operators/random.py +154 -0
  673. qiskit/quantum_info/operators/scalar_op.py +253 -0
  674. qiskit/quantum_info/operators/symplectic/__init__.py +23 -0
  675. qiskit/quantum_info/operators/symplectic/base_pauli.py +720 -0
  676. qiskit/quantum_info/operators/symplectic/clifford.py +1022 -0
  677. qiskit/quantum_info/operators/symplectic/clifford_circuits.py +558 -0
  678. qiskit/quantum_info/operators/symplectic/pauli.py +699 -0
  679. qiskit/quantum_info/operators/symplectic/pauli_list.py +1209 -0
  680. qiskit/quantum_info/operators/symplectic/pauli_utils.py +40 -0
  681. qiskit/quantum_info/operators/symplectic/random.py +264 -0
  682. qiskit/quantum_info/operators/symplectic/sparse_pauli_op.py +1156 -0
  683. qiskit/quantum_info/operators/utils/__init__.py +20 -0
  684. qiskit/quantum_info/operators/utils/anti_commutator.py +36 -0
  685. qiskit/quantum_info/operators/utils/commutator.py +36 -0
  686. qiskit/quantum_info/operators/utils/double_commutator.py +76 -0
  687. qiskit/quantum_info/random.py +26 -0
  688. qiskit/quantum_info/states/__init__.py +28 -0
  689. qiskit/quantum_info/states/densitymatrix.py +848 -0
  690. qiskit/quantum_info/states/measures.py +288 -0
  691. qiskit/quantum_info/states/quantum_state.py +503 -0
  692. qiskit/quantum_info/states/random.py +157 -0
  693. qiskit/quantum_info/states/stabilizerstate.py +638 -0
  694. qiskit/quantum_info/states/statevector.py +961 -0
  695. qiskit/quantum_info/states/utils.py +245 -0
  696. qiskit/quantum_info/synthesis/__init__.py +20 -0
  697. qiskit/quantum_info/synthesis/clifford_decompose.py +69 -0
  698. qiskit/quantum_info/synthesis/cnotdihedral_decompose.py +50 -0
  699. qiskit/quantum_info/synthesis/ion_decompose.py +55 -0
  700. qiskit/quantum_info/synthesis/local_invariance.py +93 -0
  701. qiskit/quantum_info/synthesis/one_qubit_decompose.py +284 -0
  702. qiskit/quantum_info/synthesis/qsd.py +269 -0
  703. qiskit/quantum_info/synthesis/quaternion.py +156 -0
  704. qiskit/quantum_info/synthesis/two_qubit_decompose.py +1567 -0
  705. qiskit/quantum_info/synthesis/weyl.py +98 -0
  706. qiskit/quantum_info/synthesis/xx_decompose/__init__.py +19 -0
  707. qiskit/quantum_info/synthesis/xx_decompose/circuits.py +299 -0
  708. qiskit/quantum_info/synthesis/xx_decompose/decomposer.py +314 -0
  709. qiskit/quantum_info/synthesis/xx_decompose/embodiments.py +163 -0
  710. qiskit/quantum_info/synthesis/xx_decompose/paths.py +412 -0
  711. qiskit/quantum_info/synthesis/xx_decompose/polytopes.py +264 -0
  712. qiskit/quantum_info/synthesis/xx_decompose/utilities.py +40 -0
  713. qiskit/quantum_info/synthesis/xx_decompose/weyl.py +133 -0
  714. qiskit/result/__init__.py +67 -0
  715. qiskit/result/counts.py +189 -0
  716. qiskit/result/distributions/__init__.py +17 -0
  717. qiskit/result/distributions/probability.py +100 -0
  718. qiskit/result/distributions/quasi.py +154 -0
  719. qiskit/result/exceptions.py +40 -0
  720. qiskit/result/mitigation/__init__.py +13 -0
  721. qiskit/result/mitigation/base_readout_mitigator.py +79 -0
  722. qiskit/result/mitigation/correlated_readout_mitigator.py +268 -0
  723. qiskit/result/mitigation/local_readout_mitigator.py +319 -0
  724. qiskit/result/mitigation/utils.py +161 -0
  725. qiskit/result/models.py +233 -0
  726. qiskit/result/postprocess.py +239 -0
  727. qiskit/result/result.py +397 -0
  728. qiskit/result/sampled_expval.py +75 -0
  729. qiskit/result/utils.py +295 -0
  730. qiskit/scheduler/__init__.py +31 -0
  731. qiskit/scheduler/config.py +35 -0
  732. qiskit/scheduler/lowering.py +187 -0
  733. qiskit/scheduler/methods/__init__.py +22 -0
  734. qiskit/scheduler/methods/basic.py +137 -0
  735. qiskit/scheduler/schedule_circuit.py +67 -0
  736. qiskit/scheduler/sequence.py +102 -0
  737. qiskit/synthesis/__init__.py +122 -0
  738. qiskit/synthesis/clifford/__init__.py +19 -0
  739. qiskit/synthesis/clifford/clifford_decompose_ag.py +176 -0
  740. qiskit/synthesis/clifford/clifford_decompose_bm.py +276 -0
  741. qiskit/synthesis/clifford/clifford_decompose_full.py +61 -0
  742. qiskit/synthesis/clifford/clifford_decompose_greedy.py +347 -0
  743. qiskit/synthesis/clifford/clifford_decompose_layers.py +443 -0
  744. qiskit/synthesis/cnotdihedral/__init__.py +17 -0
  745. qiskit/synthesis/cnotdihedral/cnotdihedral_decompose_full.py +46 -0
  746. qiskit/synthesis/cnotdihedral/cnotdihedral_decompose_general.py +140 -0
  747. qiskit/synthesis/cnotdihedral/cnotdihedral_decompose_two_qubits.py +264 -0
  748. qiskit/synthesis/discrete_basis/__init__.py +16 -0
  749. qiskit/synthesis/discrete_basis/commutator_decompose.py +241 -0
  750. qiskit/synthesis/discrete_basis/gate_sequence.py +415 -0
  751. qiskit/synthesis/discrete_basis/generate_basis_approximations.py +163 -0
  752. qiskit/synthesis/discrete_basis/solovay_kitaev.py +207 -0
  753. qiskit/synthesis/evolution/__init__.py +20 -0
  754. qiskit/synthesis/evolution/evolution_synthesis.py +46 -0
  755. qiskit/synthesis/evolution/lie_trotter.py +123 -0
  756. qiskit/synthesis/evolution/matrix_synthesis.py +47 -0
  757. qiskit/synthesis/evolution/product_formula.py +328 -0
  758. qiskit/synthesis/evolution/qdrift.py +102 -0
  759. qiskit/synthesis/evolution/suzuki_trotter.py +145 -0
  760. qiskit/synthesis/linear/__init__.py +25 -0
  761. qiskit/synthesis/linear/cnot_synth.py +141 -0
  762. qiskit/synthesis/linear/linear_circuits_utils.py +127 -0
  763. qiskit/synthesis/linear/linear_depth_lnn.py +275 -0
  764. qiskit/synthesis/linear/linear_matrix_utils.py +175 -0
  765. qiskit/synthesis/linear_phase/__init__.py +17 -0
  766. qiskit/synthesis/linear_phase/cnot_phase_synth.py +203 -0
  767. qiskit/synthesis/linear_phase/cx_cz_depth_lnn.py +262 -0
  768. qiskit/synthesis/linear_phase/cz_depth_lnn.py +206 -0
  769. qiskit/synthesis/permutation/__init__.py +17 -0
  770. qiskit/synthesis/permutation/permutation_full.py +90 -0
  771. qiskit/synthesis/permutation/permutation_lnn.py +68 -0
  772. qiskit/synthesis/permutation/permutation_utils.py +73 -0
  773. qiskit/synthesis/stabilizer/__init__.py +15 -0
  774. qiskit/synthesis/stabilizer/stabilizer_decompose.py +188 -0
  775. qiskit/test/__init__.py +18 -0
  776. qiskit/test/_canonical.py +125 -0
  777. qiskit/test/base.py +331 -0
  778. qiskit/test/decorators.py +308 -0
  779. qiskit/test/ibmq_mock.py +45 -0
  780. qiskit/test/mock/__init__.py +40 -0
  781. qiskit/test/mock/backends/__init__.py +32 -0
  782. qiskit/test/mock/backends/almaden/__init__.py +32 -0
  783. qiskit/test/mock/backends/armonk/__init__.py +32 -0
  784. qiskit/test/mock/backends/athens/__init__.py +32 -0
  785. qiskit/test/mock/backends/belem/__init__.py +32 -0
  786. qiskit/test/mock/backends/boeblingen/__init__.py +32 -0
  787. qiskit/test/mock/backends/bogota/__init__.py +32 -0
  788. qiskit/test/mock/backends/brooklyn/__init__.py +32 -0
  789. qiskit/test/mock/backends/burlington/__init__.py +32 -0
  790. qiskit/test/mock/backends/cairo/__init__.py +32 -0
  791. qiskit/test/mock/backends/cambridge/__init__.py +32 -0
  792. qiskit/test/mock/backends/casablanca/__init__.py +32 -0
  793. qiskit/test/mock/backends/essex/__init__.py +32 -0
  794. qiskit/test/mock/backends/guadalupe/__init__.py +32 -0
  795. qiskit/test/mock/backends/hanoi/__init__.py +32 -0
  796. qiskit/test/mock/backends/jakarta/__init__.py +32 -0
  797. qiskit/test/mock/backends/johannesburg/__init__.py +32 -0
  798. qiskit/test/mock/backends/kolkata/__init__.py +32 -0
  799. qiskit/test/mock/backends/lagos/__init__.py +32 -0
  800. qiskit/test/mock/backends/lima/__init__.py +32 -0
  801. qiskit/test/mock/backends/london/__init__.py +32 -0
  802. qiskit/test/mock/backends/manhattan/__init__.py +32 -0
  803. qiskit/test/mock/backends/manila/__init__.py +32 -0
  804. qiskit/test/mock/backends/melbourne/__init__.py +32 -0
  805. qiskit/test/mock/backends/montreal/__init__.py +32 -0
  806. qiskit/test/mock/backends/mumbai/__init__.py +32 -0
  807. qiskit/test/mock/backends/nairobi/__init__.py +32 -0
  808. qiskit/test/mock/backends/ourense/__init__.py +32 -0
  809. qiskit/test/mock/backends/paris/__init__.py +32 -0
  810. qiskit/test/mock/backends/poughkeepsie/__init__.py +32 -0
  811. qiskit/test/mock/backends/quito/__init__.py +32 -0
  812. qiskit/test/mock/backends/rochester/__init__.py +32 -0
  813. qiskit/test/mock/backends/rome/__init__.py +32 -0
  814. qiskit/test/mock/backends/rueschlikon/__init__.py +32 -0
  815. qiskit/test/mock/backends/santiago/__init__.py +32 -0
  816. qiskit/test/mock/backends/singapore/__init__.py +32 -0
  817. qiskit/test/mock/backends/sydney/__init__.py +32 -0
  818. qiskit/test/mock/backends/tenerife/__init__.py +32 -0
  819. qiskit/test/mock/backends/tokyo/__init__.py +32 -0
  820. qiskit/test/mock/backends/toronto/__init__.py +32 -0
  821. qiskit/test/mock/backends/valencia/__init__.py +32 -0
  822. qiskit/test/mock/backends/vigo/__init__.py +32 -0
  823. qiskit/test/mock/backends/washington/__init__.py +32 -0
  824. qiskit/test/mock/backends/yorktown/__init__.py +32 -0
  825. qiskit/test/providers/__init__.py +16 -0
  826. qiskit/test/providers/backend.py +75 -0
  827. qiskit/test/providers/provider.py +59 -0
  828. qiskit/test/reference_circuits.py +41 -0
  829. qiskit/test/testing_options.py +93 -0
  830. qiskit/test/utils.py +87 -0
  831. qiskit/tools/__init__.py +44 -0
  832. qiskit/tools/events/__init__.py +25 -0
  833. qiskit/tools/events/progressbar.py +195 -0
  834. qiskit/tools/events/pubsub.py +158 -0
  835. qiskit/tools/jupyter/__init__.py +138 -0
  836. qiskit/tools/jupyter/backend_monitor.py +588 -0
  837. qiskit/tools/jupyter/backend_overview.py +322 -0
  838. qiskit/tools/jupyter/copyright.py +42 -0
  839. qiskit/tools/jupyter/job_watcher.py +167 -0
  840. qiskit/tools/jupyter/job_widgets.py +160 -0
  841. qiskit/tools/jupyter/jupyter_magics.py +190 -0
  842. qiskit/tools/jupyter/library.py +189 -0
  843. qiskit/tools/jupyter/monospace.py +29 -0
  844. qiskit/tools/jupyter/progressbar.py +122 -0
  845. qiskit/tools/jupyter/version_table.py +67 -0
  846. qiskit/tools/jupyter/watcher_monitor.py +74 -0
  847. qiskit/tools/monitor/__init__.py +16 -0
  848. qiskit/tools/monitor/job_monitor.py +107 -0
  849. qiskit/tools/monitor/overview.py +247 -0
  850. qiskit/tools/parallel.py +198 -0
  851. qiskit/tools/visualization.py +16 -0
  852. qiskit/transpiler/__init__.py +1287 -0
  853. qiskit/transpiler/basepasses.py +221 -0
  854. qiskit/transpiler/coupling.py +500 -0
  855. qiskit/transpiler/exceptions.py +55 -0
  856. qiskit/transpiler/fencedobjs.py +78 -0
  857. qiskit/transpiler/instruction_durations.py +278 -0
  858. qiskit/transpiler/layout.py +658 -0
  859. qiskit/transpiler/passes/__init__.py +296 -0
  860. qiskit/transpiler/passes/analysis/__init__.py +23 -0
  861. qiskit/transpiler/passes/analysis/count_ops.py +30 -0
  862. qiskit/transpiler/passes/analysis/count_ops_longest_path.py +26 -0
  863. qiskit/transpiler/passes/analysis/dag_longest_path.py +24 -0
  864. qiskit/transpiler/passes/analysis/depth.py +33 -0
  865. qiskit/transpiler/passes/analysis/num_qubits.py +26 -0
  866. qiskit/transpiler/passes/analysis/num_tensor_factors.py +26 -0
  867. qiskit/transpiler/passes/analysis/resource_estimation.py +41 -0
  868. qiskit/transpiler/passes/analysis/size.py +36 -0
  869. qiskit/transpiler/passes/analysis/width.py +27 -0
  870. qiskit/transpiler/passes/basis/__init__.py +20 -0
  871. qiskit/transpiler/passes/basis/basis_translator.py +697 -0
  872. qiskit/transpiler/passes/basis/decompose.py +98 -0
  873. qiskit/transpiler/passes/basis/translate_parameterized.py +177 -0
  874. qiskit/transpiler/passes/basis/unroll_3q_or_more.py +86 -0
  875. qiskit/transpiler/passes/basis/unroll_custom_definitions.py +107 -0
  876. qiskit/transpiler/passes/basis/unroller.py +145 -0
  877. qiskit/transpiler/passes/calibration/__init__.py +17 -0
  878. qiskit/transpiler/passes/calibration/base_builder.py +79 -0
  879. qiskit/transpiler/passes/calibration/builders.py +20 -0
  880. qiskit/transpiler/passes/calibration/exceptions.py +22 -0
  881. qiskit/transpiler/passes/calibration/pulse_gate.py +98 -0
  882. qiskit/transpiler/passes/calibration/rx_builder.py +160 -0
  883. qiskit/transpiler/passes/calibration/rzx_builder.py +394 -0
  884. qiskit/transpiler/passes/calibration/rzx_templates.py +51 -0
  885. qiskit/transpiler/passes/layout/__init__.py +27 -0
  886. qiskit/transpiler/passes/layout/_csp_custom_solver.py +65 -0
  887. qiskit/transpiler/passes/layout/apply_layout.py +108 -0
  888. qiskit/transpiler/passes/layout/csp_layout.py +132 -0
  889. qiskit/transpiler/passes/layout/dense_layout.py +202 -0
  890. qiskit/transpiler/passes/layout/disjoint_utils.py +205 -0
  891. qiskit/transpiler/passes/layout/enlarge_with_ancilla.py +49 -0
  892. qiskit/transpiler/passes/layout/full_ancilla_allocation.py +117 -0
  893. qiskit/transpiler/passes/layout/layout_2q_distance.py +77 -0
  894. qiskit/transpiler/passes/layout/noise_adaptive_layout.py +311 -0
  895. qiskit/transpiler/passes/layout/sabre_layout.py +468 -0
  896. qiskit/transpiler/passes/layout/sabre_pre_layout.py +217 -0
  897. qiskit/transpiler/passes/layout/set_layout.py +64 -0
  898. qiskit/transpiler/passes/layout/trivial_layout.py +66 -0
  899. qiskit/transpiler/passes/layout/vf2_layout.py +257 -0
  900. qiskit/transpiler/passes/layout/vf2_post_layout.py +419 -0
  901. qiskit/transpiler/passes/layout/vf2_utils.py +260 -0
  902. qiskit/transpiler/passes/optimization/__init__.py +38 -0
  903. qiskit/transpiler/passes/optimization/_gate_extension.py +80 -0
  904. qiskit/transpiler/passes/optimization/collect_1q_runs.py +31 -0
  905. qiskit/transpiler/passes/optimization/collect_2q_blocks.py +35 -0
  906. qiskit/transpiler/passes/optimization/collect_and_collapse.py +115 -0
  907. qiskit/transpiler/passes/optimization/collect_cliffords.py +97 -0
  908. qiskit/transpiler/passes/optimization/collect_linear_functions.py +80 -0
  909. qiskit/transpiler/passes/optimization/collect_multiqubit_blocks.py +227 -0
  910. qiskit/transpiler/passes/optimization/commutation_analysis.py +93 -0
  911. qiskit/transpiler/passes/optimization/commutative_cancellation.py +207 -0
  912. qiskit/transpiler/passes/optimization/commutative_inverse_cancellation.py +97 -0
  913. qiskit/transpiler/passes/optimization/consolidate_blocks.py +219 -0
  914. qiskit/transpiler/passes/optimization/crosstalk_adaptive_schedule.py +732 -0
  915. qiskit/transpiler/passes/optimization/cx_cancellation.py +55 -0
  916. qiskit/transpiler/passes/optimization/echo_rzx_weyl_decomposition.py +160 -0
  917. qiskit/transpiler/passes/optimization/hoare_opt.py +416 -0
  918. qiskit/transpiler/passes/optimization/inverse_cancellation.py +177 -0
  919. qiskit/transpiler/passes/optimization/normalize_rx_angle.py +149 -0
  920. qiskit/transpiler/passes/optimization/optimize_1q_commutation.py +268 -0
  921. qiskit/transpiler/passes/optimization/optimize_1q_decomposition.py +263 -0
  922. qiskit/transpiler/passes/optimization/optimize_1q_gates.py +384 -0
  923. qiskit/transpiler/passes/optimization/optimize_cliffords.py +89 -0
  924. qiskit/transpiler/passes/optimization/optimize_swap_before_measure.py +71 -0
  925. qiskit/transpiler/passes/optimization/remove_diagonal_gates_before_measure.py +69 -0
  926. qiskit/transpiler/passes/optimization/remove_reset_in_zero_state.py +37 -0
  927. qiskit/transpiler/passes/optimization/reset_after_measure_simplification.py +47 -0
  928. qiskit/transpiler/passes/optimization/template_matching/__init__.py +19 -0
  929. qiskit/transpiler/passes/optimization/template_matching/backward_match.py +749 -0
  930. qiskit/transpiler/passes/optimization/template_matching/forward_match.py +454 -0
  931. qiskit/transpiler/passes/optimization/template_matching/maximal_matches.py +77 -0
  932. qiskit/transpiler/passes/optimization/template_matching/template_matching.py +370 -0
  933. qiskit/transpiler/passes/optimization/template_matching/template_substitution.py +629 -0
  934. qiskit/transpiler/passes/optimization/template_optimization.py +158 -0
  935. qiskit/transpiler/passes/routing/__init__.py +21 -0
  936. qiskit/transpiler/passes/routing/algorithms/__init__.py +33 -0
  937. qiskit/transpiler/passes/routing/algorithms/token_swapper.py +105 -0
  938. qiskit/transpiler/passes/routing/algorithms/types.py +46 -0
  939. qiskit/transpiler/passes/routing/algorithms/util.py +103 -0
  940. qiskit/transpiler/passes/routing/basic_swap.py +155 -0
  941. qiskit/transpiler/passes/routing/commuting_2q_gate_routing/__init__.py +25 -0
  942. qiskit/transpiler/passes/routing/commuting_2q_gate_routing/commuting_2q_block.py +60 -0
  943. qiskit/transpiler/passes/routing/commuting_2q_gate_routing/commuting_2q_gate_router.py +387 -0
  944. qiskit/transpiler/passes/routing/commuting_2q_gate_routing/pauli_2q_evolution_commutation.py +141 -0
  945. qiskit/transpiler/passes/routing/commuting_2q_gate_routing/swap_strategy.py +306 -0
  946. qiskit/transpiler/passes/routing/layout_transformation.py +118 -0
  947. qiskit/transpiler/passes/routing/lookahead_swap.py +384 -0
  948. qiskit/transpiler/passes/routing/sabre_swap.py +430 -0
  949. qiskit/transpiler/passes/routing/stochastic_swap.py +512 -0
  950. qiskit/transpiler/passes/routing/utils.py +35 -0
  951. qiskit/transpiler/passes/scheduling/__init__.py +27 -0
  952. qiskit/transpiler/passes/scheduling/alap.py +155 -0
  953. qiskit/transpiler/passes/scheduling/alignments/__init__.py +81 -0
  954. qiskit/transpiler/passes/scheduling/alignments/align_measures.py +256 -0
  955. qiskit/transpiler/passes/scheduling/alignments/check_durations.py +75 -0
  956. qiskit/transpiler/passes/scheduling/alignments/pulse_gate_validation.py +97 -0
  957. qiskit/transpiler/passes/scheduling/alignments/reschedule.py +241 -0
  958. qiskit/transpiler/passes/scheduling/asap.py +177 -0
  959. qiskit/transpiler/passes/scheduling/base_scheduler.py +289 -0
  960. qiskit/transpiler/passes/scheduling/calibration_creators.py +27 -0
  961. qiskit/transpiler/passes/scheduling/dynamical_decoupling.py +285 -0
  962. qiskit/transpiler/passes/scheduling/padding/__init__.py +16 -0
  963. qiskit/transpiler/passes/scheduling/padding/base_padding.py +256 -0
  964. qiskit/transpiler/passes/scheduling/padding/dynamical_decoupling.py +408 -0
  965. qiskit/transpiler/passes/scheduling/padding/pad_delay.py +79 -0
  966. qiskit/transpiler/passes/scheduling/rzx_templates.py +28 -0
  967. qiskit/transpiler/passes/scheduling/scheduling/__init__.py +17 -0
  968. qiskit/transpiler/passes/scheduling/scheduling/alap.py +127 -0
  969. qiskit/transpiler/passes/scheduling/scheduling/asap.py +131 -0
  970. qiskit/transpiler/passes/scheduling/scheduling/base_scheduler.py +89 -0
  971. qiskit/transpiler/passes/scheduling/scheduling/set_io_latency.py +64 -0
  972. qiskit/transpiler/passes/scheduling/time_unit_conversion.py +135 -0
  973. qiskit/transpiler/passes/synthesis/__init__.py +19 -0
  974. qiskit/transpiler/passes/synthesis/high_level_synthesis.py +637 -0
  975. qiskit/transpiler/passes/synthesis/linear_functions_synthesis.py +63 -0
  976. qiskit/transpiler/passes/synthesis/plugin.py +597 -0
  977. qiskit/transpiler/passes/synthesis/solovay_kitaev_synthesis.py +289 -0
  978. qiskit/transpiler/passes/synthesis/unitary_synthesis.py +895 -0
  979. qiskit/transpiler/passes/utils/__init__.py +34 -0
  980. qiskit/transpiler/passes/utils/barrier_before_final_measurements.py +95 -0
  981. qiskit/transpiler/passes/utils/block_to_matrix.py +47 -0
  982. qiskit/transpiler/passes/utils/check_gate_direction.py +87 -0
  983. qiskit/transpiler/passes/utils/check_map.py +94 -0
  984. qiskit/transpiler/passes/utils/contains_instruction.py +45 -0
  985. qiskit/transpiler/passes/utils/control_flow.py +61 -0
  986. qiskit/transpiler/passes/utils/convert_conditions_to_if_ops.py +89 -0
  987. qiskit/transpiler/passes/utils/dag_fixed_point.py +36 -0
  988. qiskit/transpiler/passes/utils/error.py +69 -0
  989. qiskit/transpiler/passes/utils/filter_op_nodes.py +65 -0
  990. qiskit/transpiler/passes/utils/fixed_point.py +48 -0
  991. qiskit/transpiler/passes/utils/gate_direction.py +347 -0
  992. qiskit/transpiler/passes/utils/gates_basis.py +75 -0
  993. qiskit/transpiler/passes/utils/merge_adjacent_barriers.py +162 -0
  994. qiskit/transpiler/passes/utils/minimum_point.py +118 -0
  995. qiskit/transpiler/passes/utils/remove_barriers.py +49 -0
  996. qiskit/transpiler/passes/utils/remove_final_measurements.py +114 -0
  997. qiskit/transpiler/passes/utils/unroll_forloops.py +81 -0
  998. qiskit/transpiler/passmanager.py +617 -0
  999. qiskit/transpiler/passmanager_config.py +193 -0
  1000. qiskit/transpiler/preset_passmanagers/__init__.py +280 -0
  1001. qiskit/transpiler/preset_passmanagers/builtin_plugins.py +971 -0
  1002. qiskit/transpiler/preset_passmanagers/common.py +651 -0
  1003. qiskit/transpiler/preset_passmanagers/level0.py +113 -0
  1004. qiskit/transpiler/preset_passmanagers/level1.py +120 -0
  1005. qiskit/transpiler/preset_passmanagers/level2.py +119 -0
  1006. qiskit/transpiler/preset_passmanagers/level3.py +119 -0
  1007. qiskit/transpiler/preset_passmanagers/plugin.py +345 -0
  1008. qiskit/transpiler/propertyset.py +19 -0
  1009. qiskit/transpiler/runningpassmanager.py +174 -0
  1010. qiskit/transpiler/synthesis/__init__.py +16 -0
  1011. qiskit/transpiler/synthesis/aqc/__init__.py +178 -0
  1012. qiskit/transpiler/synthesis/aqc/approximate.py +116 -0
  1013. qiskit/transpiler/synthesis/aqc/aqc.py +170 -0
  1014. qiskit/transpiler/synthesis/aqc/aqc_plugin.py +146 -0
  1015. qiskit/transpiler/synthesis/aqc/cnot_structures.py +299 -0
  1016. qiskit/transpiler/synthesis/aqc/cnot_unit_circuit.py +103 -0
  1017. qiskit/transpiler/synthesis/aqc/cnot_unit_objective.py +299 -0
  1018. qiskit/transpiler/synthesis/aqc/elementary_operations.py +108 -0
  1019. qiskit/transpiler/synthesis/aqc/fast_gradient/__init__.py +164 -0
  1020. qiskit/transpiler/synthesis/aqc/fast_gradient/fast_grad_utils.py +237 -0
  1021. qiskit/transpiler/synthesis/aqc/fast_gradient/fast_gradient.py +225 -0
  1022. qiskit/transpiler/synthesis/aqc/fast_gradient/layer.py +370 -0
  1023. qiskit/transpiler/synthesis/aqc/fast_gradient/pmatrix.py +312 -0
  1024. qiskit/transpiler/synthesis/graysynth.py +114 -0
  1025. qiskit/transpiler/target.py +1540 -0
  1026. qiskit/transpiler/timing_constraints.py +59 -0
  1027. qiskit/user_config.py +239 -0
  1028. qiskit/utils/__init__.py +66 -0
  1029. qiskit/utils/classtools.py +146 -0
  1030. qiskit/utils/deprecation.py +489 -0
  1031. qiskit/utils/lazy_tester.py +334 -0
  1032. qiskit/utils/multiprocessing.py +48 -0
  1033. qiskit/utils/optionals.py +320 -0
  1034. qiskit/utils/units.py +143 -0
  1035. qiskit/version.py +84 -0
  1036. qiskit/visualization/__init__.py +289 -0
  1037. qiskit/visualization/array.py +204 -0
  1038. qiskit/visualization/bloch.py +741 -0
  1039. qiskit/visualization/circuit/__init__.py +15 -0
  1040. qiskit/visualization/circuit/_utils.py +633 -0
  1041. qiskit/visualization/circuit/circuit_visualization.py +717 -0
  1042. qiskit/visualization/circuit/latex.py +659 -0
  1043. qiskit/visualization/circuit/matplotlib.py +1975 -0
  1044. qiskit/visualization/circuit/qcstyle.py +420 -0
  1045. qiskit/visualization/circuit/styles/bw.json +202 -0
  1046. qiskit/visualization/circuit/styles/clifford.json +202 -0
  1047. qiskit/visualization/circuit/styles/iqp-dark.json +214 -0
  1048. qiskit/visualization/circuit/styles/iqp.json +214 -0
  1049. qiskit/visualization/circuit/styles/textbook.json +202 -0
  1050. qiskit/visualization/circuit/text.py +1802 -0
  1051. qiskit/visualization/circuit_visualization.py +19 -0
  1052. qiskit/visualization/counts_visualization.py +496 -0
  1053. qiskit/visualization/dag_visualization.py +224 -0
  1054. qiskit/visualization/exceptions.py +21 -0
  1055. qiskit/visualization/gate_map.py +1461 -0
  1056. qiskit/visualization/pass_manager_visualization.py +281 -0
  1057. qiskit/visualization/pulse_v2/__init__.py +21 -0
  1058. qiskit/visualization/pulse_v2/core.py +905 -0
  1059. qiskit/visualization/pulse_v2/device_info.py +146 -0
  1060. qiskit/visualization/pulse_v2/drawings.py +253 -0
  1061. qiskit/visualization/pulse_v2/events.py +254 -0
  1062. qiskit/visualization/pulse_v2/generators/__init__.py +40 -0
  1063. qiskit/visualization/pulse_v2/generators/barrier.py +76 -0
  1064. qiskit/visualization/pulse_v2/generators/chart.py +208 -0
  1065. qiskit/visualization/pulse_v2/generators/frame.py +437 -0
  1066. qiskit/visualization/pulse_v2/generators/snapshot.py +133 -0
  1067. qiskit/visualization/pulse_v2/generators/waveform.py +649 -0
  1068. qiskit/visualization/pulse_v2/interface.py +452 -0
  1069. qiskit/visualization/pulse_v2/layouts.py +395 -0
  1070. qiskit/visualization/pulse_v2/plotters/__init__.py +17 -0
  1071. qiskit/visualization/pulse_v2/plotters/base_plotter.py +53 -0
  1072. qiskit/visualization/pulse_v2/plotters/matplotlib.py +202 -0
  1073. qiskit/visualization/pulse_v2/stylesheet.py +322 -0
  1074. qiskit/visualization/pulse_v2/types.py +242 -0
  1075. qiskit/visualization/qcstyle.py +17 -0
  1076. qiskit/visualization/state_visualization.py +1518 -0
  1077. qiskit/visualization/timeline/__init__.py +21 -0
  1078. qiskit/visualization/timeline/core.py +457 -0
  1079. qiskit/visualization/timeline/drawings.py +260 -0
  1080. qiskit/visualization/timeline/generators.py +506 -0
  1081. qiskit/visualization/timeline/interface.py +414 -0
  1082. qiskit/visualization/timeline/layouts.py +115 -0
  1083. qiskit/visualization/timeline/plotters/__init__.py +16 -0
  1084. qiskit/visualization/timeline/plotters/base_plotter.py +58 -0
  1085. qiskit/visualization/timeline/plotters/matplotlib.py +193 -0
  1086. qiskit/visualization/timeline/stylesheet.py +311 -0
  1087. qiskit/visualization/timeline/types.py +148 -0
  1088. qiskit/visualization/transition_visualization.py +364 -0
  1089. qiskit/visualization/utils.py +49 -0
  1090. qiskit-1.0.0b1.dist-info/LICENSE.txt +203 -0
  1091. qiskit-1.0.0b1.dist-info/METADATA +430 -0
  1092. qiskit-1.0.0b1.dist-info/RECORD +1095 -0
  1093. qiskit-1.0.0b1.dist-info/WHEEL +6 -0
  1094. qiskit-1.0.0b1.dist-info/entry_points.txt +49 -0
  1095. qiskit-1.0.0b1.dist-info/top_level.txt +1 -0
@@ -0,0 +1,2105 @@
1
+ # This code is part of Qiskit.
2
+ #
3
+ # (C) Copyright IBM 2017, 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
+ Object to represent a quantum circuit as a directed acyclic graph (DAG).
15
+
16
+ The nodes in the graph are either input/output nodes or operation nodes.
17
+ The edges correspond to qubits or bits in the circuit. A directed edge
18
+ from node A to node B means that the (qu)bit passes from the output of A
19
+ to the input of B. The object's methods allow circuits to be constructed,
20
+ composed, and modified. Some natural properties like depth can be computed
21
+ directly from the graph.
22
+ """
23
+ from collections import OrderedDict, defaultdict, deque, namedtuple
24
+ import copy
25
+ import math
26
+ from typing import Dict, Generator, Any, List
27
+
28
+ import numpy as np
29
+ import rustworkx as rx
30
+
31
+ from qiskit.circuit import (
32
+ ControlFlowOp,
33
+ ForLoopOp,
34
+ IfElseOp,
35
+ WhileLoopOp,
36
+ SwitchCaseOp,
37
+ _classical_resource_map,
38
+ )
39
+ from qiskit.circuit.controlflow import condition_resources, node_resources, CONTROL_FLOW_OP_NAMES
40
+ from qiskit.circuit.quantumregister import QuantumRegister, Qubit
41
+ from qiskit.circuit.classicalregister import ClassicalRegister, Clbit
42
+ from qiskit.circuit.gate import Gate
43
+ from qiskit.circuit.instruction import Instruction
44
+ from qiskit.circuit.parameterexpression import ParameterExpression
45
+ from qiskit.dagcircuit.exceptions import DAGCircuitError
46
+ from qiskit.dagcircuit.dagnode import DAGNode, DAGOpNode, DAGInNode, DAGOutNode
47
+ from qiskit.circuit.bit import Bit
48
+
49
+
50
+ BitLocations = namedtuple("BitLocations", ("index", "registers"))
51
+
52
+
53
+ class DAGCircuit:
54
+ """
55
+ Quantum circuit as a directed acyclic graph.
56
+
57
+ There are 3 types of nodes in the graph: inputs, outputs, and operations.
58
+ The nodes are connected by directed edges that correspond to qubits and
59
+ bits.
60
+ """
61
+
62
+ # pylint: disable=invalid-name
63
+
64
+ def __init__(self):
65
+ """Create an empty circuit."""
66
+
67
+ # Circuit name. Generally, this corresponds to the name
68
+ # of the QuantumCircuit from which the DAG was generated.
69
+ self.name = None
70
+
71
+ # Circuit metadata
72
+ self.metadata = {}
73
+
74
+ # Cache of dag op node sort keys
75
+ self._key_cache = {}
76
+
77
+ # Set of wires (Register,idx) in the dag
78
+ self._wires = set()
79
+
80
+ # Map from wire (Register,idx) to input nodes of the graph
81
+ self.input_map = OrderedDict()
82
+
83
+ # Map from wire (Register,idx) to output nodes of the graph
84
+ self.output_map = OrderedDict()
85
+
86
+ # Directed multigraph whose nodes are inputs, outputs, or operations.
87
+ # Operation nodes have equal in- and out-degrees and carry
88
+ # additional data about the operation, including the argument order
89
+ # and parameter values.
90
+ # Input nodes have out-degree 1 and output nodes have in-degree 1.
91
+ # Edges carry wire labels (reg,idx) and each operation has
92
+ # corresponding in- and out-edges with the same wire labels.
93
+ self._multi_graph = rx.PyDAG()
94
+
95
+ # Map of qreg/creg name to Register object.
96
+ self.qregs = OrderedDict()
97
+ self.cregs = OrderedDict()
98
+
99
+ # List of Qubit/Clbit wires that the DAG acts on.
100
+ self.qubits: List[Qubit] = []
101
+ self.clbits: List[Clbit] = []
102
+
103
+ # Dictionary mapping of Qubit and Clbit instances to a tuple comprised of
104
+ # 0) corresponding index in dag.{qubits,clbits} and
105
+ # 1) a list of Register-int pairs for each Register containing the Bit and
106
+ # its index within that register.
107
+ self._qubit_indices: Dict[Qubit, BitLocations] = {}
108
+ self._clbit_indices: Dict[Clbit, BitLocations] = {}
109
+
110
+ self._global_phase = 0
111
+ self._calibrations = defaultdict(dict)
112
+
113
+ self._op_names = {}
114
+
115
+ self.duration = None
116
+ self.unit = "dt"
117
+
118
+ @property
119
+ def wires(self):
120
+ """Return a list of the wires in order."""
121
+ return self.qubits + self.clbits
122
+
123
+ @property
124
+ def node_counter(self):
125
+ """
126
+ Returns the number of nodes in the dag.
127
+ """
128
+ return len(self._multi_graph)
129
+
130
+ @property
131
+ def global_phase(self):
132
+ """Return the global phase of the circuit."""
133
+ return self._global_phase
134
+
135
+ @global_phase.setter
136
+ def global_phase(self, angle):
137
+ """Set the global phase of the circuit.
138
+
139
+ Args:
140
+ angle (float, ParameterExpression)
141
+ """
142
+ if isinstance(angle, ParameterExpression):
143
+ self._global_phase = angle
144
+ else:
145
+ # Set the phase to the [0, 2π) interval
146
+ angle = float(angle)
147
+ if not angle:
148
+ self._global_phase = 0
149
+ else:
150
+ self._global_phase = angle % (2 * math.pi)
151
+
152
+ @property
153
+ def calibrations(self):
154
+ """Return calibration dictionary.
155
+
156
+ The custom pulse definition of a given gate is of the form
157
+ {'gate_name': {(qubits, params): schedule}}
158
+ """
159
+ return dict(self._calibrations)
160
+
161
+ @calibrations.setter
162
+ def calibrations(self, calibrations):
163
+ """Set the circuit calibration data from a dictionary of calibration definition.
164
+
165
+ Args:
166
+ calibrations (dict): A dictionary of input in the format
167
+ {'gate_name': {(qubits, gate_params): schedule}}
168
+ """
169
+ self._calibrations = defaultdict(dict, calibrations)
170
+
171
+ def add_calibration(self, gate, qubits, schedule, params=None):
172
+ """Register a low-level, custom pulse definition for the given gate.
173
+
174
+ Args:
175
+ gate (Union[Gate, str]): Gate information.
176
+ qubits (Union[int, Tuple[int]]): List of qubits to be measured.
177
+ schedule (Schedule): Schedule information.
178
+ params (Optional[List[Union[float, Parameter]]]): A list of parameters.
179
+
180
+ Raises:
181
+ Exception: if the gate is of type string and params is None.
182
+ """
183
+
184
+ def _format(operand):
185
+ try:
186
+ # Using float/complex value as a dict key is not good idea.
187
+ # This makes the mapping quite sensitive to the rounding error.
188
+ # However, the mechanism is already tied to the execution model (i.e. pulse gate)
189
+ # and we cannot easily update this rule.
190
+ # The same logic exists in QuantumCircuit.add_calibration.
191
+ evaluated = complex(operand)
192
+ if np.isreal(evaluated):
193
+ evaluated = float(evaluated.real)
194
+ if evaluated.is_integer():
195
+ evaluated = int(evaluated)
196
+ return evaluated
197
+ except TypeError:
198
+ # Unassigned parameter
199
+ return operand
200
+
201
+ if isinstance(gate, Gate):
202
+ params = gate.params
203
+ gate = gate.name
204
+ if params is not None:
205
+ params = tuple(map(_format, params))
206
+ else:
207
+ params = ()
208
+
209
+ self._calibrations[gate][(tuple(qubits), params)] = schedule
210
+
211
+ def has_calibration_for(self, node):
212
+ """Return True if the dag has a calibration defined for the node operation. In this
213
+ case, the operation does not need to be translated to the device basis.
214
+ """
215
+ if not self.calibrations or node.op.name not in self.calibrations:
216
+ return False
217
+ qubits = tuple(self.qubits.index(qubit) for qubit in node.qargs)
218
+ params = []
219
+ for p in node.op.params:
220
+ if isinstance(p, ParameterExpression) and not p.parameters:
221
+ params.append(float(p))
222
+ else:
223
+ params.append(p)
224
+ params = tuple(params)
225
+ return (qubits, params) in self.calibrations[node.op.name]
226
+
227
+ def remove_all_ops_named(self, opname):
228
+ """Remove all operation nodes with the given name."""
229
+ for n in self.named_nodes(opname):
230
+ self.remove_op_node(n)
231
+
232
+ def add_qubits(self, qubits):
233
+ """Add individual qubit wires."""
234
+ if any(not isinstance(qubit, Qubit) for qubit in qubits):
235
+ raise DAGCircuitError("not a Qubit instance.")
236
+
237
+ duplicate_qubits = set(self.qubits).intersection(qubits)
238
+ if duplicate_qubits:
239
+ raise DAGCircuitError("duplicate qubits %s" % duplicate_qubits)
240
+
241
+ for qubit in qubits:
242
+ self.qubits.append(qubit)
243
+ self._qubit_indices[qubit] = BitLocations(len(self.qubits) - 1, [])
244
+ self._add_wire(qubit)
245
+
246
+ def add_clbits(self, clbits):
247
+ """Add individual clbit wires."""
248
+ if any(not isinstance(clbit, Clbit) for clbit in clbits):
249
+ raise DAGCircuitError("not a Clbit instance.")
250
+
251
+ duplicate_clbits = set(self.clbits).intersection(clbits)
252
+ if duplicate_clbits:
253
+ raise DAGCircuitError("duplicate clbits %s" % duplicate_clbits)
254
+
255
+ for clbit in clbits:
256
+ self.clbits.append(clbit)
257
+ self._clbit_indices[clbit] = BitLocations(len(self.clbits) - 1, [])
258
+ self._add_wire(clbit)
259
+
260
+ def add_qreg(self, qreg):
261
+ """Add all wires in a quantum register."""
262
+ if not isinstance(qreg, QuantumRegister):
263
+ raise DAGCircuitError("not a QuantumRegister instance.")
264
+ if qreg.name in self.qregs:
265
+ raise DAGCircuitError("duplicate register %s" % qreg.name)
266
+ self.qregs[qreg.name] = qreg
267
+ existing_qubits = set(self.qubits)
268
+ for j in range(qreg.size):
269
+ if qreg[j] in self._qubit_indices:
270
+ self._qubit_indices[qreg[j]].registers.append((qreg, j))
271
+ if qreg[j] not in existing_qubits:
272
+ self.qubits.append(qreg[j])
273
+ self._qubit_indices[qreg[j]] = BitLocations(
274
+ len(self.qubits) - 1, registers=[(qreg, j)]
275
+ )
276
+ self._add_wire(qreg[j])
277
+
278
+ def add_creg(self, creg):
279
+ """Add all wires in a classical register."""
280
+ if not isinstance(creg, ClassicalRegister):
281
+ raise DAGCircuitError("not a ClassicalRegister instance.")
282
+ if creg.name in self.cregs:
283
+ raise DAGCircuitError("duplicate register %s" % creg.name)
284
+ self.cregs[creg.name] = creg
285
+ existing_clbits = set(self.clbits)
286
+ for j in range(creg.size):
287
+ if creg[j] in self._clbit_indices:
288
+ self._clbit_indices[creg[j]].registers.append((creg, j))
289
+ if creg[j] not in existing_clbits:
290
+ self.clbits.append(creg[j])
291
+ self._clbit_indices[creg[j]] = BitLocations(
292
+ len(self.clbits) - 1, registers=[(creg, j)]
293
+ )
294
+ self._add_wire(creg[j])
295
+
296
+ def _add_wire(self, wire):
297
+ """Add a qubit or bit to the circuit.
298
+
299
+ Args:
300
+ wire (Bit): the wire to be added
301
+
302
+ This adds a pair of in and out nodes connected by an edge.
303
+
304
+ Raises:
305
+ DAGCircuitError: if trying to add duplicate wire
306
+ """
307
+ if wire not in self._wires:
308
+ self._wires.add(wire)
309
+
310
+ inp_node = DAGInNode(wire=wire)
311
+ outp_node = DAGOutNode(wire=wire)
312
+ input_map_id, output_map_id = self._multi_graph.add_nodes_from([inp_node, outp_node])
313
+ inp_node._node_id = input_map_id
314
+ outp_node._node_id = output_map_id
315
+ self.input_map[wire] = inp_node
316
+ self.output_map[wire] = outp_node
317
+ self._multi_graph.add_edge(inp_node._node_id, outp_node._node_id, wire)
318
+ else:
319
+ raise DAGCircuitError(f"duplicate wire {wire}")
320
+
321
+ def find_bit(self, bit: Bit) -> BitLocations:
322
+ """
323
+ Finds locations in the circuit, by mapping the Qubit and Clbit to positional index
324
+ BitLocations is defined as: BitLocations = namedtuple("BitLocations", ("index", "registers"))
325
+
326
+ Args:
327
+ bit (Bit): The bit to locate.
328
+
329
+ Returns:
330
+ namedtuple(int, List[Tuple(Register, int)]): A 2-tuple. The first element (``index``)
331
+ contains the index at which the ``Bit`` can be found (in either
332
+ :obj:`~DAGCircuit.qubits`, :obj:`~DAGCircuit.clbits`, depending on its
333
+ type). The second element (``registers``) is a list of ``(register, index)``
334
+ pairs with an entry for each :obj:`~Register` in the circuit which contains the
335
+ :obj:`~Bit` (and the index in the :obj:`~Register` at which it can be found).
336
+
337
+ Raises:
338
+ DAGCircuitError: If the supplied :obj:`~Bit` was of an unknown type.
339
+ DAGCircuitError: If the supplied :obj:`~Bit` could not be found on the circuit.
340
+ """
341
+ try:
342
+ if isinstance(bit, Qubit):
343
+ return self._qubit_indices[bit]
344
+ elif isinstance(bit, Clbit):
345
+ return self._clbit_indices[bit]
346
+ else:
347
+ raise DAGCircuitError(f"Could not locate bit of unknown type: {type(bit)}")
348
+ except KeyError as err:
349
+ raise DAGCircuitError(
350
+ f"Could not locate provided bit: {bit}. Has it been added to the DAGCircuit?"
351
+ ) from err
352
+
353
+ def remove_clbits(self, *clbits):
354
+ """
355
+ Remove classical bits from the circuit. All bits MUST be idle.
356
+ Any registers with references to at least one of the specified bits will
357
+ also be removed.
358
+
359
+ Args:
360
+ clbits (List[Clbit]): The bits to remove.
361
+
362
+ Raises:
363
+ DAGCircuitError: a clbit is not a :obj:`.Clbit`, is not in the circuit,
364
+ or is not idle.
365
+ """
366
+ if any(not isinstance(clbit, Clbit) for clbit in clbits):
367
+ raise DAGCircuitError(
368
+ "clbits not of type Clbit: %s" % [b for b in clbits if not isinstance(b, Clbit)]
369
+ )
370
+
371
+ clbits = set(clbits)
372
+ unknown_clbits = clbits.difference(self.clbits)
373
+ if unknown_clbits:
374
+ raise DAGCircuitError("clbits not in circuit: %s" % unknown_clbits)
375
+
376
+ busy_clbits = {bit for bit in clbits if not self._is_wire_idle(bit)}
377
+ if busy_clbits:
378
+ raise DAGCircuitError("clbits not idle: %s" % busy_clbits)
379
+
380
+ # remove any references to bits
381
+ cregs_to_remove = {creg for creg in self.cregs.values() if not clbits.isdisjoint(creg)}
382
+ self.remove_cregs(*cregs_to_remove)
383
+
384
+ for clbit in clbits:
385
+ self._remove_idle_wire(clbit)
386
+ self.clbits.remove(clbit)
387
+ del self._clbit_indices[clbit]
388
+
389
+ # Update the indices of remaining clbits
390
+ for i, clbit in enumerate(self.clbits):
391
+ self._clbit_indices[clbit] = self._clbit_indices[clbit]._replace(index=i)
392
+
393
+ def remove_cregs(self, *cregs):
394
+ """
395
+ Remove classical registers from the circuit, leaving underlying bits
396
+ in place.
397
+
398
+ Raises:
399
+ DAGCircuitError: a creg is not a ClassicalRegister, or is not in
400
+ the circuit.
401
+ """
402
+ if any(not isinstance(creg, ClassicalRegister) for creg in cregs):
403
+ raise DAGCircuitError(
404
+ "cregs not of type ClassicalRegister: %s"
405
+ % [r for r in cregs if not isinstance(r, ClassicalRegister)]
406
+ )
407
+
408
+ unknown_cregs = set(cregs).difference(self.cregs.values())
409
+ if unknown_cregs:
410
+ raise DAGCircuitError("cregs not in circuit: %s" % unknown_cregs)
411
+
412
+ for creg in cregs:
413
+ del self.cregs[creg.name]
414
+ for j in range(creg.size):
415
+ bit = creg[j]
416
+ bit_position = self._clbit_indices[bit]
417
+ bit_position.registers.remove((creg, j))
418
+
419
+ def remove_qubits(self, *qubits):
420
+ """
421
+ Remove quantum bits from the circuit. All bits MUST be idle.
422
+ Any registers with references to at least one of the specified bits will
423
+ also be removed.
424
+
425
+ Args:
426
+ qubits (List[~qiskit.circuit.Qubit]): The bits to remove.
427
+
428
+ Raises:
429
+ DAGCircuitError: a qubit is not a :obj:`~.circuit.Qubit`, is not in the circuit,
430
+ or is not idle.
431
+ """
432
+ if any(not isinstance(qubit, Qubit) for qubit in qubits):
433
+ raise DAGCircuitError(
434
+ "qubits not of type Qubit: %s" % [b for b in qubits if not isinstance(b, Qubit)]
435
+ )
436
+
437
+ qubits = set(qubits)
438
+ unknown_qubits = qubits.difference(self.qubits)
439
+ if unknown_qubits:
440
+ raise DAGCircuitError("qubits not in circuit: %s" % unknown_qubits)
441
+
442
+ busy_qubits = {bit for bit in qubits if not self._is_wire_idle(bit)}
443
+ if busy_qubits:
444
+ raise DAGCircuitError("qubits not idle: %s" % busy_qubits)
445
+
446
+ # remove any references to bits
447
+ qregs_to_remove = {qreg for qreg in self.qregs.values() if not qubits.isdisjoint(qreg)}
448
+ self.remove_qregs(*qregs_to_remove)
449
+
450
+ for qubit in qubits:
451
+ self._remove_idle_wire(qubit)
452
+ self.qubits.remove(qubit)
453
+ del self._qubit_indices[qubit]
454
+
455
+ # Update the indices of remaining qubits
456
+ for i, qubit in enumerate(self.qubits):
457
+ self._qubit_indices[qubit] = self._qubit_indices[qubit]._replace(index=i)
458
+
459
+ def remove_qregs(self, *qregs):
460
+ """
461
+ Remove classical registers from the circuit, leaving underlying bits
462
+ in place.
463
+
464
+ Raises:
465
+ DAGCircuitError: a qreg is not a QuantumRegister, or is not in
466
+ the circuit.
467
+ """
468
+ if any(not isinstance(qreg, QuantumRegister) for qreg in qregs):
469
+ raise DAGCircuitError(
470
+ "qregs not of type QuantumRegister: %s"
471
+ % [r for r in qregs if not isinstance(r, QuantumRegister)]
472
+ )
473
+
474
+ unknown_qregs = set(qregs).difference(self.qregs.values())
475
+ if unknown_qregs:
476
+ raise DAGCircuitError("qregs not in circuit: %s" % unknown_qregs)
477
+
478
+ for qreg in qregs:
479
+ del self.qregs[qreg.name]
480
+ for j in range(qreg.size):
481
+ bit = qreg[j]
482
+ bit_position = self._qubit_indices[bit]
483
+ bit_position.registers.remove((qreg, j))
484
+
485
+ def _is_wire_idle(self, wire):
486
+ """Check if a wire is idle.
487
+
488
+ Args:
489
+ wire (Bit): a wire in the circuit.
490
+
491
+ Returns:
492
+ bool: true if the wire is idle, false otherwise.
493
+
494
+ Raises:
495
+ DAGCircuitError: the wire is not in the circuit.
496
+ """
497
+ if wire not in self._wires:
498
+ raise DAGCircuitError("wire %s not in circuit" % wire)
499
+
500
+ try:
501
+ child = next(self.successors(self.input_map[wire]))
502
+ except StopIteration as e:
503
+ raise DAGCircuitError(
504
+ "Invalid dagcircuit input node %s has no output" % self.input_map[wire]
505
+ ) from e
506
+ return child is self.output_map[wire]
507
+
508
+ def _remove_idle_wire(self, wire):
509
+ """Remove an idle qubit or bit from the circuit.
510
+
511
+ Args:
512
+ wire (Bit): the wire to be removed, which MUST be idle.
513
+ """
514
+ inp_node = self.input_map[wire]
515
+ oup_node = self.output_map[wire]
516
+
517
+ self._multi_graph.remove_node(inp_node._node_id)
518
+ self._multi_graph.remove_node(oup_node._node_id)
519
+ self._wires.remove(wire)
520
+ del self.input_map[wire]
521
+ del self.output_map[wire]
522
+
523
+ def _check_condition(self, name, condition):
524
+ """Verify that the condition is valid.
525
+
526
+ Args:
527
+ name (string): used for error reporting
528
+ condition (tuple or None): a condition tuple (ClassicalRegister, int) or (Clbit, bool)
529
+
530
+ Raises:
531
+ DAGCircuitError: if conditioning on an invalid register
532
+ """
533
+ if condition is None:
534
+ return
535
+ resources = condition_resources(condition)
536
+ for creg in resources.cregs:
537
+ if creg.name not in self.cregs:
538
+ raise DAGCircuitError(f"invalid creg in condition for {name}")
539
+ if not set(resources.clbits).issubset(self.clbits):
540
+ raise DAGCircuitError(f"invalid clbits in condition for {name}")
541
+
542
+ def _check_bits(self, args, amap):
543
+ """Check the values of a list of (qu)bit arguments.
544
+
545
+ For each element of args, check that amap contains it.
546
+
547
+ Args:
548
+ args (list[Bit]): the elements to be checked
549
+ amap (dict): a dictionary keyed on Qubits/Clbits
550
+
551
+ Raises:
552
+ DAGCircuitError: if a qubit is not contained in amap
553
+ """
554
+ # Check for each wire
555
+ for wire in args:
556
+ if wire not in amap:
557
+ raise DAGCircuitError(f"(qu)bit {wire} not found in {amap}")
558
+
559
+ @staticmethod
560
+ def _bits_in_operation(operation):
561
+ """Return an iterable over the classical bits that are inherent to an instruction. This
562
+ includes a `condition`, or the `target` of a :class:`.ControlFlowOp`.
563
+
564
+ Args:
565
+ instruction: the :class:`~.circuit.Instruction` instance for a node.
566
+
567
+ Returns:
568
+ Iterable[Clbit]: the :class:`.Clbit`\\ s involved.
569
+ """
570
+ # If updating this, also update the fast-path checker `DAGCirucit._operation_may_have_bits`.
571
+ if (condition := getattr(operation, "condition", None)) is not None:
572
+ yield from condition_resources(condition).clbits
573
+ if isinstance(operation, SwitchCaseOp):
574
+ target = operation.target
575
+ if isinstance(target, Clbit):
576
+ yield target
577
+ elif isinstance(target, ClassicalRegister):
578
+ yield from target
579
+ else:
580
+ yield from node_resources(target).clbits
581
+
582
+ @staticmethod
583
+ def _operation_may_have_bits(operation) -> bool:
584
+ """Return whether a given :class:`.Operation` may contain any :class:`.Clbit` instances
585
+ in itself (e.g. a control-flow operation).
586
+
587
+ Args:
588
+ operation (qiskit.circuit.Operation): the operation to check.
589
+ """
590
+ # This is separate to `_bits_in_operation` because most of the time there won't be any bits,
591
+ # so we want a fast path to be able to skip creating and testing a generator for emptiness.
592
+ #
593
+ # If updating this, also update `DAGCirucit._bits_in_operation`.
594
+ return getattr(operation, "condition", None) is not None or isinstance(
595
+ operation, SwitchCaseOp
596
+ )
597
+
598
+ def _increment_op(self, op):
599
+ if op.name in self._op_names:
600
+ self._op_names[op.name] += 1
601
+ else:
602
+ self._op_names[op.name] = 1
603
+
604
+ def _decrement_op(self, op):
605
+ if self._op_names[op.name] == 1:
606
+ del self._op_names[op.name]
607
+ else:
608
+ self._op_names[op.name] -= 1
609
+
610
+ def copy_empty_like(self):
611
+ """Return a copy of self with the same structure but empty.
612
+
613
+ That structure includes:
614
+ * name and other metadata
615
+ * global phase
616
+ * duration
617
+ * all the qubits and clbits, including the registers.
618
+
619
+ Returns:
620
+ DAGCircuit: An empty copy of self.
621
+ """
622
+ target_dag = DAGCircuit()
623
+ target_dag.name = self.name
624
+ target_dag._global_phase = self._global_phase
625
+ target_dag.duration = self.duration
626
+ target_dag.unit = self.unit
627
+ target_dag.metadata = self.metadata
628
+ target_dag._key_cache = self._key_cache
629
+
630
+ target_dag.add_qubits(self.qubits)
631
+ target_dag.add_clbits(self.clbits)
632
+
633
+ for qreg in self.qregs.values():
634
+ target_dag.add_qreg(qreg)
635
+ for creg in self.cregs.values():
636
+ target_dag.add_creg(creg)
637
+
638
+ return target_dag
639
+
640
+ def apply_operation_back(self, op, qargs=(), cargs=(), *, check=True):
641
+ """Apply an operation to the output of the circuit.
642
+
643
+ Args:
644
+ op (qiskit.circuit.Operation): the operation associated with the DAG node
645
+ qargs (tuple[~qiskit.circuit.Qubit]): qubits that op will be applied to
646
+ cargs (tuple[Clbit]): cbits that op will be applied to
647
+ check (bool): If ``True`` (default), this function will enforce that the
648
+ :class:`.DAGCircuit` data-structure invariants are maintained (all ``qargs`` are
649
+ :class:`~.circuit.Qubit`\\ s, all are in the DAG, etc). If ``False``, the caller *must*
650
+ uphold these invariants itself, but the cost of several checks will be skipped.
651
+ This is most useful when building a new DAG from a source of known-good nodes.
652
+ Returns:
653
+ DAGOpNode: the node for the op that was added to the dag
654
+
655
+ Raises:
656
+ DAGCircuitError: if a leaf node is connected to multiple outputs
657
+
658
+ """
659
+ qargs = tuple(qargs)
660
+ cargs = tuple(cargs)
661
+
662
+ if self._operation_may_have_bits(op):
663
+ # This is the slow path; most of the time, this won't happen.
664
+ all_cbits = set(self._bits_in_operation(op)).union(cargs)
665
+ else:
666
+ all_cbits = cargs
667
+
668
+ if check:
669
+ self._check_condition(op.name, getattr(op, "condition", None))
670
+ self._check_bits(qargs, self.output_map)
671
+ self._check_bits(all_cbits, self.output_map)
672
+
673
+ node = DAGOpNode(op=op, qargs=qargs, cargs=cargs, dag=self)
674
+ node._node_id = self._multi_graph.add_node(node)
675
+ self._increment_op(op)
676
+
677
+ # Add new in-edges from predecessors of the output nodes to the
678
+ # operation node while deleting the old in-edges of the output nodes
679
+ # and adding new edges from the operation node to each output node
680
+ self._multi_graph.insert_node_on_in_edges_multiple(
681
+ node._node_id,
682
+ [self.output_map[bit]._node_id for bits in (qargs, all_cbits) for bit in bits],
683
+ )
684
+ return node
685
+
686
+ def apply_operation_front(self, op, qargs=(), cargs=(), *, check=True):
687
+ """Apply an operation to the input of the circuit.
688
+
689
+ Args:
690
+ op (qiskit.circuit.Operation): the operation associated with the DAG node
691
+ qargs (tuple[~qiskit.circuit.Qubit]): qubits that op will be applied to
692
+ cargs (tuple[Clbit]): cbits that op will be applied to
693
+ check (bool): If ``True`` (default), this function will enforce that the
694
+ :class:`.DAGCircuit` data-structure invariants are maintained (all ``qargs`` are
695
+ :class:`~.circuit.Qubit`\\ s, all are in the DAG, etc). If ``False``, the caller *must*
696
+ uphold these invariants itself, but the cost of several checks will be skipped.
697
+ This is most useful when building a new DAG from a source of known-good nodes.
698
+ Returns:
699
+ DAGOpNode: the node for the op that was added to the dag
700
+
701
+ Raises:
702
+ DAGCircuitError: if initial nodes connected to multiple out edges
703
+ """
704
+ qargs = tuple(qargs)
705
+ cargs = tuple(cargs)
706
+
707
+ if self._operation_may_have_bits(op):
708
+ # This is the slow path; most of the time, this won't happen.
709
+ all_cbits = set(self._bits_in_operation(op)).union(cargs)
710
+ else:
711
+ all_cbits = cargs
712
+
713
+ if check:
714
+ self._check_condition(op.name, getattr(op, "condition", None))
715
+ self._check_bits(qargs, self.input_map)
716
+ self._check_bits(all_cbits, self.input_map)
717
+
718
+ node = DAGOpNode(op=op, qargs=qargs, cargs=cargs, dag=self)
719
+ node._node_id = self._multi_graph.add_node(node)
720
+ self._increment_op(op)
721
+
722
+ # Add new out-edges to successors of the input nodes from the
723
+ # operation node while deleting the old out-edges of the input nodes
724
+ # and adding new edges to the operation node from each input node
725
+ self._multi_graph.insert_node_on_out_edges_multiple(
726
+ node._node_id,
727
+ [self.input_map[bit]._node_id for bits in (qargs, all_cbits) for bit in bits],
728
+ )
729
+ return node
730
+
731
+ def compose(self, other, qubits=None, clbits=None, front=False, inplace=True):
732
+ """Compose the ``other`` circuit onto the output of this circuit.
733
+
734
+ A subset of input wires of ``other`` are mapped
735
+ to a subset of output wires of this circuit.
736
+
737
+ ``other`` can be narrower or of equal width to ``self``.
738
+
739
+ Args:
740
+ other (DAGCircuit): circuit to compose with self
741
+ qubits (list[~qiskit.circuit.Qubit|int]): qubits of self to compose onto.
742
+ clbits (list[Clbit|int]): clbits of self to compose onto.
743
+ front (bool): If True, front composition will be performed (not implemented yet)
744
+ inplace (bool): If True, modify the object. Otherwise return composed circuit.
745
+
746
+ Returns:
747
+ DAGCircuit: the composed dag (returns None if inplace==True).
748
+
749
+ Raises:
750
+ DAGCircuitError: if ``other`` is wider or there are duplicate edge mappings.
751
+ """
752
+ if front:
753
+ raise DAGCircuitError("Front composition not supported yet.")
754
+
755
+ if len(other.qubits) > len(self.qubits) or len(other.clbits) > len(self.clbits):
756
+ raise DAGCircuitError(
757
+ "Trying to compose with another DAGCircuit which has more 'in' edges."
758
+ )
759
+
760
+ # number of qubits and clbits must match number in circuit or None
761
+ identity_qubit_map = dict(zip(other.qubits, self.qubits))
762
+ identity_clbit_map = dict(zip(other.clbits, self.clbits))
763
+ if qubits is None:
764
+ qubit_map = identity_qubit_map
765
+ elif len(qubits) != len(other.qubits):
766
+ raise DAGCircuitError(
767
+ "Number of items in qubits parameter does not"
768
+ " match number of qubits in the circuit."
769
+ )
770
+ else:
771
+ qubit_map = {
772
+ other.qubits[i]: (self.qubits[q] if isinstance(q, int) else q)
773
+ for i, q in enumerate(qubits)
774
+ }
775
+ if clbits is None:
776
+ clbit_map = identity_clbit_map
777
+ elif len(clbits) != len(other.clbits):
778
+ raise DAGCircuitError(
779
+ "Number of items in clbits parameter does not"
780
+ " match number of clbits in the circuit."
781
+ )
782
+ else:
783
+ clbit_map = {
784
+ other.clbits[i]: (self.clbits[c] if isinstance(c, int) else c)
785
+ for i, c in enumerate(clbits)
786
+ }
787
+ edge_map = {**qubit_map, **clbit_map} or None
788
+
789
+ # if no edge_map, try to do a 1-1 mapping in order
790
+ if edge_map is None:
791
+ edge_map = {**identity_qubit_map, **identity_clbit_map}
792
+
793
+ # Check the edge_map for duplicate values
794
+ if len(set(edge_map.values())) != len(edge_map):
795
+ raise DAGCircuitError("duplicates in wire_map")
796
+
797
+ # Compose
798
+ if inplace:
799
+ dag = self
800
+ else:
801
+ dag = copy.deepcopy(self)
802
+ dag.global_phase += other.global_phase
803
+
804
+ for gate, cals in other.calibrations.items():
805
+ dag._calibrations[gate].update(cals)
806
+
807
+ # Ensure that the error raised here is a `DAGCircuitError` for backwards compatibility.
808
+ def _reject_new_register(reg):
809
+ raise DAGCircuitError(f"No register with '{reg.bits}' to map this expression onto.")
810
+
811
+ variable_mapper = _classical_resource_map.VariableMapper(
812
+ dag.cregs.values(), edge_map, _reject_new_register
813
+ )
814
+ for nd in other.topological_nodes():
815
+ if isinstance(nd, DAGInNode):
816
+ # if in edge_map, get new name, else use existing name
817
+ m_wire = edge_map.get(nd.wire, nd.wire)
818
+ # the mapped wire should already exist
819
+ if m_wire not in dag.output_map:
820
+ raise DAGCircuitError(
821
+ "wire %s[%d] not in self" % (m_wire.register.name, m_wire.index)
822
+ )
823
+ if nd.wire not in other._wires:
824
+ raise DAGCircuitError(
825
+ "inconsistent wire type for %s[%d] in other"
826
+ % (nd.register.name, nd.wire.index)
827
+ )
828
+ elif isinstance(nd, DAGOutNode):
829
+ # ignore output nodes
830
+ pass
831
+ elif isinstance(nd, DAGOpNode):
832
+ m_qargs = [edge_map.get(x, x) for x in nd.qargs]
833
+ m_cargs = [edge_map.get(x, x) for x in nd.cargs]
834
+ op = nd.op.copy()
835
+ if (condition := getattr(op, "condition", None)) is not None:
836
+ if not isinstance(op, ControlFlowOp):
837
+ op = op.c_if(*variable_mapper.map_condition(condition, allow_reorder=True))
838
+ else:
839
+ op.condition = variable_mapper.map_condition(condition, allow_reorder=True)
840
+ elif isinstance(op, SwitchCaseOp):
841
+ op.target = variable_mapper.map_target(op.target)
842
+ dag.apply_operation_back(op, m_qargs, m_cargs, check=False)
843
+ else:
844
+ raise DAGCircuitError("bad node type %s" % type(nd))
845
+
846
+ if not inplace:
847
+ return dag
848
+ else:
849
+ return None
850
+
851
+ def reverse_ops(self):
852
+ """Reverse the operations in the ``self`` circuit.
853
+
854
+ Returns:
855
+ DAGCircuit: the reversed dag.
856
+ """
857
+ # TODO: speed up
858
+ # pylint: disable=cyclic-import
859
+ from qiskit.converters import dag_to_circuit, circuit_to_dag
860
+
861
+ qc = dag_to_circuit(self)
862
+ reversed_qc = qc.reverse_ops()
863
+ reversed_dag = circuit_to_dag(reversed_qc)
864
+ return reversed_dag
865
+
866
+ def idle_wires(self, ignore=None):
867
+ """Return idle wires.
868
+
869
+ Args:
870
+ ignore (list(str)): List of node names to ignore. Default: []
871
+
872
+ Yields:
873
+ Bit: Bit in idle wire.
874
+
875
+ Raises:
876
+ DAGCircuitError: If the DAG is invalid
877
+ """
878
+ if ignore is None:
879
+ ignore = set()
880
+ ignore_set = set(ignore)
881
+ for wire in self._wires:
882
+ if not ignore:
883
+ if self._is_wire_idle(wire):
884
+ yield wire
885
+ else:
886
+ for node in self.nodes_on_wire(wire, only_ops=True):
887
+ if node.op.name not in ignore_set:
888
+ # If we found an op node outside of ignore we can stop iterating over the wire
889
+ break
890
+ else:
891
+ yield wire
892
+
893
+ def size(self, *, recurse: bool = False):
894
+ """Return the number of operations. If there is control flow present, this count may only
895
+ be an estimate, as the complete control-flow path cannot be statically known.
896
+
897
+ Args:
898
+ recurse: if ``True``, then recurse into control-flow operations. For loops with
899
+ known-length iterators are counted unrolled. If-else blocks sum both of the two
900
+ branches. While loops are counted as if the loop body runs once only. Defaults to
901
+ ``False`` and raises :class:`.DAGCircuitError` if any control flow is present, to
902
+ avoid silently returning a mostly meaningless number.
903
+
904
+ Returns:
905
+ int: the circuit size
906
+
907
+ Raises:
908
+ DAGCircuitError: if an unknown :class:`.ControlFlowOp` is present in a call with
909
+ ``recurse=True``, or any control flow is present in a non-recursive call.
910
+ """
911
+ length = len(self._multi_graph) - 2 * len(self._wires)
912
+ if not recurse:
913
+ if any(x in self._op_names for x in CONTROL_FLOW_OP_NAMES):
914
+ raise DAGCircuitError(
915
+ "Size with control flow is ambiguous."
916
+ " You may use `recurse=True` to get a result,"
917
+ " but see this method's documentation for the meaning of this."
918
+ )
919
+ return length
920
+ # pylint: disable=cyclic-import
921
+ from qiskit.converters import circuit_to_dag
922
+
923
+ for node in self.op_nodes(ControlFlowOp):
924
+ if isinstance(node.op, ForLoopOp):
925
+ indexset = node.op.params[0]
926
+ inner = len(indexset) * circuit_to_dag(node.op.blocks[0]).size(recurse=True)
927
+ elif isinstance(node.op, WhileLoopOp):
928
+ inner = circuit_to_dag(node.op.blocks[0]).size(recurse=True)
929
+ elif isinstance(node.op, (IfElseOp, SwitchCaseOp)):
930
+ inner = sum(circuit_to_dag(block).size(recurse=True) for block in node.op.blocks)
931
+ else:
932
+ raise DAGCircuitError(f"unknown control-flow type: '{node.op.name}'")
933
+ # Replace the "1" for the node itself with the actual count.
934
+ length += inner - 1
935
+ return length
936
+
937
+ def depth(self, *, recurse: bool = False):
938
+ """Return the circuit depth. If there is control flow present, this count may only be an
939
+ estimate, as the complete control-flow path cannot be statically known.
940
+
941
+ Args:
942
+ recurse: if ``True``, then recurse into control-flow operations. For loops
943
+ with known-length iterators are counted as if the loop had been manually unrolled
944
+ (*i.e.* with each iteration of the loop body written out explicitly).
945
+ If-else blocks take the longer case of the two branches. While loops are counted as
946
+ if the loop body runs once only. Defaults to ``False`` and raises
947
+ :class:`.DAGCircuitError` if any control flow is present, to avoid silently
948
+ returning a nonsensical number.
949
+
950
+ Returns:
951
+ int: the circuit depth
952
+
953
+ Raises:
954
+ DAGCircuitError: if not a directed acyclic graph
955
+ DAGCircuitError: if unknown control flow is present in a recursive call, or any control
956
+ flow is present in a non-recursive call.
957
+ """
958
+ if recurse:
959
+ from qiskit.converters import circuit_to_dag # pylint: disable=cyclic-import
960
+
961
+ node_lookup = {}
962
+ for node in self.op_nodes(ControlFlowOp):
963
+ weight = len(node.op.params[0]) if isinstance(node.op, ForLoopOp) else 1
964
+ if weight == 0:
965
+ node_lookup[node._node_id] = 0
966
+ else:
967
+ node_lookup[node._node_id] = weight * max(
968
+ circuit_to_dag(block).depth(recurse=True) for block in node.op.blocks
969
+ )
970
+
971
+ def weight_fn(_source, target, _edge):
972
+ return node_lookup.get(target, 1)
973
+
974
+ else:
975
+ if any(x in self._op_names for x in CONTROL_FLOW_OP_NAMES):
976
+ raise DAGCircuitError(
977
+ "Depth with control flow is ambiguous."
978
+ " You may use `recurse=True` to get a result,"
979
+ " but see this method's documentation for the meaning of this."
980
+ )
981
+ weight_fn = None
982
+
983
+ try:
984
+ depth = rx.dag_longest_path_length(self._multi_graph, weight_fn) - 1
985
+ except rx.DAGHasCycle as ex:
986
+ raise DAGCircuitError("not a DAG") from ex
987
+ return depth if depth >= 0 else 0
988
+
989
+ def width(self):
990
+ """Return the total number of qubits + clbits used by the circuit.
991
+ This function formerly returned the number of qubits by the calculation
992
+ return len(self._wires) - self.num_clbits()
993
+ but was changed by issue #2564 to return number of qubits + clbits
994
+ with the new function DAGCircuit.num_qubits replacing the former
995
+ semantic of DAGCircuit.width().
996
+ """
997
+ return len(self._wires)
998
+
999
+ def num_qubits(self):
1000
+ """Return the total number of qubits used by the circuit.
1001
+ num_qubits() replaces former use of width().
1002
+ DAGCircuit.width() now returns qubits + clbits for
1003
+ consistency with Circuit.width() [qiskit-terra #2564].
1004
+ """
1005
+ return len(self.qubits)
1006
+
1007
+ def num_clbits(self):
1008
+ """Return the total number of classical bits used by the circuit."""
1009
+ return len(self.clbits)
1010
+
1011
+ def num_tensor_factors(self):
1012
+ """Compute how many components the circuit can decompose into."""
1013
+ return rx.number_weakly_connected_components(self._multi_graph)
1014
+
1015
+ def __eq__(self, other):
1016
+ # Try to convert to float, but in case of unbound ParameterExpressions
1017
+ # a TypeError will be raise, fallback to normal equality in those
1018
+ # cases
1019
+ try:
1020
+ self_phase = float(self.global_phase)
1021
+ other_phase = float(other.global_phase)
1022
+ if (
1023
+ abs((self_phase - other_phase + np.pi) % (2 * np.pi) - np.pi) > 1.0e-10
1024
+ ): # TODO: atol?
1025
+ return False
1026
+ except TypeError:
1027
+ if self.global_phase != other.global_phase:
1028
+ return False
1029
+ if self.calibrations != other.calibrations:
1030
+ return False
1031
+
1032
+ self_bit_indices = {bit: idx for idx, bit in enumerate(self.qubits + self.clbits)}
1033
+ other_bit_indices = {bit: idx for idx, bit in enumerate(other.qubits + other.clbits)}
1034
+
1035
+ self_qreg_indices = {
1036
+ regname: [self_bit_indices[bit] for bit in reg] for regname, reg in self.qregs.items()
1037
+ }
1038
+ self_creg_indices = {
1039
+ regname: [self_bit_indices[bit] for bit in reg] for regname, reg in self.cregs.items()
1040
+ }
1041
+
1042
+ other_qreg_indices = {
1043
+ regname: [other_bit_indices[bit] for bit in reg] for regname, reg in other.qregs.items()
1044
+ }
1045
+ other_creg_indices = {
1046
+ regname: [other_bit_indices[bit] for bit in reg] for regname, reg in other.cregs.items()
1047
+ }
1048
+ if self_qreg_indices != other_qreg_indices or self_creg_indices != other_creg_indices:
1049
+ return False
1050
+
1051
+ def node_eq(node_self, node_other):
1052
+ return DAGNode.semantic_eq(node_self, node_other, self_bit_indices, other_bit_indices)
1053
+
1054
+ return rx.is_isomorphic_node_match(self._multi_graph, other._multi_graph, node_eq)
1055
+
1056
+ def topological_nodes(self, key=None):
1057
+ """
1058
+ Yield nodes in topological order.
1059
+
1060
+ Args:
1061
+ key (Callable): A callable which will take a DAGNode object and
1062
+ return a string sort key. If not specified the
1063
+ :attr:`~qiskit.dagcircuit.DAGNode.sort_key` attribute will be
1064
+ used as the sort key for each node.
1065
+
1066
+ Returns:
1067
+ generator(DAGOpNode, DAGInNode, or DAGOutNode): node in topological order
1068
+ """
1069
+
1070
+ def _key(x):
1071
+ return x.sort_key
1072
+
1073
+ if key is None:
1074
+ key = _key
1075
+
1076
+ return iter(rx.lexicographical_topological_sort(self._multi_graph, key=key))
1077
+
1078
+ def topological_op_nodes(self, key=None) -> Generator[DAGOpNode, Any, Any]:
1079
+ """
1080
+ Yield op nodes in topological order.
1081
+
1082
+ Allowed to pass in specific key to break ties in top order
1083
+
1084
+ Args:
1085
+ key (Callable): A callable which will take a DAGNode object and
1086
+ return a string sort key. If not specified the
1087
+ :attr:`~qiskit.dagcircuit.DAGNode.sort_key` attribute will be
1088
+ used as the sort key for each node.
1089
+
1090
+ Returns:
1091
+ generator(DAGOpNode): op node in topological order
1092
+ """
1093
+ return (nd for nd in self.topological_nodes(key) if isinstance(nd, DAGOpNode))
1094
+
1095
+ def replace_block_with_op(self, node_block, op, wire_pos_map, cycle_check=True):
1096
+ """Replace a block of nodes with a single node.
1097
+
1098
+ This is used to consolidate a block of DAGOpNodes into a single
1099
+ operation. A typical example is a block of gates being consolidated
1100
+ into a single ``UnitaryGate`` representing the unitary matrix of the
1101
+ block.
1102
+
1103
+ Args:
1104
+ node_block (List[DAGNode]): A list of dag nodes that represents the
1105
+ node block to be replaced
1106
+ op (qiskit.circuit.Operation): The operation to replace the
1107
+ block with
1108
+ wire_pos_map (Dict[Bit, int]): The dictionary mapping the bits to their positions in the
1109
+ output ``qargs`` or ``cargs``. This is necessary to reconstruct the arg order over
1110
+ multiple gates in the combined single op node. If a :class:`.Bit` is not in the
1111
+ dictionary, it will not be added to the args; this can be useful when dealing with
1112
+ control-flow operations that have inherent bits in their ``condition`` or ``target``
1113
+ fields.
1114
+ cycle_check (bool): When set to True this method will check that
1115
+ replacing the provided ``node_block`` with a single node
1116
+ would introduce a cycle (which would invalidate the
1117
+ ``DAGCircuit``) and will raise a ``DAGCircuitError`` if a cycle
1118
+ would be introduced. This checking comes with a run time
1119
+ penalty. If you can guarantee that your input ``node_block`` is
1120
+ a contiguous block and won't introduce a cycle when it's
1121
+ contracted to a single node, this can be set to ``False`` to
1122
+ improve the runtime performance of this method.
1123
+
1124
+ Raises:
1125
+ DAGCircuitError: if ``cycle_check`` is set to ``True`` and replacing
1126
+ the specified block introduces a cycle or if ``node_block`` is
1127
+ empty.
1128
+
1129
+ Returns:
1130
+ DAGOpNode: The op node that replaces the block.
1131
+ """
1132
+ block_qargs = set()
1133
+ block_cargs = set()
1134
+ block_ids = [x._node_id for x in node_block]
1135
+
1136
+ # If node block is empty return early
1137
+ if not node_block:
1138
+ raise DAGCircuitError("Can't replace an empty node_block")
1139
+
1140
+ for nd in node_block:
1141
+ block_qargs |= set(nd.qargs)
1142
+ block_cargs |= set(nd.cargs)
1143
+ if (condition := getattr(nd.op, "condition", None)) is not None:
1144
+ block_cargs.update(condition_resources(condition).clbits)
1145
+ elif isinstance(nd.op, SwitchCaseOp):
1146
+ if isinstance(nd.op.target, Clbit):
1147
+ block_cargs.add(nd.op.target)
1148
+ elif isinstance(nd.op.target, ClassicalRegister):
1149
+ block_cargs.update(nd.op.target)
1150
+ else:
1151
+ block_cargs.update(node_resources(nd.op.target).clbits)
1152
+
1153
+ block_qargs = [bit for bit in block_qargs if bit in wire_pos_map]
1154
+ block_qargs.sort(key=wire_pos_map.get)
1155
+ block_cargs = [bit for bit in block_cargs if bit in wire_pos_map]
1156
+ block_cargs.sort(key=wire_pos_map.get)
1157
+ new_node = DAGOpNode(op, block_qargs, block_cargs, dag=self)
1158
+
1159
+ try:
1160
+ new_node._node_id = self._multi_graph.contract_nodes(
1161
+ block_ids, new_node, check_cycle=cycle_check
1162
+ )
1163
+ except rx.DAGWouldCycle as ex:
1164
+ raise DAGCircuitError(
1165
+ "Replacing the specified node block would introduce a cycle"
1166
+ ) from ex
1167
+
1168
+ self._increment_op(op)
1169
+
1170
+ for nd in node_block:
1171
+ self._decrement_op(nd.op)
1172
+
1173
+ return new_node
1174
+
1175
+ def substitute_node_with_dag(self, node, input_dag, wires=None, propagate_condition=True):
1176
+ """Replace one node with dag.
1177
+
1178
+ Args:
1179
+ node (DAGOpNode): node to substitute
1180
+ input_dag (DAGCircuit): circuit that will substitute the node
1181
+ wires (list[Bit] | Dict[Bit, Bit]): gives an order for (qu)bits
1182
+ in the input circuit. If a list, then the bits refer to those in the ``input_dag``,
1183
+ and the order gets matched to the node wires by qargs first, then cargs, then
1184
+ conditions. If a dictionary, then a mapping of bits in the ``input_dag`` to those
1185
+ that the ``node`` acts on.
1186
+ propagate_condition (bool): If ``True`` (default), then any ``condition`` attribute on
1187
+ the operation within ``node`` is propagated to each node in the ``input_dag``. If
1188
+ ``False``, then the ``input_dag`` is assumed to faithfully implement suitable
1189
+ conditional logic already. This is ignored for :class:`.ControlFlowOp`\\ s (i.e.
1190
+ treated as if it is ``False``); replacements of those must already fulfill the same
1191
+ conditional logic or this function would be close to useless for them.
1192
+
1193
+ Returns:
1194
+ dict: maps node IDs from `input_dag` to their new node incarnations in `self`.
1195
+
1196
+ Raises:
1197
+ DAGCircuitError: if met with unexpected predecessor/successors
1198
+ """
1199
+ if not isinstance(node, DAGOpNode):
1200
+ raise DAGCircuitError(f"expected node DAGOpNode, got {type(node)}")
1201
+
1202
+ if isinstance(wires, dict):
1203
+ wire_map = wires
1204
+ else:
1205
+ wires = input_dag.wires if wires is None else wires
1206
+ node_cargs = set(node.cargs)
1207
+ node_wire_order = list(node.qargs) + list(node.cargs)
1208
+ # If we're not propagating it, the number of wires in the input DAG should include the
1209
+ # condition as well.
1210
+ if not propagate_condition and self._operation_may_have_bits(node.op):
1211
+ node_wire_order += [
1212
+ bit for bit in self._bits_in_operation(node.op) if bit not in node_cargs
1213
+ ]
1214
+ if len(wires) != len(node_wire_order):
1215
+ raise DAGCircuitError(
1216
+ f"bit mapping invalid: expected {len(node_wire_order)}, got {len(wires)}"
1217
+ )
1218
+ wire_map = dict(zip(wires, node_wire_order))
1219
+ if len(wire_map) != len(node_wire_order):
1220
+ raise DAGCircuitError("bit mapping invalid: some bits have duplicate entries")
1221
+ for input_dag_wire, our_wire in wire_map.items():
1222
+ if our_wire not in self.input_map:
1223
+ raise DAGCircuitError(f"bit mapping invalid: {our_wire} is not in this DAG")
1224
+ # Support mapping indiscriminately between Qubit and AncillaQubit, etc.
1225
+ check_type = Qubit if isinstance(our_wire, Qubit) else Clbit
1226
+ if not isinstance(input_dag_wire, check_type):
1227
+ raise DAGCircuitError(
1228
+ f"bit mapping invalid: {input_dag_wire} and {our_wire} are different bit types"
1229
+ )
1230
+
1231
+ reverse_wire_map = {b: a for a, b in wire_map.items()}
1232
+ # It doesn't make sense to try and propagate a condition from a control-flow op; a
1233
+ # replacement for the control-flow op should implement the operation completely.
1234
+ if (
1235
+ propagate_condition
1236
+ and not isinstance(node.op, ControlFlowOp)
1237
+ and (op_condition := getattr(node.op, "condition", None)) is not None
1238
+ ):
1239
+ in_dag = input_dag.copy_empty_like()
1240
+ # The remapping of `condition` below is still using the old code that assumes a 2-tuple.
1241
+ # This is because this remapping code only makes sense in the case of non-control-flow
1242
+ # operations being replaced. These can only have the 2-tuple conditions, and the
1243
+ # ability to set a condition at an individual node level will be deprecated and removed
1244
+ # in favour of the new-style conditional blocks. The extra logic in here to add
1245
+ # additional wires into the map as necessary would hugely complicate matters if we tried
1246
+ # to abstract it out into the `VariableMapper` used elsewhere.
1247
+ target, value = op_condition
1248
+ if isinstance(target, Clbit):
1249
+ new_target = reverse_wire_map.get(target, Clbit())
1250
+ if new_target not in wire_map:
1251
+ in_dag.add_clbits([new_target])
1252
+ wire_map[new_target], reverse_wire_map[target] = target, new_target
1253
+ target_cargs = {new_target}
1254
+ else: # ClassicalRegister
1255
+ mapped_bits = [reverse_wire_map.get(bit, Clbit()) for bit in target]
1256
+ for ours, theirs in zip(target, mapped_bits):
1257
+ # Update to any new dummy bits we just created to the wire maps.
1258
+ wire_map[theirs], reverse_wire_map[ours] = ours, theirs
1259
+ new_target = ClassicalRegister(bits=mapped_bits)
1260
+ in_dag.add_creg(new_target)
1261
+ target_cargs = set(new_target)
1262
+ new_condition = (new_target, value)
1263
+ for in_node in input_dag.topological_op_nodes():
1264
+ if getattr(in_node.op, "condition", None) is not None:
1265
+ raise DAGCircuitError(
1266
+ "cannot propagate a condition to an element that already has one"
1267
+ )
1268
+ if target_cargs.intersection(in_node.cargs):
1269
+ # This is for backwards compatibility with early versions of the method, as it is
1270
+ # a tested part of the API. In the newer model of a condition being an integral
1271
+ # part of the operation (not a separate property to be copied over), this error
1272
+ # is overzealous, because it forbids a custom instruction from implementing the
1273
+ # condition within its definition rather than at the top level.
1274
+ raise DAGCircuitError(
1275
+ "cannot propagate a condition to an element that acts on those bits"
1276
+ )
1277
+ new_op = copy.copy(in_node.op)
1278
+ if new_condition:
1279
+ if not isinstance(new_op, ControlFlowOp):
1280
+ new_op = new_op.c_if(*new_condition)
1281
+ else:
1282
+ new_op.condition = new_condition
1283
+ in_dag.apply_operation_back(new_op, in_node.qargs, in_node.cargs, check=False)
1284
+ else:
1285
+ in_dag = input_dag
1286
+
1287
+ if in_dag.global_phase:
1288
+ self.global_phase += in_dag.global_phase
1289
+
1290
+ # Add wire from pred to succ if no ops on mapped wire on ``in_dag``
1291
+ # rustworkx's substitute_node_with_subgraph lacks the DAGCircuit
1292
+ # context to know what to do in this case (the method won't even see
1293
+ # these nodes because they're filtered) so we manually retain the
1294
+ # edges prior to calling substitute_node_with_subgraph and set the
1295
+ # edge_map_fn callback kwarg to skip these edges when they're
1296
+ # encountered.
1297
+ for in_dag_wire, self_wire in wire_map.items():
1298
+ input_node = in_dag.input_map[in_dag_wire]
1299
+ output_node = in_dag.output_map[in_dag_wire]
1300
+ if in_dag._multi_graph.has_edge(input_node._node_id, output_node._node_id):
1301
+ pred = self._multi_graph.find_predecessors_by_edge(
1302
+ node._node_id, lambda edge, wire=self_wire: edge == wire
1303
+ )[0]
1304
+ succ = self._multi_graph.find_successors_by_edge(
1305
+ node._node_id, lambda edge, wire=self_wire: edge == wire
1306
+ )[0]
1307
+ self._multi_graph.add_edge(pred._node_id, succ._node_id, self_wire)
1308
+
1309
+ # Exlude any nodes from in_dag that are not a DAGOpNode or are on
1310
+ # bits outside the set specified by the wires kwarg
1311
+ def filter_fn(node):
1312
+ if not isinstance(node, DAGOpNode):
1313
+ return False
1314
+ for qarg in node.qargs:
1315
+ if qarg not in wire_map:
1316
+ return False
1317
+ return True
1318
+
1319
+ # Map edges into and out of node to the appropriate node from in_dag
1320
+ def edge_map_fn(source, _target, self_wire):
1321
+ wire = reverse_wire_map[self_wire]
1322
+ # successor edge
1323
+ if source == node._node_id:
1324
+ wire_output_id = in_dag.output_map[wire]._node_id
1325
+ out_index = in_dag._multi_graph.predecessor_indices(wire_output_id)[0]
1326
+ # Edge directly from from input nodes to output nodes in in_dag are
1327
+ # already handled prior to calling rustworkx. Don't map these edges
1328
+ # in rustworkx.
1329
+ if not isinstance(in_dag._multi_graph[out_index], DAGOpNode):
1330
+ return None
1331
+ # predecessor edge
1332
+ else:
1333
+ wire_input_id = in_dag.input_map[wire]._node_id
1334
+ out_index = in_dag._multi_graph.successor_indices(wire_input_id)[0]
1335
+ # Edge directly from from input nodes to output nodes in in_dag are
1336
+ # already handled prior to calling rustworkx. Don't map these edges
1337
+ # in rustworkx.
1338
+ if not isinstance(in_dag._multi_graph[out_index], DAGOpNode):
1339
+ return None
1340
+ return out_index
1341
+
1342
+ # Adjust edge weights from in_dag
1343
+ def edge_weight_map(wire):
1344
+ return wire_map[wire]
1345
+
1346
+ node_map = self._multi_graph.substitute_node_with_subgraph(
1347
+ node._node_id, in_dag._multi_graph, edge_map_fn, filter_fn, edge_weight_map
1348
+ )
1349
+ self._decrement_op(node.op)
1350
+
1351
+ variable_mapper = _classical_resource_map.VariableMapper(
1352
+ self.cregs.values(), wire_map, self.add_creg
1353
+ )
1354
+ # Iterate over nodes of input_circuit and update wires in node objects migrated
1355
+ # from in_dag
1356
+ for old_node_index, new_node_index in node_map.items():
1357
+ # update node attributes
1358
+ old_node = in_dag._multi_graph[old_node_index]
1359
+ if isinstance(old_node.op, SwitchCaseOp):
1360
+ m_op = SwitchCaseOp(
1361
+ variable_mapper.map_target(old_node.op.target),
1362
+ old_node.op.cases_specifier(),
1363
+ label=old_node.op.label,
1364
+ )
1365
+ elif getattr(old_node.op, "condition", None) is not None:
1366
+ m_op = old_node.op
1367
+ if not isinstance(old_node.op, ControlFlowOp):
1368
+ new_condition = variable_mapper.map_condition(m_op.condition)
1369
+ if new_condition is not None:
1370
+ m_op = m_op.c_if(*new_condition)
1371
+ else:
1372
+ m_op.condition = variable_mapper.map_condition(m_op.condition)
1373
+ else:
1374
+ m_op = old_node.op
1375
+ m_qargs = [wire_map[x] for x in old_node.qargs]
1376
+ m_cargs = [wire_map[x] for x in old_node.cargs]
1377
+ new_node = DAGOpNode(m_op, qargs=m_qargs, cargs=m_cargs, dag=self)
1378
+ new_node._node_id = new_node_index
1379
+ self._multi_graph[new_node_index] = new_node
1380
+ self._increment_op(new_node.op)
1381
+
1382
+ return {k: self._multi_graph[v] for k, v in node_map.items()}
1383
+
1384
+ def substitute_node(self, node, op, inplace=False, propagate_condition=True):
1385
+ """Replace an DAGOpNode with a single operation. qargs, cargs and
1386
+ conditions for the new operation will be inferred from the node to be
1387
+ replaced. The new operation will be checked to match the shape of the
1388
+ replaced operation.
1389
+
1390
+ Args:
1391
+ node (DAGOpNode): Node to be replaced
1392
+ op (qiskit.circuit.Operation): The :class:`qiskit.circuit.Operation`
1393
+ instance to be added to the DAG
1394
+ inplace (bool): Optional, default False. If True, existing DAG node
1395
+ will be modified to include op. Otherwise, a new DAG node will
1396
+ be used.
1397
+ propagate_condition (bool): Optional, default True. If True, a condition on the
1398
+ ``node`` to be replaced will be applied to the new ``op``. This is the legacy
1399
+ behaviour. If either node is a control-flow operation, this will be ignored. If
1400
+ the ``op`` already has a condition, :exc:`.DAGCircuitError` is raised.
1401
+
1402
+ Returns:
1403
+ DAGOpNode: the new node containing the added operation.
1404
+
1405
+ Raises:
1406
+ DAGCircuitError: If replacement operation was incompatible with
1407
+ location of target node.
1408
+ """
1409
+
1410
+ if not isinstance(node, DAGOpNode):
1411
+ raise DAGCircuitError("Only DAGOpNodes can be replaced.")
1412
+
1413
+ if node.op.num_qubits != op.num_qubits or node.op.num_clbits != op.num_clbits:
1414
+ raise DAGCircuitError(
1415
+ "Cannot replace node of width ({} qubits, {} clbits) with "
1416
+ "operation of mismatched width ({} qubits, {} clbits).".format(
1417
+ node.op.num_qubits, node.op.num_clbits, op.num_qubits, op.num_clbits
1418
+ )
1419
+ )
1420
+
1421
+ # This might include wires that are inherent to the node, like in its `condition` or
1422
+ # `target` fields, so might be wider than `node.op.num_{qu,cl}bits`.
1423
+ current_wires = {wire for _, _, wire in self.edges(node)}
1424
+ new_wires = set(node.qargs) | set(node.cargs)
1425
+ if (new_condition := getattr(op, "condition", None)) is not None:
1426
+ new_wires.update(condition_resources(new_condition).clbits)
1427
+ elif isinstance(op, SwitchCaseOp):
1428
+ if isinstance(op.target, Clbit):
1429
+ new_wires.add(op.target)
1430
+ elif isinstance(op.target, ClassicalRegister):
1431
+ new_wires.update(op.target)
1432
+ else:
1433
+ new_wires.update(node_resources(op.target).clbits)
1434
+
1435
+ if propagate_condition and not (
1436
+ isinstance(node.op, ControlFlowOp) or isinstance(op, ControlFlowOp)
1437
+ ):
1438
+ if new_condition is not None:
1439
+ raise DAGCircuitError(
1440
+ "Cannot propagate a condition to an operation that already has one."
1441
+ )
1442
+ if (old_condition := getattr(node.op, "condition", None)) is not None:
1443
+ if not isinstance(op, Instruction):
1444
+ raise DAGCircuitError("Cannot add a condition on a generic Operation.")
1445
+ if not isinstance(node.op, ControlFlowOp):
1446
+ op = op.c_if(*old_condition)
1447
+ else:
1448
+ op.condition = old_condition
1449
+ new_wires.update(condition_resources(old_condition).clbits)
1450
+
1451
+ if new_wires != current_wires:
1452
+ # The new wires must be a non-strict subset of the current wires; if they add new wires,
1453
+ # we'd not know where to cut the existing wire to insert the new dependency.
1454
+ raise DAGCircuitError(
1455
+ f"New operation '{op}' does not span the same wires as the old node '{node}'."
1456
+ f" New wires: {new_wires}, old wires: {current_wires}."
1457
+ )
1458
+
1459
+ if inplace:
1460
+ if op.name != node.op.name:
1461
+ self._increment_op(op)
1462
+ self._decrement_op(node.op)
1463
+ node.op = op
1464
+ return node
1465
+
1466
+ new_node = copy.copy(node)
1467
+ new_node.op = op
1468
+ self._multi_graph[node._node_id] = new_node
1469
+ if op.name != node.op.name:
1470
+ self._increment_op(op)
1471
+ self._decrement_op(node.op)
1472
+ return new_node
1473
+
1474
+ def separable_circuits(self, remove_idle_qubits=False) -> List["DAGCircuit"]:
1475
+ """Decompose the circuit into sets of qubits with no gates connecting them.
1476
+
1477
+ Args:
1478
+ remove_idle_qubits (bool): Flag denoting whether to remove idle qubits from
1479
+ the separated circuits. If ``False``, each output circuit will contain the
1480
+ same number of qubits as ``self``.
1481
+
1482
+ Returns:
1483
+ List[DAGCircuit]: The circuits resulting from separating ``self`` into sets
1484
+ of disconnected qubits
1485
+
1486
+ Each :class:`~.DAGCircuit` instance returned by this method will contain the same number of
1487
+ clbits as ``self``. The global phase information in ``self`` will not be maintained
1488
+ in the subcircuits returned by this method.
1489
+ """
1490
+ connected_components = rx.weakly_connected_components(self._multi_graph)
1491
+
1492
+ # Collect each disconnected subgraph
1493
+ disconnected_subgraphs = []
1494
+ for components in connected_components:
1495
+ disconnected_subgraphs.append(self._multi_graph.subgraph(list(components)))
1496
+
1497
+ # Helper function for ensuring rustworkx nodes are returned in lexicographical,
1498
+ # topological order
1499
+ def _key(x):
1500
+ return x.sort_key
1501
+
1502
+ # Create new DAGCircuit objects from each of the rustworkx subgraph objects
1503
+ decomposed_dags = []
1504
+ for subgraph in disconnected_subgraphs:
1505
+ new_dag = self.copy_empty_like()
1506
+ new_dag.global_phase = 0
1507
+ subgraph_is_classical = True
1508
+ for node in rx.lexicographical_topological_sort(subgraph, key=_key):
1509
+ if isinstance(node, DAGInNode):
1510
+ if isinstance(node.wire, Qubit):
1511
+ subgraph_is_classical = False
1512
+ if not isinstance(node, DAGOpNode):
1513
+ continue
1514
+ new_dag.apply_operation_back(node.op, node.qargs, node.cargs, check=False)
1515
+
1516
+ # Ignore DAGs created for empty clbits
1517
+ if not subgraph_is_classical:
1518
+ decomposed_dags.append(new_dag)
1519
+
1520
+ if remove_idle_qubits:
1521
+ for dag in decomposed_dags:
1522
+ dag.remove_qubits(*(bit for bit in dag.idle_wires() if isinstance(bit, Qubit)))
1523
+
1524
+ return decomposed_dags
1525
+
1526
+ def swap_nodes(self, node1, node2):
1527
+ """Swap connected nodes e.g. due to commutation.
1528
+
1529
+ Args:
1530
+ node1 (OpNode): predecessor node
1531
+ node2 (OpNode): successor node
1532
+
1533
+ Raises:
1534
+ DAGCircuitError: if either node is not an OpNode or nodes are not connected
1535
+ """
1536
+ if not (isinstance(node1, DAGOpNode) and isinstance(node2, DAGOpNode)):
1537
+ raise DAGCircuitError("nodes to swap are not both DAGOpNodes")
1538
+ try:
1539
+ connected_edges = self._multi_graph.get_all_edge_data(node1._node_id, node2._node_id)
1540
+ except rx.NoEdgeBetweenNodes as no_common_edge:
1541
+ raise DAGCircuitError("attempt to swap unconnected nodes") from no_common_edge
1542
+ node1_id = node1._node_id
1543
+ node2_id = node2._node_id
1544
+ for edge in connected_edges[::-1]:
1545
+ edge_find = lambda x, y=edge: x == y
1546
+ edge_parent = self._multi_graph.find_predecessors_by_edge(node1_id, edge_find)[0]
1547
+ self._multi_graph.remove_edge(edge_parent._node_id, node1_id)
1548
+ self._multi_graph.add_edge(edge_parent._node_id, node2_id, edge)
1549
+ edge_child = self._multi_graph.find_successors_by_edge(node2_id, edge_find)[0]
1550
+ self._multi_graph.remove_edge(node1_id, node2_id)
1551
+ self._multi_graph.add_edge(node2_id, node1_id, edge)
1552
+ self._multi_graph.remove_edge(node2_id, edge_child._node_id)
1553
+ self._multi_graph.add_edge(node1_id, edge_child._node_id, edge)
1554
+
1555
+ def node(self, node_id):
1556
+ """Get the node in the dag.
1557
+
1558
+ Args:
1559
+ node_id(int): Node identifier.
1560
+
1561
+ Returns:
1562
+ node: the node.
1563
+ """
1564
+ return self._multi_graph[node_id]
1565
+
1566
+ def nodes(self):
1567
+ """Iterator for node values.
1568
+
1569
+ Yield:
1570
+ node: the node.
1571
+ """
1572
+ yield from self._multi_graph.nodes()
1573
+
1574
+ def edges(self, nodes=None):
1575
+ """Iterator for edge values and source and dest node
1576
+
1577
+ This works by returning the output edges from the specified nodes. If
1578
+ no nodes are specified all edges from the graph are returned.
1579
+
1580
+ Args:
1581
+ nodes(DAGOpNode, DAGInNode, or DAGOutNode|list(DAGOpNode, DAGInNode, or DAGOutNode):
1582
+ Either a list of nodes or a single input node. If none is specified,
1583
+ all edges are returned from the graph.
1584
+
1585
+ Yield:
1586
+ edge: the edge in the same format as out_edges the tuple
1587
+ (source node, destination node, edge data)
1588
+ """
1589
+ if nodes is None:
1590
+ nodes = self._multi_graph.nodes()
1591
+
1592
+ elif isinstance(nodes, (DAGOpNode, DAGInNode, DAGOutNode)):
1593
+ nodes = [nodes]
1594
+ for node in nodes:
1595
+ raw_nodes = self._multi_graph.out_edges(node._node_id)
1596
+ for source, dest, edge in raw_nodes:
1597
+ yield (self._multi_graph[source], self._multi_graph[dest], edge)
1598
+
1599
+ def op_nodes(self, op=None, include_directives=True):
1600
+ """Get the list of "op" nodes in the dag.
1601
+
1602
+ Args:
1603
+ op (Type): :class:`qiskit.circuit.Operation` subclass op nodes to
1604
+ return. If None, return all op nodes.
1605
+ include_directives (bool): include `barrier`, `snapshot` etc.
1606
+
1607
+ Returns:
1608
+ list[DAGOpNode]: the list of node ids containing the given op.
1609
+ """
1610
+ nodes = []
1611
+ for node in self._multi_graph.nodes():
1612
+ if isinstance(node, DAGOpNode):
1613
+ if not include_directives and getattr(node.op, "_directive", False):
1614
+ continue
1615
+ if op is None or isinstance(node.op, op):
1616
+ nodes.append(node)
1617
+ return nodes
1618
+
1619
+ def gate_nodes(self):
1620
+ """Get the list of gate nodes in the dag.
1621
+
1622
+ Returns:
1623
+ list[DAGOpNode]: the list of DAGOpNodes that represent gates.
1624
+ """
1625
+ nodes = []
1626
+ for node in self.op_nodes():
1627
+ if isinstance(node.op, Gate):
1628
+ nodes.append(node)
1629
+ return nodes
1630
+
1631
+ def named_nodes(self, *names):
1632
+ """Get the set of "op" nodes with the given name."""
1633
+ named_nodes = []
1634
+ for node in self._multi_graph.nodes():
1635
+ if isinstance(node, DAGOpNode) and node.op.name in names:
1636
+ named_nodes.append(node)
1637
+ return named_nodes
1638
+
1639
+ def two_qubit_ops(self):
1640
+ """Get list of 2 qubit operations. Ignore directives like snapshot and barrier."""
1641
+ ops = []
1642
+ for node in self.op_nodes(include_directives=False):
1643
+ if len(node.qargs) == 2:
1644
+ ops.append(node)
1645
+ return ops
1646
+
1647
+ def multi_qubit_ops(self):
1648
+ """Get list of 3+ qubit operations. Ignore directives like snapshot and barrier."""
1649
+ ops = []
1650
+ for node in self.op_nodes(include_directives=False):
1651
+ if len(node.qargs) >= 3:
1652
+ ops.append(node)
1653
+ return ops
1654
+
1655
+ def longest_path(self):
1656
+ """Returns the longest path in the dag as a list of DAGOpNodes, DAGInNodes, and DAGOutNodes."""
1657
+ return [self._multi_graph[x] for x in rx.dag_longest_path(self._multi_graph)]
1658
+
1659
+ def successors(self, node):
1660
+ """Returns iterator of the successors of a node as DAGOpNodes and DAGOutNodes."""
1661
+ return iter(self._multi_graph.successors(node._node_id))
1662
+
1663
+ def predecessors(self, node):
1664
+ """Returns iterator of the predecessors of a node as DAGOpNodes and DAGInNodes."""
1665
+ return iter(self._multi_graph.predecessors(node._node_id))
1666
+
1667
+ def is_successor(self, node, node_succ):
1668
+ """Checks if a second node is in the successors of node."""
1669
+ return self._multi_graph.has_edge(node._node_id, node_succ._node_id)
1670
+
1671
+ def is_predecessor(self, node, node_pred):
1672
+ """Checks if a second node is in the predecessors of node."""
1673
+ return self._multi_graph.has_edge(node_pred._node_id, node._node_id)
1674
+
1675
+ def quantum_predecessors(self, node):
1676
+ """Returns iterator of the predecessors of a node that are
1677
+ connected by a quantum edge as DAGOpNodes and DAGInNodes."""
1678
+ return iter(
1679
+ self._multi_graph.find_predecessors_by_edge(
1680
+ node._node_id, lambda edge_data: isinstance(edge_data, Qubit)
1681
+ )
1682
+ )
1683
+
1684
+ def classical_predecessors(self, node):
1685
+ """Returns iterator of the predecessors of a node that are
1686
+ connected by a classical edge as DAGOpNodes and DAGInNodes."""
1687
+ return iter(
1688
+ self._multi_graph.find_predecessors_by_edge(
1689
+ node._node_id, lambda edge_data: isinstance(edge_data, Clbit)
1690
+ )
1691
+ )
1692
+
1693
+ def ancestors(self, node):
1694
+ """Returns set of the ancestors of a node as DAGOpNodes and DAGInNodes."""
1695
+ return {self._multi_graph[x] for x in rx.ancestors(self._multi_graph, node._node_id)}
1696
+
1697
+ def descendants(self, node):
1698
+ """Returns set of the descendants of a node as DAGOpNodes and DAGOutNodes."""
1699
+ return {self._multi_graph[x] for x in rx.descendants(self._multi_graph, node._node_id)}
1700
+
1701
+ def bfs_successors(self, node):
1702
+ """
1703
+ Returns an iterator of tuples of (DAGNode, [DAGNodes]) where the DAGNode is the current node
1704
+ and [DAGNode] is its successors in BFS order.
1705
+ """
1706
+ return iter(rx.bfs_successors(self._multi_graph, node._node_id))
1707
+
1708
+ def quantum_successors(self, node):
1709
+ """Returns iterator of the successors of a node that are
1710
+ connected by a quantum edge as Opnodes and DAGOutNodes."""
1711
+ return iter(
1712
+ self._multi_graph.find_successors_by_edge(
1713
+ node._node_id, lambda edge_data: isinstance(edge_data, Qubit)
1714
+ )
1715
+ )
1716
+
1717
+ def classical_successors(self, node):
1718
+ """Returns iterator of the successors of a node that are
1719
+ connected by a classical edge as DAGOpNodes and DAGInNodes."""
1720
+ return iter(
1721
+ self._multi_graph.find_successors_by_edge(
1722
+ node._node_id, lambda edge_data: isinstance(edge_data, Clbit)
1723
+ )
1724
+ )
1725
+
1726
+ def remove_op_node(self, node):
1727
+ """Remove an operation node n.
1728
+
1729
+ Add edges from predecessors to successors.
1730
+ """
1731
+ if not isinstance(node, DAGOpNode):
1732
+ raise DAGCircuitError(
1733
+ 'The method remove_op_node only works on DAGOpNodes. A "%s" '
1734
+ "node type was wrongly provided." % type(node)
1735
+ )
1736
+
1737
+ self._multi_graph.remove_node_retain_edges(
1738
+ node._node_id, use_outgoing=False, condition=lambda edge1, edge2: edge1 == edge2
1739
+ )
1740
+ self._decrement_op(node.op)
1741
+
1742
+ def remove_ancestors_of(self, node):
1743
+ """Remove all of the ancestor operation nodes of node."""
1744
+ anc = rx.ancestors(self._multi_graph, node)
1745
+ # TODO: probably better to do all at once using
1746
+ # multi_graph.remove_nodes_from; same for related functions ...
1747
+
1748
+ for anc_node in anc:
1749
+ if isinstance(anc_node, DAGOpNode):
1750
+ self.remove_op_node(anc_node)
1751
+
1752
+ def remove_descendants_of(self, node):
1753
+ """Remove all of the descendant operation nodes of node."""
1754
+ desc = rx.descendants(self._multi_graph, node)
1755
+ for desc_node in desc:
1756
+ if isinstance(desc_node, DAGOpNode):
1757
+ self.remove_op_node(desc_node)
1758
+
1759
+ def remove_nonancestors_of(self, node):
1760
+ """Remove all of the non-ancestors operation nodes of node."""
1761
+ anc = rx.ancestors(self._multi_graph, node)
1762
+ comp = list(set(self._multi_graph.nodes()) - set(anc))
1763
+ for n in comp:
1764
+ if isinstance(n, DAGOpNode):
1765
+ self.remove_op_node(n)
1766
+
1767
+ def remove_nondescendants_of(self, node):
1768
+ """Remove all of the non-descendants operation nodes of node."""
1769
+ dec = rx.descendants(self._multi_graph, node)
1770
+ comp = list(set(self._multi_graph.nodes()) - set(dec))
1771
+ for n in comp:
1772
+ if isinstance(n, DAGOpNode):
1773
+ self.remove_op_node(n)
1774
+
1775
+ def front_layer(self):
1776
+ """Return a list of op nodes in the first layer of this dag."""
1777
+ graph_layers = self.multigraph_layers()
1778
+ try:
1779
+ next(graph_layers) # Remove input nodes
1780
+ except StopIteration:
1781
+ return []
1782
+
1783
+ op_nodes = [node for node in next(graph_layers) if isinstance(node, DAGOpNode)]
1784
+
1785
+ return op_nodes
1786
+
1787
+ def layers(self):
1788
+ """Yield a shallow view on a layer of this DAGCircuit for all d layers of this circuit.
1789
+
1790
+ A layer is a circuit whose gates act on disjoint qubits, i.e.,
1791
+ a layer has depth 1. The total number of layers equals the
1792
+ circuit depth d. The layers are indexed from 0 to d-1 with the
1793
+ earliest layer at index 0. The layers are constructed using a
1794
+ greedy algorithm. Each returned layer is a dict containing
1795
+ {"graph": circuit graph, "partition": list of qubit lists}.
1796
+
1797
+ The returned layer contains new (but semantically equivalent) DAGOpNodes, DAGInNodes,
1798
+ and DAGOutNodes. These are not the same as nodes of the original dag, but are equivalent
1799
+ via DAGNode.semantic_eq(node1, node2).
1800
+
1801
+ TODO: Gates that use the same cbits will end up in different
1802
+ layers as this is currently implemented. This may not be
1803
+ the desired behavior.
1804
+ """
1805
+ graph_layers = self.multigraph_layers()
1806
+ try:
1807
+ next(graph_layers) # Remove input nodes
1808
+ except StopIteration:
1809
+ return
1810
+
1811
+ for graph_layer in graph_layers:
1812
+
1813
+ # Get the op nodes from the layer, removing any input and output nodes.
1814
+ op_nodes = [node for node in graph_layer if isinstance(node, DAGOpNode)]
1815
+
1816
+ # Sort to make sure they are in the order they were added to the original DAG
1817
+ # It has to be done by node_id as graph_layer is just a list of nodes
1818
+ # with no implied topology
1819
+ # Drawing tools rely on _node_id to infer order of node creation
1820
+ # so we need this to be preserved by layers()
1821
+ op_nodes.sort(key=lambda nd: nd._node_id)
1822
+
1823
+ # Stop yielding once there are no more op_nodes in a layer.
1824
+ if not op_nodes:
1825
+ return
1826
+
1827
+ # Construct a shallow copy of self
1828
+ new_layer = self.copy_empty_like()
1829
+
1830
+ for node in op_nodes:
1831
+ # this creates new DAGOpNodes in the new_layer
1832
+ new_layer.apply_operation_back(node.op, node.qargs, node.cargs, check=False)
1833
+
1834
+ # The quantum registers that have an operation in this layer.
1835
+ support_list = [
1836
+ op_node.qargs
1837
+ for op_node in new_layer.op_nodes()
1838
+ if not getattr(op_node.op, "_directive", False)
1839
+ ]
1840
+
1841
+ yield {"graph": new_layer, "partition": support_list}
1842
+
1843
+ def serial_layers(self):
1844
+ """Yield a layer for all gates of this circuit.
1845
+
1846
+ A serial layer is a circuit with one gate. The layers have the
1847
+ same structure as in layers().
1848
+ """
1849
+ for next_node in self.topological_op_nodes():
1850
+ new_layer = self.copy_empty_like()
1851
+
1852
+ # Save the support of the operation we add to the layer
1853
+ support_list = []
1854
+ # Operation data
1855
+ op = copy.copy(next_node.op)
1856
+ qargs = copy.copy(next_node.qargs)
1857
+ cargs = copy.copy(next_node.cargs)
1858
+
1859
+ # Add node to new_layer
1860
+ new_layer.apply_operation_back(op, qargs, cargs, check=False)
1861
+ # Add operation to partition
1862
+ if not getattr(next_node.op, "_directive", False):
1863
+ support_list.append(list(qargs))
1864
+ l_dict = {"graph": new_layer, "partition": support_list}
1865
+ yield l_dict
1866
+
1867
+ def multigraph_layers(self):
1868
+ """Yield layers of the multigraph."""
1869
+ first_layer = [x._node_id for x in self.input_map.values()]
1870
+ return iter(rx.layers(self._multi_graph, first_layer))
1871
+
1872
+ def collect_runs(self, namelist):
1873
+ """Return a set of non-conditional runs of "op" nodes with the given names.
1874
+
1875
+ For example, "... h q[0]; cx q[0],q[1]; cx q[0],q[1]; h q[1]; .."
1876
+ would produce the tuple of cx nodes as an element of the set returned
1877
+ from a call to collect_runs(["cx"]). If instead the cx nodes were
1878
+ "cx q[0],q[1]; cx q[1],q[0];", the method would still return the
1879
+ pair in a tuple. The namelist can contain names that are not
1880
+ in the circuit's basis.
1881
+
1882
+ Nodes must have only one successor to continue the run.
1883
+ """
1884
+
1885
+ def filter_fn(node):
1886
+ return (
1887
+ isinstance(node, DAGOpNode)
1888
+ and node.op.name in namelist
1889
+ and getattr(node.op, "condition", None) is None
1890
+ )
1891
+
1892
+ group_list = rx.collect_runs(self._multi_graph, filter_fn)
1893
+ return {tuple(x) for x in group_list}
1894
+
1895
+ def collect_1q_runs(self):
1896
+ """Return a set of non-conditional runs of 1q "op" nodes."""
1897
+
1898
+ def filter_fn(node):
1899
+ return (
1900
+ isinstance(node, DAGOpNode)
1901
+ and len(node.qargs) == 1
1902
+ and len(node.cargs) == 0
1903
+ and isinstance(node.op, Gate)
1904
+ and hasattr(node.op, "__array__")
1905
+ and getattr(node.op, "condition", None) is None
1906
+ and not node.op.is_parameterized()
1907
+ )
1908
+
1909
+ return rx.collect_runs(self._multi_graph, filter_fn)
1910
+
1911
+ def collect_2q_runs(self):
1912
+ """Return a set of non-conditional runs of 2q "op" nodes."""
1913
+
1914
+ to_qid = {}
1915
+ for i, qubit in enumerate(self.qubits):
1916
+ to_qid[qubit] = i
1917
+
1918
+ def filter_fn(node):
1919
+ if isinstance(node, DAGOpNode):
1920
+ return (
1921
+ isinstance(node.op, Gate)
1922
+ and len(node.qargs) <= 2
1923
+ and not getattr(node.op, "condition", None)
1924
+ and not node.op.is_parameterized()
1925
+ )
1926
+ else:
1927
+ return None
1928
+
1929
+ def color_fn(edge):
1930
+ if isinstance(edge, Qubit):
1931
+ return to_qid[edge]
1932
+ else:
1933
+ return None
1934
+
1935
+ return rx.collect_bicolor_runs(self._multi_graph, filter_fn, color_fn)
1936
+
1937
+ def nodes_on_wire(self, wire, only_ops=False):
1938
+ """
1939
+ Iterator for nodes that affect a given wire.
1940
+
1941
+ Args:
1942
+ wire (Bit): the wire to be looked at.
1943
+ only_ops (bool): True if only the ops nodes are wanted;
1944
+ otherwise, all nodes are returned.
1945
+ Yield:
1946
+ Iterator: the successive nodes on the given wire
1947
+
1948
+ Raises:
1949
+ DAGCircuitError: if the given wire doesn't exist in the DAG
1950
+ """
1951
+ current_node = self.input_map.get(wire, None)
1952
+
1953
+ if not current_node:
1954
+ raise DAGCircuitError("The given wire %s is not present in the circuit" % str(wire))
1955
+
1956
+ more_nodes = True
1957
+ while more_nodes:
1958
+ more_nodes = False
1959
+ # allow user to just get ops on the wire - not the input/output nodes
1960
+ if isinstance(current_node, DAGOpNode) or not only_ops:
1961
+ yield current_node
1962
+
1963
+ try:
1964
+ current_node = self._multi_graph.find_adjacent_node_by_edge(
1965
+ current_node._node_id, lambda x: wire == x
1966
+ )
1967
+ more_nodes = True
1968
+ except rx.NoSuitableNeighbors:
1969
+ pass
1970
+
1971
+ def count_ops(self, *, recurse: bool = True):
1972
+ """Count the occurrences of operation names.
1973
+
1974
+ Args:
1975
+ recurse: if ``True`` (default), then recurse into control-flow operations. In all
1976
+ cases, this counts only the number of times the operation appears in any possible
1977
+ block; both branches of if-elses are counted, and for- and while-loop blocks are
1978
+ only counted once.
1979
+
1980
+ Returns:
1981
+ Mapping[str, int]: a mapping of operation names to the number of times it appears.
1982
+ """
1983
+ if not recurse or not CONTROL_FLOW_OP_NAMES.intersection(self._op_names):
1984
+ return self._op_names.copy()
1985
+
1986
+ # pylint: disable=cyclic-import
1987
+ from qiskit.converters import circuit_to_dag
1988
+
1989
+ def inner(dag, counts):
1990
+ for name, count in dag._op_names.items():
1991
+ counts[name] += count
1992
+ for node in dag.op_nodes(ControlFlowOp):
1993
+ for block in node.op.blocks:
1994
+ counts = inner(circuit_to_dag(block), counts)
1995
+ return counts
1996
+
1997
+ return dict(inner(self, defaultdict(int)))
1998
+
1999
+ def count_ops_longest_path(self):
2000
+ """Count the occurrences of operation names on the longest path.
2001
+
2002
+ Returns a dictionary of counts keyed on the operation name.
2003
+ """
2004
+ op_dict = {}
2005
+ path = self.longest_path()
2006
+ path = path[1:-1] # remove qubits at beginning and end of path
2007
+ for node in path:
2008
+ name = node.op.name
2009
+ if name not in op_dict:
2010
+ op_dict[name] = 1
2011
+ else:
2012
+ op_dict[name] += 1
2013
+ return op_dict
2014
+
2015
+ def quantum_causal_cone(self, qubit):
2016
+ """
2017
+ Returns causal cone of a qubit.
2018
+
2019
+ A qubit's causal cone is the set of qubits that can influence the output of that
2020
+ qubit through interactions, whether through multi-qubit gates or operations. Knowing
2021
+ the causal cone of a qubit can be useful when debugging faulty circuits, as it can
2022
+ help identify which wire(s) may be causing the problem.
2023
+
2024
+ This method does not consider any classical data dependency in the ``DAGCircuit``,
2025
+ classical bit wires are ignored for the purposes of building the causal cone.
2026
+
2027
+ Args:
2028
+ qubit (~qiskit.circuit.Qubit): The output qubit for which we want to find the causal cone.
2029
+
2030
+ Returns:
2031
+ Set[~qiskit.circuit.Qubit]: The set of qubits whose interactions affect ``qubit``.
2032
+ """
2033
+ # Retrieve the output node from the qubit
2034
+ output_node = self.output_map.get(qubit, None)
2035
+ if not output_node:
2036
+ raise DAGCircuitError(f"Qubit {qubit} is not part of this circuit.")
2037
+ # Add the qubit to the causal cone.
2038
+ qubits_to_check = {qubit}
2039
+ # Add predecessors of output node to the queue.
2040
+ queue = deque(self.predecessors(output_node))
2041
+
2042
+ # While queue isn't empty
2043
+ while queue:
2044
+ # Pop first element.
2045
+ node_to_check = queue.popleft()
2046
+ # Check whether element is input or output node.
2047
+ if isinstance(node_to_check, DAGOpNode):
2048
+ # Keep all the qubits in the operation inside a set.
2049
+ qubit_set = set(node_to_check.qargs)
2050
+ # Check if there are any qubits in common and that the operation is not a barrier.
2051
+ if (
2052
+ len(qubit_set.intersection(qubits_to_check)) > 0
2053
+ and node_to_check.op.name != "barrier"
2054
+ and not getattr(node_to_check.op, "_directive")
2055
+ ):
2056
+ # If so, add all the qubits to the causal cone.
2057
+ qubits_to_check = qubits_to_check.union(qubit_set)
2058
+ # For each predecessor of the current node, filter input/output nodes,
2059
+ # also make sure it has at least one qubit in common. Then append.
2060
+ for node in self.quantum_predecessors(node_to_check):
2061
+ if (
2062
+ isinstance(node, DAGOpNode)
2063
+ and len(qubits_to_check.intersection(set(node.qargs))) > 0
2064
+ ):
2065
+ queue.append(node)
2066
+ return qubits_to_check
2067
+
2068
+ def properties(self):
2069
+ """Return a dictionary of circuit properties."""
2070
+ summary = {
2071
+ "size": self.size(),
2072
+ "depth": self.depth(),
2073
+ "width": self.width(),
2074
+ "qubits": self.num_qubits(),
2075
+ "bits": self.num_clbits(),
2076
+ "factors": self.num_tensor_factors(),
2077
+ "operations": self.count_ops(),
2078
+ }
2079
+ return summary
2080
+
2081
+ def draw(self, scale=0.7, filename=None, style="color"):
2082
+ """
2083
+ Draws the dag circuit.
2084
+
2085
+ This function needs `Graphviz <https://www.graphviz.org/>`_ to be
2086
+ installed. Graphviz is not a python package and can't be pip installed
2087
+ (the ``graphviz`` package on PyPI is a Python interface library for
2088
+ Graphviz and does not actually install Graphviz). You can refer to
2089
+ `the Graphviz documentation <https://www.graphviz.org/download/>`__ on
2090
+ how to install it.
2091
+
2092
+ Args:
2093
+ scale (float): scaling factor
2094
+ filename (str): file path to save image to (format inferred from name)
2095
+ style (str):
2096
+ 'plain': B&W graph;
2097
+ 'color' (default): color input/output/op nodes
2098
+
2099
+ Returns:
2100
+ Ipython.display.Image: if in Jupyter notebook and not saving to file,
2101
+ otherwise None.
2102
+ """
2103
+ from qiskit.visualization.dag_visualization import dag_drawer
2104
+
2105
+ return dag_drawer(dag=self, scale=scale, filename=filename, style=style)