diff --git a/doc/gates.md b/doc/gates.md index 05b314b59..de61175c3 100644 --- a/doc/gates.md +++ b/doc/gates.md @@ -26,6 +26,7 @@ - [CXSWAP](#CXSWAP) - [CY](#CY) - [CZ](#CZ) + - [CZSWAP](#CZSWAP) - [ISWAP](#ISWAP) - [ISWAP_DAG](#ISWAP_DAG) - [SQRT_XX](#SQRT_XX) @@ -36,6 +37,7 @@ - [SQRT_ZZ_DAG](#SQRT_ZZ_DAG) - [SWAP](#SWAP) - [SWAPCX](#SWAPCX) + - [SWAPCZ](#SWAPCZ) - [XCX](#XCX) - [XCY](#XCY) - [XCZ](#XCZ) @@ -1103,6 +1105,51 @@ Decomposition (into H, S, CX, M, R): H 1 + +### The 'CZSWAP' Gate + +Alternate name: `SWAPCZ` + +A combination CZ-and-SWAP gate. +This gate is kak-equivalent to the iswap gate. + +Parens Arguments: + + This instruction takes no parens arguments. + +Targets: + + Qubit pairs to operate on. + +Example: + + CZSWAP 5 6 + CZSWAP 42 43 + CZSWAP 5 6 42 43 + +Stabilizer Generators: + + X_ -> ZX + Z_ -> _Z + _X -> XZ + _Z -> Z_ + +Unitary Matrix (little endian): + + [+1 , , , ] + [ , , +1 , ] + [ , +1 , , ] + [ , , , -1 ] + +Decomposition (into H, S, CX, M, R): + + # The following circuit is equivalent (up to global phase) to `CZSWAP 0 1` + H 0 + CX 0 1 + CX 1 0 + H 1 + + ### The 'ISWAP' Gate diff --git a/file_lists/benchmark_files b/file_lists/benchmark_files index a2b5f1e28..3e16d7f9d 100644 --- a/file_lists/benchmark_files +++ b/file_lists/benchmark_files @@ -1,7 +1,7 @@ src/stim/benchmark_main.perf.cc src/stim/benchmark_util.perf.cc src/stim/circuit/circuit.perf.cc -src/stim/circuit/gate_data.perf.cc +src/stim/gates/gates.perf.cc src/stim/io/measure_record_reader.perf.cc src/stim/main_namespaced.perf.cc src/stim/mem/simd_bit_table.perf.cc diff --git a/file_lists/python_api_files b/file_lists/python_api_files index 50bc6ecac..25a80a28d 100644 --- a/file_lists/python_api_files +++ b/file_lists/python_api_files @@ -1,13 +1,13 @@ src/stim/circuit/circuit.pybind.cc src/stim/circuit/circuit_instruction.pybind.cc src/stim/circuit/circuit_repeat_block.pybind.cc -src/stim/circuit/gate_data.pybind.cc src/stim/circuit/gate_target.pybind.cc src/stim/cmd/command_diagram.pybind.cc src/stim/dem/detector_error_model.pybind.cc src/stim/dem/detector_error_model_instruction.pybind.cc src/stim/dem/detector_error_model_repeat_block.pybind.cc src/stim/dem/detector_error_model_target.pybind.cc +src/stim/gates/gates.pybind.cc src/stim/io/read_write.pybind.cc src/stim/py/base.pybind.cc src/stim/py/compiled_detector_sampler.pybind.cc diff --git a/file_lists/source_files_no_main b/file_lists/source_files_no_main index 27511b6ea..b9694b6e5 100644 --- a/file_lists/source_files_no_main +++ b/file_lists/source_files_no_main @@ -3,20 +3,6 @@ src/stim/arg_parse.cc src/stim/circuit/circuit.cc src/stim/circuit/circuit_instruction.cc src/stim/circuit/export_qasm.cc -src/stim/circuit/gate_data.cc -src/stim/circuit/gate_data_annotations.cc -src/stim/circuit/gate_data_blocks.cc -src/stim/circuit/gate_data_collapsing.cc -src/stim/circuit/gate_data_controlled.cc -src/stim/circuit/gate_data_hada.cc -src/stim/circuit/gate_data_heralded.cc -src/stim/circuit/gate_data_noisy.cc -src/stim/circuit/gate_data_pair_measure.cc -src/stim/circuit/gate_data_pauli.cc -src/stim/circuit/gate_data_period_3.cc -src/stim/circuit/gate_data_period_4.cc -src/stim/circuit/gate_data_pp.cc -src/stim/circuit/gate_data_swaps.cc src/stim/circuit/gate_decomposition.cc src/stim/circuit/gate_target.cc src/stim/cmd/command_analyze_errors.cc @@ -51,6 +37,20 @@ src/stim/diagram/lattice_map.cc src/stim/diagram/timeline/timeline_3d_drawer.cc src/stim/diagram/timeline/timeline_ascii_drawer.cc src/stim/diagram/timeline/timeline_svg_drawer.cc +src/stim/gates/gate_data_annotations.cc +src/stim/gates/gate_data_blocks.cc +src/stim/gates/gate_data_collapsing.cc +src/stim/gates/gate_data_controlled.cc +src/stim/gates/gate_data_hada.cc +src/stim/gates/gate_data_heralded.cc +src/stim/gates/gate_data_noisy.cc +src/stim/gates/gate_data_pair_measure.cc +src/stim/gates/gate_data_pauli.cc +src/stim/gates/gate_data_period_3.cc +src/stim/gates/gate_data_period_4.cc +src/stim/gates/gate_data_pp.cc +src/stim/gates/gate_data_swaps.cc +src/stim/gates/gates.cc src/stim/gen/circuit_gen_params.cc src/stim/gen/gen_color_code.cc src/stim/gen/gen_rep_code.cc diff --git a/file_lists/test_files b/file_lists/test_files index 51d1ffdef..6506e3f6a 100644 --- a/file_lists/test_files +++ b/file_lists/test_files @@ -2,7 +2,6 @@ src/stim.test.cc src/stim/arg_parse.test.cc src/stim/circuit/circuit.test.cc src/stim/circuit/export_qasm.test.cc -src/stim/circuit/gate_data.test.cc src/stim/circuit/gate_decomposition.test.cc src/stim/circuit/gate_target.test.cc src/stim/circuit/stabilizer_flow.test.cc @@ -26,6 +25,7 @@ src/stim/diagram/json_obj.test.cc src/stim/diagram/timeline/timeline_3d_drawer.test.cc src/stim/diagram/timeline/timeline_ascii_drawer.test.cc src/stim/diagram/timeline/timeline_svg_drawer.test.cc +src/stim/gates/gates.test.cc src/stim/gen/circuit_gen_params.test.cc src/stim/gen/gen_color_code.test.cc src/stim/gen/gen_rep_code.test.cc @@ -74,6 +74,7 @@ src/stim/stabilizers/conversions.test.cc src/stim/stabilizers/flex_pauli_string.test.cc src/stim/stabilizers/pauli_string.test.cc src/stim/stabilizers/pauli_string_iter.test.cc +src/stim/stabilizers/pauli_string_ref.test.cc src/stim/stabilizers/tableau.test.cc src/stim/stabilizers/tableau_iter.test.cc src/stim/str_util.test.cc diff --git a/glue/cirq/stimcirq/__init__.py b/glue/cirq/stimcirq/__init__.py index 1e5543696..daef84e16 100644 --- a/glue/cirq/stimcirq/__init__.py +++ b/glue/cirq/stimcirq/__init__.py @@ -1,6 +1,7 @@ __version__ = '1.13.dev0' from ._cirq_to_stim import cirq_circuit_to_stim_circuit from ._cx_swap_gate import CXSwapGate +from ._cz_swap_gate import CZSwapGate from ._det_annotation import DetAnnotation from ._obs_annotation import CumulativeObservableAnnotation from ._shift_coords_annotation import ShiftCoordsAnnotation @@ -20,5 +21,6 @@ "SweepPauli": SweepPauli, "TwoQubitAsymmetricDepolarizingChannel": TwoQubitAsymmetricDepolarizingChannel, "CXSwapGate": CXSwapGate, + "CZSwapGate": CZSwapGate, } JSON_RESOLVER = JSON_RESOLVERS_DICT.get diff --git a/glue/cirq/stimcirq/_cz_swap_gate.py b/glue/cirq/stimcirq/_cz_swap_gate.py new file mode 100644 index 000000000..6f9ded68c --- /dev/null +++ b/glue/cirq/stimcirq/_cz_swap_gate.py @@ -0,0 +1,46 @@ +from typing import Any, Dict, List + +import cirq +import stim + + +@cirq.value_equality +class CZSwapGate(cirq.Gate): + """Handles explaining stim's CZSWAP gates to cirq.""" + + def _num_qubits_(self) -> int: + return 2 + + def _circuit_diagram_info_(self, args: cirq.CircuitDiagramInfoArgs) -> List[str]: + return ['ZSWAP', 'ZSWAP'] + + def _value_equality_values_(self): + return () + + def _decompose_(self, qubits): + a, b = qubits + yield cirq.SWAP(a, b) + yield cirq.CZ(a, b) + + def _stim_conversion_(self, edit_circuit: stim.Circuit, targets: List[int], **kwargs): + edit_circuit.append_operation('CZSWAP', targets) + + def __pow__(self, power: int) -> 'CZSwapGate': + if power == +1: + return self + if power == -1: + return self + return NotImplemented + + def __str__(self) -> str: + return 'CZSWAP' + + def __repr__(self): + return f'stimcirq.CZSwapGate()' + + @staticmethod + def _json_namespace_() -> str: + return '' + + def _json_dict_(self) -> Dict[str, Any]: + return {} diff --git a/glue/cirq/stimcirq/_cz_swap_test.py b/glue/cirq/stimcirq/_cz_swap_test.py new file mode 100644 index 000000000..34c2af445 --- /dev/null +++ b/glue/cirq/stimcirq/_cz_swap_test.py @@ -0,0 +1,72 @@ +import cirq +import stim +import stimcirq + + +def test_stim_conversion(): + a, b, c = cirq.LineQubit.range(3) + + cirq_circuit = cirq.Circuit( + stimcirq.CZSwapGate().on(a, b), + stimcirq.CZSwapGate().on(b, c), + ) + stim_circuit = stim.Circuit( + """ + CZSWAP 0 1 + TICK + CZSWAP 1 2 + TICK + """ + ) + assert stimcirq.cirq_circuit_to_stim_circuit(cirq_circuit) == stim_circuit + assert stimcirq.stim_circuit_to_cirq_circuit(stim_circuit) == cirq_circuit + + +def test_diagram(): + a, b = cirq.LineQubit.range(2) + cirq.testing.assert_has_diagram( + cirq.Circuit( + stimcirq.CZSwapGate()(a, b), + stimcirq.CZSwapGate()(a, b), + ), + """ +0: ---ZSWAP---ZSWAP--- + | | +1: ---ZSWAP---ZSWAP--- + """, + use_unicode_characters=False, + ) + + +def test_inverse(): + a = stimcirq.CZSwapGate() + assert a**+1 == a + assert a**-1 == a + + +def test_repr(): + val = stimcirq.CZSwapGate() + assert eval(repr(val), {"stimcirq": stimcirq}) == val + + +def test_equality(): + eq = cirq.testing.EqualsTester() + eq.add_equality_group(stimcirq.CZSwapGate(), stimcirq.CZSwapGate()) + + +def test_json_serialization(): + a, b, d = cirq.LineQubit.range(3) + c = cirq.Circuit( + stimcirq.CZSwapGate()(a, b), + stimcirq.CZSwapGate()(b, d), + ) + json = cirq.to_json(c) + c2 = cirq.read_json(json_text=json, resolvers=[*cirq.DEFAULT_RESOLVERS, stimcirq.JSON_RESOLVER]) + assert c == c2 + + +def test_json_backwards_compat_exact(): + raw = stimcirq.CZSwapGate() + packed = '{\n "cirq_type": "CZSwapGate"\n}' + assert cirq.to_json(raw) == packed + assert cirq.read_json(json_text=packed, resolvers=[*cirq.DEFAULT_RESOLVERS, stimcirq.JSON_RESOLVER]) == raw diff --git a/glue/cirq/stimcirq/_stim_to_cirq.py b/glue/cirq/stimcirq/_stim_to_cirq.py index 43bf516d0..af8edeb8f 100644 --- a/glue/cirq/stimcirq/_stim_to_cirq.py +++ b/glue/cirq/stimcirq/_stim_to_cirq.py @@ -16,6 +16,7 @@ import stim from ._cx_swap_gate import CXSwapGate +from ._cz_swap_gate import CZSwapGate from ._det_annotation import DetAnnotation from ._measure_and_or_reset_gate import MeasureAndOrResetGate from ._obs_annotation import CumulativeObservableAnnotation @@ -424,6 +425,7 @@ def handler( measure=False, reset=True, basis='X', invert_measure=False, key='' ) ), + "CZSWAP": gate(CZSwapGate()), "CXSWAP": gate(CXSwapGate(inverted=False)), "SWAPCX": gate(CXSwapGate(inverted=True)), "RY": gate( diff --git a/glue/javascript/tableau.js.cc b/glue/javascript/tableau.js.cc index e31a2cbb6..920d7079c 100644 --- a/glue/javascript/tableau.js.cc +++ b/glue/javascript/tableau.js.cc @@ -3,7 +3,7 @@ #include #include "common.js.h" -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim.h b/src/stim.h index bddc097cf..c2777add7 100644 --- a/src/stim.h +++ b/src/stim.h @@ -7,8 +7,6 @@ #include "stim/circuit/circuit.h" #include "stim/circuit/circuit_instruction.h" #include "stim/circuit/export_qasm.h" -#include "stim/circuit/gate_data.h" -#include "stim/circuit/gate_data_table.h" #include "stim/circuit/gate_decomposition.h" #include "stim/circuit/gate_target.h" #include "stim/circuit/stabilizer_flow.h" @@ -45,6 +43,7 @@ #include "stim/diagram/timeline/timeline_3d_drawer.h" #include "stim/diagram/timeline/timeline_ascii_drawer.h" #include "stim/diagram/timeline/timeline_svg_drawer.h" +#include "stim/gates/gates.h" #include "stim/gen/circuit_gen_params.h" #include "stim/gen/gen_color_code.h" #include "stim/gen/gen_rep_code.h" diff --git a/src/stim/circuit/circuit.cc b/src/stim/circuit/circuit.cc index f9cf17763..fd8348ac4 100644 --- a/src/stim/circuit/circuit.cc +++ b/src/stim/circuit/circuit.cc @@ -14,11 +14,12 @@ #include "stim/circuit/circuit.h" +#include #include #include -#include "stim/circuit/gate_data.h" #include "stim/circuit/gate_target.h" +#include "stim/gates/gates.h" #include "stim/str_util.h" using namespace stim; diff --git a/src/stim/circuit/circuit.h b/src/stim/circuit/circuit.h index a69202b9f..bb9fd3c3c 100644 --- a/src/stim/circuit/circuit.h +++ b/src/stim/circuit/circuit.h @@ -28,8 +28,8 @@ #include #include "stim/circuit/circuit_instruction.h" -#include "stim/circuit/gate_data.h" #include "stim/circuit/gate_target.h" +#include "stim/gates/gates.h" #include "stim/mem/monotonic_buffer.h" #include "stim/mem/span_ref.h" diff --git a/src/stim/circuit/circuit.test.cc b/src/stim/circuit/circuit.test.cc index 6615ae520..0103bf64b 100644 --- a/src/stim/circuit/circuit.test.cc +++ b/src/stim/circuit/circuit.test.cc @@ -1619,6 +1619,7 @@ Circuit stim::generate_test_circuit_with_all_operations() { ISWAP_DAG 4 5 SWAP 6 7 SWAPCX 8 9 + CZSWAP 10 11 SQRT_XX 0 1 SQRT_XX_DAG 2 3 SQRT_YY 4 5 diff --git a/src/stim/circuit/circuit_instruction.cc b/src/stim/circuit/circuit_instruction.cc index 8178008ee..d7c327ae7 100644 --- a/src/stim/circuit/circuit_instruction.cc +++ b/src/stim/circuit/circuit_instruction.cc @@ -17,8 +17,8 @@ #include #include "stim/circuit/circuit.h" -#include "stim/circuit/gate_data.h" #include "stim/circuit/gate_target.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/circuit_instruction.pybind.cc b/src/stim/circuit/circuit_instruction.pybind.cc index a554d174d..c29700ed6 100644 --- a/src/stim/circuit/circuit_instruction.pybind.cc +++ b/src/stim/circuit/circuit_instruction.pybind.cc @@ -14,8 +14,8 @@ #include "stim/circuit/circuit_instruction.pybind.h" -#include "stim/circuit/gate_data.h" #include "stim/circuit/gate_target.pybind.h" +#include "stim/gates/gates.h" #include "stim/py/base.pybind.h" #include "stim/str_util.h" diff --git a/src/stim/circuit/circuit_instruction.pybind.h b/src/stim/circuit/circuit_instruction.pybind.h index 59a7e4ea2..48873543b 100644 --- a/src/stim/circuit/circuit_instruction.pybind.h +++ b/src/stim/circuit/circuit_instruction.pybind.h @@ -18,8 +18,8 @@ #include #include "stim/circuit/circuit_instruction.h" -#include "stim/circuit/gate_data.h" #include "stim/circuit/gate_target.h" +#include "stim/gates/gates.h" namespace stim_pybind { diff --git a/src/stim/circuit/export_qasm.cc b/src/stim/circuit/export_qasm.cc index a5a9ec8da..d18a2a19d 100644 --- a/src/stim/circuit/export_qasm.cc +++ b/src/stim/circuit/export_qasm.cc @@ -259,7 +259,9 @@ struct QasmExporter { target = t1; break; default: - throw std::invalid_argument("Not implemented: " + instruction.str()); + throw std::invalid_argument( + "Not implemented in output_two_qubit_unitary_instruction_with_possible_feedback: " + + instruction.str()); } out << "if ("; @@ -279,7 +281,9 @@ struct QasmExporter { } out << "sweep[" << control.value() << "]"; } else { - throw std::invalid_argument("Not implemented: " + instruction.str()); + throw std::invalid_argument( + "Not implemented in output_two_qubit_unitary_instruction_with_possible_feedback: " + + instruction.str()); } out << ") {\n"; out << " " << basis << " q[" << target.qubit_value() << "];\n"; @@ -415,6 +419,7 @@ struct QasmExporter { define_custom_single_qubit_gate(GateType::SQRT_Y_DAG, "sydg"); define_custom_decomposed_gate(GateType::CXSWAP, "cxswap"); + define_custom_decomposed_gate(GateType::CZSWAP, "czswap"); define_custom_decomposed_gate(GateType::ISWAP, "iswap"); define_custom_decomposed_gate(GateType::ISWAP_DAG, "iswapdg"); define_custom_decomposed_gate(GateType::SQRT_XX, "sxx"); @@ -557,7 +562,7 @@ struct QasmExporter { } } - throw std::invalid_argument("Not implemented: " + instruction.str()); + throw std::invalid_argument("Not implemented in QasmExporter::output_instruction: " + instruction.str()); } }; diff --git a/src/stim/circuit/export_qasm.test.cc b/src/stim/circuit/export_qasm.test.cc index d8f9d86f4..d0f67e86e 100644 --- a/src/stim/circuit/export_qasm.test.cc +++ b/src/stim/circuit/export_qasm.test.cc @@ -261,6 +261,7 @@ gate hyz q0 { U(pi/2, pi/2, pi/2) q0; } gate sy q0 { U(pi/2, 0, 0) q0; } gate sydg q0 { U(pi/2, pi/2, pi/2) q0; } gate cxswap q0, q1 { cx q1, q0; cx q0, q1; } +gate czswap q0, q1 { h q0; cx q0, q1; cx q1, q0; h q1; } gate iswap q0, q1 { h q0; cx q0, q1; cx q1, q0; h q1; s q1; s q0; } gate iswapdg q0, q1 { s q0; s q0; s q0; s q1; s q1; s q1; h q1; cx q1, q0; cx q0, q1; h q0; } gate sxx q0, q1 { h q0; cx q0, q1; h q1; s q0; s q1; h q0; h q1; } @@ -317,6 +318,7 @@ iswap q[2], q[3]; iswapdg q[4], q[5]; swap q[6], q[7]; swapcx q[8], q[9]; +czswap q[10], q[11]; sxx q[0], q[1]; sxxdg q[2], q[3]; syy q[4], q[5]; @@ -419,6 +421,7 @@ gate hyz q0 { U(pi/2, pi/2, pi/2) q0; } gate sy q0 { U(pi/2, 0, 0) q0; } gate sydg q0 { U(pi/2, pi/2, pi/2) q0; } gate cxswap q0, q1 { cx q1, q0; cx q0, q1; } +gate czswap q0, q1 { h q0; cx q0, q1; cx q1, q0; h q1; } gate iswap q0, q1 { h q0; cx q0, q1; cx q1, q0; h q1; s q1; s q0; } gate iswapdg q0, q1 { s q0; s q0; s q0; s q1; s q1; s q1; h q1; cx q1, q0; cx q0, q1; h q0; } gate sxx q0, q1 { h q0; cx q0, q1; h q1; s q0; s q1; h q0; h q1; } @@ -462,6 +465,7 @@ iswap q[2], q[3]; iswapdg q[4], q[5]; swap q[6], q[7]; swapcx q[8], q[9]; +czswap q[10], q[11]; sxx q[0], q[1]; sxxdg q[2], q[3]; syy q[4], q[5]; diff --git a/src/stim/circuit/gate_data_table.h b/src/stim/circuit/gate_data_table.h deleted file mode 100644 index e1d9a5a78..000000000 --- a/src/stim/circuit/gate_data_table.h +++ /dev/null @@ -1,51 +0,0 @@ -#ifndef _STIM_CIRCUIT_GATE_DATA_TABLE_H -#define _STIM_CIRCUIT_GATE_DATA_TABLE_H - -#include - -#include "gate_data.h" - -namespace stim { - -/// A class that maps each gate to a value. -/// -/// The only real difference between this class and a raw array is that this class makes it -/// harder to forget to define a value for a gate, and supports key-value style -/// initialization at compile time. -/// -/// The value for gate G is stored at: vtable.data[G.id] -template -struct GateVTable { - std::array data; - - // Must construct explicitly, but can copy and move. - GateVTable() = delete; - GateVTable(const GateVTable &) = default; - GateVTable(GateVTable &&) = default; - GateVTable &operator=(const GateVTable &) = default; - GateVTable &operator=(GateVTable &&) = default; - - // Construct from gate-value pairs. - constexpr GateVTable(const std::array, NUM_DEFINED_GATES> &gate_data_pairs) - : data({}) { - for (const auto &[gate_id, value] : gate_data_pairs) { - data[gate_id] = value; - } -#ifndef NDEBUG - std::array seen{}; - for (const auto &[gate_id, value] : gate_data_pairs) { - seen[(size_t)gate_id] = true; - } - for (const auto &gate : GATE_DATA.items) { - if (!seen[(size_t)gate.id]) { - throw std::invalid_argument( - "Missing gate data! A value was not defined for '" + std::string(gate.name) + "'."); - } - } -#endif - } -}; - -} // namespace stim - -#endif diff --git a/src/stim/circuit/gate_decomposition.h b/src/stim/circuit/gate_decomposition.h index 5cf89ad20..5cbcd505c 100644 --- a/src/stim/circuit/gate_decomposition.h +++ b/src/stim/circuit/gate_decomposition.h @@ -17,9 +17,11 @@ #ifndef _STIM_GATE_DECOMPOSITION_H #define _STIM_GATE_DECOMPOSITION_H +#include + #include "stim/circuit/circuit_instruction.h" -#include "stim/circuit/gate_data.h" #include "stim/circuit/gate_target.h" +#include "stim/gates/gates.h" #include "stim/mem/simd_bits.h" namespace stim { diff --git a/src/stim/circuit/gate_target.h b/src/stim/circuit/gate_target.h index c42b22766..fa964a386 100644 --- a/src/stim/circuit/gate_target.h +++ b/src/stim/circuit/gate_target.h @@ -19,7 +19,7 @@ #include -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" #include "stim/mem/span_ref.h" namespace stim { diff --git a/src/stim/cmd/command_help.cc b/src/stim/cmd/command_help.cc index b05fbfbaa..909bb6965 100644 --- a/src/stim/cmd/command_help.cc +++ b/src/stim/cmd/command_help.cc @@ -31,9 +31,9 @@ #include "command_sample.h" #include "command_sample_dem.h" #include "stim/arg_parse.h" -#include "stim/circuit/gate_data.h" #include "stim/circuit/stabilizer_flow.h" #include "stim/cmd/command_analyze_errors.h" +#include "stim/gates/gates.h" #include "stim/io/stim_data_formats.h" #include "stim/stabilizers/tableau.h" diff --git a/src/stim/diagram/circuit_timeline_helper.h b/src/stim/diagram/circuit_timeline_helper.h index ae912057d..d23ef4e8c 100644 --- a/src/stim/diagram/circuit_timeline_helper.h +++ b/src/stim/diagram/circuit_timeline_helper.h @@ -17,6 +17,7 @@ #ifndef _STIM_DIAGRAM_TIMELINE_ASCII_DIAGRAM_CIRCUIT_WITH_RESOLVED_TIMELINE_INFO_H #define _STIM_DIAGRAM_TIMELINE_ASCII_DIAGRAM_CIRCUIT_WITH_RESOLVED_TIMELINE_INFO_H +#include #include #include "lattice_map.h" diff --git a/src/stim/diagram/detector_slice/detector_slice_set.h b/src/stim/diagram/detector_slice/detector_slice_set.h index 24101644b..41f3a85f0 100644 --- a/src/stim/diagram/detector_slice/detector_slice_set.h +++ b/src/stim/diagram/detector_slice/detector_slice_set.h @@ -17,6 +17,7 @@ #ifndef _STIM_DIAGRAM_DETECTOR_SLICE_DETECTOR_SLICE_SET_H #define _STIM_DIAGRAM_DETECTOR_SLICE_DETECTOR_SLICE_SET_H +#include #include #include "stim/circuit/circuit.h" diff --git a/src/stim/diagram/diagram_util.cc b/src/stim/diagram/diagram_util.cc index 8fee5212d..56d34851f 100644 --- a/src/stim/diagram/diagram_util.cc +++ b/src/stim/diagram/diagram_util.cc @@ -25,6 +25,8 @@ std::pair stim_draw_internal::two_qubit_gate_pieces(Ga return {"Y", "Z"}; } else if (gate_type == GateType::CXSWAP) { return {"ZSWAP", "XSWAP"}; + } else if (gate_type == GateType::CZSWAP) { + return {"ZSWAP", "ZSWAP"}; } else if (gate_type == GateType::SWAPCX) { return {"XSWAP", "ZSWAP"}; } else { diff --git a/src/stim/diagram/gltf.h b/src/stim/diagram/gltf.h index 1644b3bab..ccef9fc03 100644 --- a/src/stim/diagram/gltf.h +++ b/src/stim/diagram/gltf.h @@ -1,6 +1,7 @@ #ifndef _STIM_DRAW_3D_GLTF_H #define _STIM_DRAW_3D_GLTF_H +#include #include #include "base64.h" diff --git a/src/stim/diagram/timeline/timeline_ascii_drawer.test.cc b/src/stim/diagram/timeline/timeline_ascii_drawer.test.cc index 54c5b56bc..63dbd63fc 100644 --- a/src/stim/diagram/timeline/timeline_ascii_drawer.test.cc +++ b/src/stim/diagram/timeline/timeline_ascii_drawer.test.cc @@ -535,9 +535,9 @@ TEST(circuit_diagram_timeline_text, test_circuit_all_ops) { | | | | | q9: ----------------------------------------ZSWAP-----SQRT_ZZ-----Y-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------R----------------------|------------|---------------------------------------------------------------------------------------------------------------------------------------------------------- | | -q10: --------------------------------------------------SQRT_ZZ_DAG-Y------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------|---------------------------------------------------------------------------------------------------------------------------------------------------------- - | | | | -q11: --------------------------------------------------SQRT_ZZ_DAG-@------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------|---------------------------------------------------------------------------------------------------------------------------------------------------------- +q10: ----------------------------------------ZSWAP-----SQRT_ZZ_DAG-Y------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------|---------------------------------------------------------------------------------------------------------------------------------------------------------- + | | | | | +q11: ----------------------------------------ZSWAP-----SQRT_ZZ_DAG-@------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------|---------------------------------------------------------------------------------------------------------------------------------------------------------- | | q12: --------------------------------------------------------------@------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------|---------------------------------------------------------------------------------------------------------------------------------------------------------- | | | diff --git a/src/stim/circuit/gate_data_annotations.cc b/src/stim/gates/gate_data_annotations.cc similarity index 99% rename from src/stim/circuit/gate_data_annotations.cc rename to src/stim/gates/gate_data_annotations.cc index 0a3e5a485..ec533521e 100644 --- a/src/stim/circuit/gate_data_annotations.cc +++ b/src/stim/gates/gate_data_annotations.cc @@ -12,7 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_blocks.cc b/src/stim/gates/gate_data_blocks.cc similarity index 98% rename from src/stim/circuit/gate_data_blocks.cc rename to src/stim/gates/gate_data_blocks.cc index 7d47f564d..d48062c8f 100644 --- a/src/stim/circuit/gate_data_blocks.cc +++ b/src/stim/gates/gate_data_blocks.cc @@ -12,7 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_collapsing.cc b/src/stim/gates/gate_data_collapsing.cc similarity index 99% rename from src/stim/circuit/gate_data_collapsing.cc rename to src/stim/gates/gate_data_collapsing.cc index 2dc2172da..8b5e04f2c 100644 --- a/src/stim/circuit/gate_data_collapsing.cc +++ b/src/stim/gates/gate_data_collapsing.cc @@ -12,8 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "stim/circuit/circuit.h" -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_controlled.cc b/src/stim/gates/gate_data_controlled.cc similarity index 99% rename from src/stim/circuit/gate_data_controlled.cc rename to src/stim/gates/gate_data_controlled.cc index 4f5be99c3..49ba28f83 100644 --- a/src/stim/circuit/gate_data_controlled.cc +++ b/src/stim/gates/gate_data_controlled.cc @@ -12,9 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include - -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_hada.cc b/src/stim/gates/gate_data_hada.cc similarity index 98% rename from src/stim/circuit/gate_data_hada.cc rename to src/stim/gates/gate_data_hada.cc index ac00b654f..419cc9b4d 100644 --- a/src/stim/circuit/gate_data_hada.cc +++ b/src/stim/gates/gate_data_hada.cc @@ -12,9 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include - -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_heralded.cc b/src/stim/gates/gate_data_heralded.cc similarity index 99% rename from src/stim/circuit/gate_data_heralded.cc rename to src/stim/gates/gate_data_heralded.cc index a398d5dd2..45515bb58 100644 --- a/src/stim/circuit/gate_data_heralded.cc +++ b/src/stim/gates/gate_data_heralded.cc @@ -12,7 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_noisy.cc b/src/stim/gates/gate_data_noisy.cc similarity index 99% rename from src/stim/circuit/gate_data_noisy.cc rename to src/stim/gates/gate_data_noisy.cc index 60daf2dc1..e58b370f4 100644 --- a/src/stim/circuit/gate_data_noisy.cc +++ b/src/stim/gates/gate_data_noisy.cc @@ -12,7 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_pair_measure.cc b/src/stim/gates/gate_data_pair_measure.cc similarity index 99% rename from src/stim/circuit/gate_data_pair_measure.cc rename to src/stim/gates/gate_data_pair_measure.cc index 7ede6538c..55472c2a7 100644 --- a/src/stim/circuit/gate_data_pair_measure.cc +++ b/src/stim/gates/gate_data_pair_measure.cc @@ -12,8 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "stim/circuit/circuit.h" -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_pauli.cc b/src/stim/gates/gate_data_pauli.cc similarity index 98% rename from src/stim/circuit/gate_data_pauli.cc rename to src/stim/gates/gate_data_pauli.cc index 73822d04c..a00bdb739 100644 --- a/src/stim/circuit/gate_data_pauli.cc +++ b/src/stim/gates/gate_data_pauli.cc @@ -12,9 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include - -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_period_3.cc b/src/stim/gates/gate_data_period_3.cc similarity index 97% rename from src/stim/circuit/gate_data_period_3.cc rename to src/stim/gates/gate_data_period_3.cc index d212737dc..fc77b4668 100644 --- a/src/stim/circuit/gate_data_period_3.cc +++ b/src/stim/gates/gate_data_period_3.cc @@ -12,9 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include - -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_period_4.cc b/src/stim/gates/gate_data_period_4.cc similarity index 98% rename from src/stim/circuit/gate_data_period_4.cc rename to src/stim/gates/gate_data_period_4.cc index 6f1299362..585246688 100644 --- a/src/stim/circuit/gate_data_period_4.cc +++ b/src/stim/gates/gate_data_period_4.cc @@ -12,9 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include - -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_pp.cc b/src/stim/gates/gate_data_pp.cc similarity index 99% rename from src/stim/circuit/gate_data_pp.cc rename to src/stim/gates/gate_data_pp.cc index f3a60c25c..cab1c9935 100644 --- a/src/stim/circuit/gate_data_pp.cc +++ b/src/stim/gates/gate_data_pp.cc @@ -12,9 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include - -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data_swaps.cc b/src/stim/gates/gate_data_swaps.cc similarity index 84% rename from src/stim/circuit/gate_data_swaps.cc rename to src/stim/gates/gate_data_swaps.cc index 26f03fe8f..5a47a811a 100644 --- a/src/stim/circuit/gate_data_swaps.cc +++ b/src/stim/gates/gate_data_swaps.cc @@ -12,9 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include - -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" using namespace stim; @@ -177,4 +175,36 @@ CNOT 0 1 CNOT 1 0 )CIRCUIT", }); + + add_gate( + failed, + Gate{ + .name = "CZSWAP", + .id = GateType::CZSWAP, + .best_candidate_inverse_id = GateType::CZSWAP, + .arg_count = 0, + .flags = (GateFlags)(GATE_IS_UNITARY | GATE_TARGETS_PAIRS), + .category = "C_Two Qubit Clifford Gates", + .help = R"MARKDOWN( +A combination CZ-and-SWAP gate. +This gate is kak-equivalent to the iswap gate. + +Parens Arguments: + + This instruction takes no parens arguments. + +Targets: + + Qubit pairs to operate on. +)MARKDOWN", + .unitary_data = {{1, 0, 0, 0}, {0, 0, 1, 0}, {0, 1, 0, 0}, {0, 0, 0, -1}}, + .flow_data = {"+ZX", "+IZ", "+XZ", "+ZI"}, + .h_s_cx_m_r_decomposition = R"CIRCUIT( +H 0 +CX 0 1 +CX 1 0 +H 1 +)CIRCUIT", + }); + add_gate_alias(failed, "SWAPCZ", "CZSWAP"); } diff --git a/src/stim/circuit/gate_data.cc b/src/stim/gates/gates.cc similarity index 99% rename from src/stim/circuit/gate_data.cc rename to src/stim/gates/gates.cc index d10fbcd52..089b753a8 100644 --- a/src/stim/circuit/gate_data.cc +++ b/src/stim/gates/gates.cc @@ -12,9 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "stim/circuit/gate_data.h" - -#include +#include "stim/gates/gates.h" using namespace stim; diff --git a/src/stim/circuit/gate_data.h b/src/stim/gates/gates.h similarity index 98% rename from src/stim/circuit/gate_data.h rename to src/stim/gates/gates.h index 721ddb32c..f6f9c577f 100644 --- a/src/stim/circuit/gate_data.h +++ b/src/stim/gates/gates.h @@ -14,20 +14,14 @@ * limitations under the License. */ -#ifndef _STIM_CIRCUIT_GATE_DATA_H -#define _STIM_CIRCUIT_GATE_DATA_H +#ifndef _STIM_GATES_GATE_DATA_H +#define _STIM_GATES_GATE_DATA_H -#include #include #include #include #include -#include -#include #include -#include -#include -#include #include #include "stim/mem/fixed_cap_vector.h" @@ -82,7 +76,7 @@ constexpr inline uint16_t gate_name_to_hash(const char *c) { return gate_name_to_hash(c, std::char_traits::length(c)); } -constexpr const size_t NUM_DEFINED_GATES = 67; +constexpr const size_t NUM_DEFINED_GATES = 68; enum class GateType : uint8_t { NOT_A_GATE = 0, @@ -160,6 +154,7 @@ enum class GateType : uint8_t { ISWAP, CXSWAP, SWAPCX, + CZSWAP, ISWAP_DAG, // Pair measurement gates MXX, diff --git a/src/stim/circuit/gate_data.perf.cc b/src/stim/gates/gates.perf.cc similarity index 96% rename from src/stim/circuit/gate_data.perf.cc rename to src/stim/gates/gates.perf.cc index a51e03566..27e411ec4 100644 --- a/src/stim/circuit/gate_data.perf.cc +++ b/src/stim/gates/gates.perf.cc @@ -12,7 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" #include diff --git a/src/stim/circuit/gate_data.pybind.cc b/src/stim/gates/gates.pybind.cc similarity index 99% rename from src/stim/circuit/gate_data.pybind.cc rename to src/stim/gates/gates.pybind.cc index 72e742202..a70b60de7 100644 --- a/src/stim/circuit/gate_data.pybind.cc +++ b/src/stim/gates/gates.pybind.cc @@ -12,13 +12,11 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "stim/circuit/gate_data.pybind.h" +#include "stim/gates/gates.h" -#include "stim/circuit/gate_data.h" #include "stim/circuit/stabilizer_flow.h" +#include "stim/gates/gates.pybind.h" #include "stim/py/base.pybind.h" -#include "stim/stabilizers/pauli_string.pybind.h" -#include "stim/stabilizers/tableau.pybind.h" #include "stim/str_util.h" using namespace stim; diff --git a/src/stim/circuit/gate_data.pybind.h b/src/stim/gates/gates.pybind.h similarity index 87% rename from src/stim/circuit/gate_data.pybind.h rename to src/stim/gates/gates.pybind.h index 25fa37b96..021668bab 100644 --- a/src/stim/circuit/gate_data.pybind.h +++ b/src/stim/gates/gates.pybind.h @@ -12,12 +12,12 @@ // See the License for the specific language governing permissions and // limitations under the License. -#ifndef _STIM_CIRCUIT_GATE_DATA_PYBIND_H -#define _STIM_CIRCUIT_GATE_DATA_PYBIND_H +#ifndef _STIM_GATES_GATE_DATA_PYBIND_H +#define _STIM_GATES_GATE_DATA_PYBIND_H #include -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" namespace stim_pybind { diff --git a/src/stim/circuit/gate_data.test.cc b/src/stim/gates/gates.test.cc similarity index 99% rename from src/stim/circuit/gate_data.test.cc rename to src/stim/gates/gates.test.cc index 93afe26b0..cf5ed6e8a 100644 --- a/src/stim/circuit/gate_data.test.cc +++ b/src/stim/gates/gates.test.cc @@ -12,7 +12,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" #include "gtest/gtest.h" diff --git a/src/stim/circuit/gate_data_test.py b/src/stim/gates/gates_test.py similarity index 100% rename from src/stim/circuit/gate_data_test.py rename to src/stim/gates/gates_test.py diff --git a/src/stim/gen/gen_surface_code.cc b/src/stim/gen/gen_surface_code.cc index 4ff30577a..91e27fa2d 100644 --- a/src/stim/gen/gen_surface_code.cc +++ b/src/stim/gen/gen_surface_code.cc @@ -2,6 +2,7 @@ #include #include +#include #include #include #include diff --git a/src/stim/py/stim.pybind.cc b/src/stim/py/stim.pybind.cc index 159d172e2..0c34f7ff1 100644 --- a/src/stim/py/stim.pybind.cc +++ b/src/stim/py/stim.pybind.cc @@ -18,13 +18,13 @@ #include "stim/circuit/circuit.pybind.h" #include "stim/circuit/circuit_instruction.pybind.h" #include "stim/circuit/circuit_repeat_block.pybind.h" -#include "stim/circuit/gate_data.pybind.h" #include "stim/circuit/gate_target.pybind.h" #include "stim/cmd/command_diagram.pybind.h" #include "stim/dem/detector_error_model.pybind.h" #include "stim/dem/detector_error_model_instruction.pybind.h" #include "stim/dem/detector_error_model_repeat_block.pybind.h" #include "stim/dem/detector_error_model_target.pybind.h" +#include "stim/gates/gates.pybind.h" #include "stim/io/read_write.pybind.h" #include "stim/main_namespaced.h" #include "stim/py/base.pybind.h" diff --git a/src/stim/py/stim_pybind_test.py b/src/stim/py/stim_pybind_test.py index ab9395791..34db65338 100644 --- a/src/stim/py/stim_pybind_test.py +++ b/src/stim/py/stim_pybind_test.py @@ -59,7 +59,7 @@ def test_targets(): def test_gate_data(): data = stim.gate_data() - assert len(data) == 66 + assert len(data) == 67 assert data["CX"].name == "CX" assert data["CX"].aliases == ["CNOT", "CX", "ZCX"] assert data["X"].is_unitary diff --git a/src/stim/simulators/error_analyzer.cc b/src/stim/simulators/error_analyzer.cc index fb89c458b..9f8e5245c 100644 --- a/src/stim/simulators/error_analyzer.cc +++ b/src/stim/simulators/error_analyzer.cc @@ -221,6 +221,9 @@ void ErrorAnalyzer::undo_gate(const CircuitInstruction &inst) { case GateType::CXSWAP: undo_CXSWAP(inst); break; + case GateType::CZSWAP: + undo_CZSWAP(inst); + break; case GateType::SWAPCX: undo_SWAPCX(inst); break; @@ -609,6 +612,9 @@ void ErrorAnalyzer::undo_ISWAP(const CircuitInstruction &dat) { void ErrorAnalyzer::undo_CXSWAP(const CircuitInstruction &dat) { tracker.undo_CXSWAP(dat); } +void ErrorAnalyzer::undo_CZSWAP(const CircuitInstruction &dat) { + tracker.undo_CZSWAP(dat); +} void ErrorAnalyzer::undo_SWAPCX(const CircuitInstruction &dat) { tracker.undo_SWAPCX(dat); } @@ -1541,7 +1547,7 @@ void ErrorAnalyzer::add_error_combinations( } catch (const std::out_of_range &ex) { std::stringstream message; message - << "An error case in a composite error exceeded that max supported number of symptoms " + << "An error case in a composite error exceeded the max supported number of symptoms " "(<=15). "; message << "\nThe " << std::to_string(s) << " basis error cases (e.g. X, Z) used to form the combined "; diff --git a/src/stim/simulators/error_analyzer.h b/src/stim/simulators/error_analyzer.h index 3a6ee1240..2a08b2322 100644 --- a/src/stim/simulators/error_analyzer.h +++ b/src/stim/simulators/error_analyzer.h @@ -214,6 +214,7 @@ struct ErrorAnalyzer { void undo_PAULI_CHANNEL_2(const CircuitInstruction &inst); void undo_ISWAP(const CircuitInstruction &inst); void undo_CXSWAP(const CircuitInstruction &inst); + void undo_CZSWAP(const CircuitInstruction &inst); void undo_SWAPCX(const CircuitInstruction &inst); void undo_RX_with_context(const CircuitInstruction &inst, const char *context_op); diff --git a/src/stim/simulators/error_matcher.cc b/src/stim/simulators/error_matcher.cc index e8c860326..8d7f3157b 100644 --- a/src/stim/simulators/error_matcher.cc +++ b/src/stim/simulators/error_matcher.cc @@ -267,7 +267,8 @@ void ErrorMatcher::rev_process_instruction(const CircuitInstruction &op) { } else if (op.gate_type == GateType::M || op.gate_type == GateType::MR) { err_m(op, TARGET_PAULI_Z_BIT); } else { - throw std::invalid_argument("Not implemented: " + std::string(GATE_DATA[op.gate_type].name)); + throw std::invalid_argument( + "Not implemented in ErrorMatcher::rev_process_instruction: " + std::string(GATE_DATA[op.gate_type].name)); } } diff --git a/src/stim/simulators/frame_simulator.h b/src/stim/simulators/frame_simulator.h index aff1a18df..9ea036082 100644 --- a/src/stim/simulators/frame_simulator.h +++ b/src/stim/simulators/frame_simulator.h @@ -120,6 +120,7 @@ struct FrameSimulator { void do_SWAP(const CircuitInstruction &inst); void do_ISWAP(const CircuitInstruction &inst); void do_CXSWAP(const CircuitInstruction &inst); + void do_CZSWAP(const CircuitInstruction &inst); void do_SWAPCX(const CircuitInstruction &inst); void do_MPP(const CircuitInstruction &inst); void do_MXX(const CircuitInstruction &inst); diff --git a/src/stim/simulators/frame_simulator.inl b/src/stim/simulators/frame_simulator.inl index 152642238..b2ed4a14c 100644 --- a/src/stim/simulators/frame_simulator.inl +++ b/src/stim/simulators/frame_simulator.inl @@ -504,6 +504,17 @@ void FrameSimulator::do_CXSWAP(const CircuitInstruction &target_data) { }); } +template +void FrameSimulator::do_CZSWAP(const CircuitInstruction &target_data) { + for_each_target_pair( + *this, target_data, [](simd_word &x1, simd_word &z1, simd_word &x2, simd_word &z2) { + std::swap(z1, z2); + std::swap(x1, x2); + z1 ^= x2; + z2 ^= x1; + }); +} + template void FrameSimulator::do_SWAPCX(const CircuitInstruction &target_data) { for_each_target_pair( @@ -1006,6 +1017,9 @@ void FrameSimulator::do_gate(const CircuitInstruction &inst) { case GateType::CXSWAP: do_CXSWAP(inst); break; + case GateType::CZSWAP: + do_CZSWAP(inst); + break; case GateType::SWAPCX: do_SWAPCX(inst); break; @@ -1065,7 +1079,7 @@ void FrameSimulator::do_gate(const CircuitInstruction &inst) { break; default: - throw std::invalid_argument("Not implemented: " + inst.str()); + throw std::invalid_argument("Not implemented in FrameSimulator::do_gate: " + inst.str()); } } diff --git a/src/stim/simulators/graph_simulator.cc b/src/stim/simulators/graph_simulator.cc index 1b5d2a250..55e46fbba 100644 --- a/src/stim/simulators/graph_simulator.cc +++ b/src/stim/simulators/graph_simulator.cc @@ -12,8 +12,8 @@ GraphSimulator::GraphSimulator(size_t num_qubits) void GraphSimulator::do_1q_gate(GateType gate, size_t qubit) { GateTarget t = GateTarget::qubit(qubit); - x2outs.ref().after_inplace(CircuitInstruction{gate, {}, &t}, false); - z2outs.ref().after_inplace(CircuitInstruction{gate, {}, &t}, false); + x2outs.ref().do_instruction(CircuitInstruction{gate, {}, &t}); + z2outs.ref().do_instruction(CircuitInstruction{gate, {}, &t}); paulis.xs[qubit] ^= z2outs.sign; paulis.zs[qubit] ^= x2outs.sign; x2outs.sign = 0; diff --git a/src/stim/simulators/graph_simulator.test.cc b/src/stim/simulators/graph_simulator.test.cc index ef72b16fe..eefb74616 100644 --- a/src/stim/simulators/graph_simulator.test.cc +++ b/src/stim/simulators/graph_simulator.test.cc @@ -42,8 +42,7 @@ void expect_graph_sim_effect_matches_tableau_sim(const GraphSimulator &state, co auto s2 = tableau_sim2.canonical_stabilizers(); if (s1 != s2) { EXPECT_EQ(s1, s2) << "EFFECT:\nstim::Circuit(R\"CIRCUIT(\n" - << effect << "\n)CIRCUIT\")" - << "\n" + << effect << "\n)CIRCUIT\")" << "\n" << "STATE:\n" << state << "\n"; } diff --git a/src/stim/simulators/matched_error.pybind.cc b/src/stim/simulators/matched_error.pybind.cc index 746ad664e..a5a3db382 100644 --- a/src/stim/simulators/matched_error.pybind.cc +++ b/src/stim/simulators/matched_error.pybind.cc @@ -14,10 +14,10 @@ #include "stim/simulators/matched_error.pybind.h" -#include "stim/circuit/gate_data.h" #include "stim/circuit/gate_target.h" #include "stim/circuit/gate_target.pybind.h" #include "stim/dem/detector_error_model_target.pybind.h" +#include "stim/gates/gates.h" #include "stim/py/base.pybind.h" #include "stim/simulators/matched_error.h" diff --git a/src/stim/simulators/measurements_to_detection_events.inl b/src/stim/simulators/measurements_to_detection_events.inl index 53c7109fd..77a652ef1 100644 --- a/src/stim/simulators/measurements_to_detection_events.inl +++ b/src/stim/simulators/measurements_to_detection_events.inl @@ -14,7 +14,7 @@ #include -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" #include "stim/io/measure_record_batch_writer.h" #include "stim/io/measure_record_reader.h" #include "stim/io/stim_data_formats.h" diff --git a/src/stim/simulators/sparse_rev_frame_tracker.cc b/src/stim/simulators/sparse_rev_frame_tracker.cc index 62d1de9d6..ef5a1925b 100644 --- a/src/stim/simulators/sparse_rev_frame_tracker.cc +++ b/src/stim/simulators/sparse_rev_frame_tracker.cc @@ -95,6 +95,9 @@ void SparseUnsignedRevFrameTracker::undo_gate(const CircuitInstruction &inst) { case GateType::CXSWAP: undo_CXSWAP(inst); break; + case GateType::CZSWAP: + undo_CZSWAP(inst); + break; case GateType::SWAPCX: undo_SWAPCX(inst); break; @@ -729,6 +732,17 @@ void SparseUnsignedRevFrameTracker::undo_CXSWAP(const CircuitInstruction &dat) { } } +void SparseUnsignedRevFrameTracker::undo_CZSWAP(const CircuitInstruction &dat) { + for (size_t k = dat.targets.size() - 2; k + 2 != 0; k -= 2) { + auto a = dat.targets[k].data; + auto b = dat.targets[k + 1].data; + zs[a] ^= xs[b]; + zs[b] ^= xs[a]; + std::swap(xs[a], xs[b]); + std::swap(zs[a], zs[b]); + } +} + void SparseUnsignedRevFrameTracker::undo_SWAPCX(const CircuitInstruction &dat) { for (size_t k = dat.targets.size() - 2; k + 2 != 0; k -= 2) { auto a = dat.targets[k].data; diff --git a/src/stim/simulators/sparse_rev_frame_tracker.h b/src/stim/simulators/sparse_rev_frame_tracker.h index 0de3958a3..acebecb32 100644 --- a/src/stim/simulators/sparse_rev_frame_tracker.h +++ b/src/stim/simulators/sparse_rev_frame_tracker.h @@ -18,7 +18,6 @@ #define _STIM_SIMULATORS_SPARSE_REV_FRAME_TRACKER_H #include "stim/circuit/circuit.h" -#include "stim/circuit/gate_data_table.h" #include "stim/dem/detector_error_model.h" #include "stim/mem/sparse_xor_vec.h" #include "stim/stabilizers/pauli_string.h" @@ -114,6 +113,7 @@ struct SparseUnsignedRevFrameTracker { void undo_SWAP(const CircuitInstruction &inst); void undo_ISWAP(const CircuitInstruction &inst); void undo_CXSWAP(const CircuitInstruction &inst); + void undo_CZSWAP(const CircuitInstruction &inst); void undo_SWAPCX(const CircuitInstruction &inst); template diff --git a/src/stim/simulators/tableau_simulator.h b/src/stim/simulators/tableau_simulator.h index 562434194..b67f1d4ab 100644 --- a/src/stim/simulators/tableau_simulator.h +++ b/src/stim/simulators/tableau_simulator.h @@ -25,7 +25,6 @@ #include #include "stim/circuit/circuit.h" -#include "stim/circuit/gate_data_table.h" #include "stim/io/measure_record.h" #include "stim/stabilizers/tableau.h" #include "stim/stabilizers/tableau_transposed_raii.h" @@ -135,6 +134,7 @@ struct TableauSimulator { void do_ISWAP(const CircuitInstruction &inst); void do_ISWAP_DAG(const CircuitInstruction &inst); void do_CXSWAP(const CircuitInstruction &inst); + void do_CZSWAP(const CircuitInstruction &inst); void do_SWAPCX(const CircuitInstruction &inst); void do_XCX(const CircuitInstruction &inst); void do_XCY(const CircuitInstruction &inst); diff --git a/src/stim/simulators/tableau_simulator.inl b/src/stim/simulators/tableau_simulator.inl index bdc739495..4fa7713fc 100644 --- a/src/stim/simulators/tableau_simulator.inl +++ b/src/stim/simulators/tableau_simulator.inl @@ -14,8 +14,8 @@ #include -#include "stim/circuit/gate_data.h" #include "stim/circuit/gate_decomposition.h" +#include "stim/gates/gates.h" #include "stim/probability_util.h" #include "stim/simulators/tableau_simulator.h" #include "stim/simulators/vector_simulator.h" @@ -752,6 +752,18 @@ void TableauSimulator::do_CXSWAP(const CircuitInstruction &target_data) { } } +template +void TableauSimulator::do_CZSWAP(const CircuitInstruction &target_data) { + const auto &targets = target_data.targets; + assert(!(targets.size() & 1)); + for (size_t k = 0; k < targets.size(); k += 2) { + auto q1 = targets[k].data; + auto q2 = targets[k + 1].data; + inv_state.prepend_ZCZ(q1, q2); + inv_state.prepend_SWAP(q2, q1); + } +} + template void TableauSimulator::do_SWAPCX(const CircuitInstruction &target_data) { const auto &targets = target_data.targets; @@ -1673,6 +1685,9 @@ void TableauSimulator::do_gate(const CircuitInstruction &inst) { case GateType::CXSWAP: do_CXSWAP(inst); break; + case GateType::CZSWAP: + do_CZSWAP(inst); + break; case GateType::SWAPCX: do_SWAPCX(inst); break; diff --git a/src/stim/simulators/tableau_simulator.test.cc b/src/stim/simulators/tableau_simulator.test.cc index aa5f39872..6e4a6bcc9 100644 --- a/src/stim/simulators/tableau_simulator.test.cc +++ b/src/stim/simulators/tableau_simulator.test.cc @@ -1807,9 +1807,7 @@ TEST_EACH_WORD_SIZE_W(TableauSimulator, postselect_x, { sim.do_ZCX(OpDat({0, 1})); sim.do_Z(OpDat(0)); ASSERT_THROW( - { - sim.postselect_x(std::vector{GateTarget::qubit(0), GateTarget::qubit(1)}, true); - }, + { sim.postselect_x(std::vector{GateTarget::qubit(0), GateTarget::qubit(1)}, true); }, std::invalid_argument); ASSERT_EQ(sim.peek_bloch(0), PauliString::from_str("-X")); ASSERT_EQ(sim.peek_bloch(1), PauliString::from_str("+X")); @@ -1887,9 +1885,7 @@ TEST_EACH_WORD_SIZE_W(TableauSimulator, postselect_y, { sim.do_H_XZ(OpDat(0)); sim.do_ZCX(OpDat({0, 1})); ASSERT_THROW( - { - sim.postselect_y(std::vector{GateTarget::qubit(0), GateTarget::qubit(1)}, true); - }, + { sim.postselect_y(std::vector{GateTarget::qubit(0), GateTarget::qubit(1)}, true); }, std::invalid_argument); ASSERT_EQ(sim.peek_bloch(0), PauliString::from_str("-Y")); ASSERT_EQ(sim.peek_bloch(1), PauliString::from_str("+Y")); @@ -1967,9 +1963,7 @@ TEST_EACH_WORD_SIZE_W(TableauSimulator, postselect_z, { sim.do_ZCX(OpDat({0, 1})); sim.do_X(OpDat(0)); ASSERT_THROW( - { - sim.postselect_z(std::vector{GateTarget::qubit(0), GateTarget::qubit(1)}, true); - }, + { sim.postselect_z(std::vector{GateTarget::qubit(0), GateTarget::qubit(1)}, true); }, std::invalid_argument); ASSERT_EQ(sim.peek_bloch(0), PauliString::from_str("-Z")); ASSERT_EQ(sim.peek_bloch(1), PauliString::from_str("+Z")); diff --git a/src/stim/simulators/vector_simulator.cc b/src/stim/simulators/vector_simulator.cc index adc81c185..78ec9c184 100644 --- a/src/stim/simulators/vector_simulator.cc +++ b/src/stim/simulators/vector_simulator.cc @@ -16,7 +16,7 @@ #include -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" #include "stim/mem/simd_util.h" #include "stim/stabilizers/pauli_string.h" diff --git a/src/stim/simulators/vector_simulator.test.cc b/src/stim/simulators/vector_simulator.test.cc index 3e8d4c9f2..806d731e4 100644 --- a/src/stim/simulators/vector_simulator.test.cc +++ b/src/stim/simulators/vector_simulator.test.cc @@ -16,7 +16,7 @@ #include "gtest/gtest.h" -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" #include "stim/mem/simd_word.test.h" #include "stim/stabilizers/pauli_string.h" diff --git a/src/stim/stabilizers/conversions.test.cc b/src/stim/stabilizers/conversions.test.cc index 1825f0695..e14e3c297 100644 --- a/src/stim/stabilizers/conversions.test.cc +++ b/src/stim/stabilizers/conversions.test.cc @@ -532,10 +532,7 @@ TEST_EACH_WORD_SIZE_W(conversions, unitary_to_tableau_fuzz_vs_tableau_to_unitary TEST_EACH_WORD_SIZE_W(conversions, unitary_to_tableau_fail, { ASSERT_THROW( - { - unitary_to_tableau({{{1}, {0}}, {{0}, {sqrtf(0.5), sqrtf(0.5)}}}, false); - }, - std::invalid_argument); + { unitary_to_tableau({{{1}, {0}}, {{0}, {sqrtf(0.5), sqrtf(0.5)}}}, false); }, std::invalid_argument); ASSERT_THROW( { unitary_to_tableau( diff --git a/src/stim/stabilizers/pauli_string.test.cc b/src/stim/stabilizers/pauli_string.test.cc index 8890c55df..b58bbe151 100644 --- a/src/stim/stabilizers/pauli_string.test.cc +++ b/src/stim/stabilizers/pauli_string.test.cc @@ -522,8 +522,8 @@ TEST_EACH_WORD_SIZE_W(pauli_string, before_after_circuit_understands_commutation MPP X2*Y3*Z4 X5*X6 H 1 )CIRCUIT"); - auto before = PauliString::from_str("+_X"); - auto after = PauliString::from_str("+_Z"); + auto before = PauliString::from_str("+_X_____"); + auto after = PauliString::from_str("+_Z_____"); ASSERT_EQ(before.ref().after(c), after); ASSERT_EQ(after.ref().before(c), before); @@ -532,12 +532,12 @@ TEST_EACH_WORD_SIZE_W(pauli_string, before_after_circuit_understands_commutation ASSERT_EQ(before.ref().after(c), after); ASSERT_EQ(after.ref().before(c), before); - before = PauliString::from_str("+_XX___"); - after = PauliString::from_str("+_ZX___"); + before = PauliString::from_str("+_XX____"); + after = PauliString::from_str("+_ZX____"); ASSERT_EQ(before.ref().after(c), after); ASSERT_EQ(after.ref().before(c), before); - before = after = PauliString::from_str("+__ZX"); + before = after = PauliString::from_str("+__ZX___"); ASSERT_EQ(before.ref().after(c), after); ASSERT_EQ(after.ref().before(c), before); @@ -612,9 +612,7 @@ TEST_EACH_WORD_SIZE_W(pauli_string, after_tableau, { std::invalid_argument); ASSERT_THROW( - { - PauliString::from_str("+XZ_").ref().after(GATE_DATA.at("CX").tableau(), std::vector{0, 5}); - }, + { PauliString::from_str("+XZ_").ref().after(GATE_DATA.at("CX").tableau(), std::vector{0, 5}); }, std::invalid_argument); }) @@ -631,9 +629,7 @@ TEST_EACH_WORD_SIZE_W(pauli_string, before_tableau, { std::invalid_argument); ASSERT_THROW( - { - PauliString::from_str("+XZ_").ref().before(GATE_DATA.at("CX").tableau(), std::vector{0, 5}); - }, + { PauliString::from_str("+XZ_").ref().before(GATE_DATA.at("CX").tableau(), std::vector{0, 5}); }, std::invalid_argument); }) diff --git a/src/stim/stabilizers/pauli_string_ref.h b/src/stim/stabilizers/pauli_string_ref.h index 365f8a684..dc1d7ecfd 100644 --- a/src/stim/stabilizers/pauli_string_ref.h +++ b/src/stim/stabilizers/pauli_string_ref.h @@ -118,9 +118,19 @@ struct PauliStringRef { /// Returns a string describing the given Pauli string, indexing the Paulis so that identities can be omitted. std::string sparse_str() const; - void after_inplace_broadcast(const Tableau &tableau, SpanRef targets, bool inverse); - void after_inplace(const Circuit &Circuit); - void after_inplace(const CircuitInstruction &operation, bool inverse); + /// Applies the given tableau to the pauli string, at the given targets. + /// + /// Args: + /// tableau: The Clifford operation to apply. + /// targets: The qubits to target. Broadcasting is supported. The length of the span must be a multiple of the + /// tableau's size. + /// inverse: When true, applies the inverse of the tableau instead of the tableau. + void do_tableau(const Tableau &tableau, SpanRef targets, bool inverse); + void do_circuit(const Circuit &circuit); + void undo_circuit(const Circuit &circuit); + void do_instruction(const CircuitInstruction &inst); + void undo_instruction(const CircuitInstruction &inst); + PauliString after(const Circuit &circuit) const; PauliString after(const Tableau &tableau, SpanRef indices) const; PauliString after(const CircuitInstruction &operation) const; @@ -129,6 +139,63 @@ struct PauliStringRef { PauliString before(const CircuitInstruction &operation) const; size_t weight() const; + + private: + void check_avoids_MPP(const CircuitInstruction &inst); + void check_avoids_reset(const CircuitInstruction &inst); + void check_avoids_measurement(const CircuitInstruction &inst); + + void do_single_cx(const CircuitInstruction &inst, uint32_t c, uint32_t t); + void do_single_cy(const CircuitInstruction &inst, uint32_t c, uint32_t t); + void do_single_cz(const CircuitInstruction &inst, uint32_t c, uint32_t t); + + void do_H_XZ(const CircuitInstruction &inst); + void do_H_YZ(const CircuitInstruction &inst); + void do_H_XY(const CircuitInstruction &inst); + void do_C_XYZ(const CircuitInstruction &inst); + void do_C_ZYX(const CircuitInstruction &inst); + void do_SQRT_X(const CircuitInstruction &inst); + void do_SQRT_Y(const CircuitInstruction &inst); + void do_SQRT_Z(const CircuitInstruction &inst); + void do_SQRT_X_DAG(const CircuitInstruction &inst); + void do_SQRT_Y_DAG(const CircuitInstruction &inst); + void do_SQRT_Z_DAG(const CircuitInstruction &inst); + void do_SQRT_XX(const CircuitInstruction &inst); + void do_SQRT_XX_DAG(const CircuitInstruction &inst); + void do_SQRT_YY(const CircuitInstruction &inst); + void do_SQRT_YY_DAG(const CircuitInstruction &inst); + void do_SQRT_ZZ(const CircuitInstruction &inst); + void do_SQRT_ZZ_DAG(const CircuitInstruction &inst); + template + void do_ZCX(const CircuitInstruction &inst); + template + void do_ZCY(const CircuitInstruction &inst); + void do_ZCZ(const CircuitInstruction &inst); + template + void do_SWAP(const CircuitInstruction &inst); + void do_X(const CircuitInstruction &inst); + void do_Y(const CircuitInstruction &inst); + void do_Z(const CircuitInstruction &inst); + template + void do_ISWAP(const CircuitInstruction &inst); + template + void do_ISWAP_DAG(const CircuitInstruction &inst); + template + void do_CXSWAP(const CircuitInstruction &inst); + template + void do_CZSWAP(const CircuitInstruction &inst); + template + void do_SWAPCX(const CircuitInstruction &inst); + void do_XCX(const CircuitInstruction &inst); + template + void do_XCY(const CircuitInstruction &inst); + template + void do_XCZ(const CircuitInstruction &inst); + template + void do_YCX(const CircuitInstruction &inst); + void do_YCY(const CircuitInstruction &inst); + template + void do_YCZ(const CircuitInstruction &inst); }; /// Writes a string describing the given Pauli string to an output stream. diff --git a/src/stim/stabilizers/pauli_string_ref.inl b/src/stim/stabilizers/pauli_string_ref.inl index 522c446eb..c91794f57 100644 --- a/src/stim/stabilizers/pauli_string_ref.inl +++ b/src/stim/stabilizers/pauli_string_ref.inl @@ -140,8 +140,7 @@ bool PauliStringRef::commutes(const PauliStringRef &other) const noexcept } template -void PauliStringRef::after_inplace_broadcast( - const Tableau &tableau, SpanRef indices, bool inverse) { +void PauliStringRef::do_tableau(const Tableau &tableau, SpanRef indices, bool inverse) { if (tableau.num_qubits == 0 || indices.size() % tableau.num_qubits != 0) { throw std::invalid_argument("len(tableau) == 0 or len(indices) % len(tableau) != 0"); } @@ -164,148 +163,470 @@ void PauliStringRef::after_inplace_broadcast( } template -void PauliStringRef::after_inplace(const Circuit &circuit) { - for (const auto &op : circuit.operations) { - if (op.gate_type == GateType::REPEAT) { - const auto &body = op.repeat_block_body(circuit); - auto reps = op.repeat_block_rep_count(); - for (size_t k = 0; k < reps; k++) { - after_inplace(body); - } - } else { - after_inplace(op, false); +void PauliStringRef::undo_circuit(const Circuit &circuit) { + circuit.for_each_operation_reverse([&](const CircuitInstruction &inst) { + undo_instruction(inst); + }); +} + +template +void PauliStringRef::do_circuit(const Circuit &circuit) { + circuit.for_each_operation([&](const CircuitInstruction &inst) { + do_instruction(inst); + }); +} + +template +void PauliStringRef::check_avoids_measurement(const CircuitInstruction &inst) { + bool x_dep, z_dep; + if (inst.gate_type == GateType::M) { + x_dep = true; + z_dep = false; + } else if (inst.gate_type == GateType::MX) { + x_dep = false; + z_dep = true; + } else if (inst.gate_type == GateType::MY) { + x_dep = true; + z_dep = true; + } else { + throw std::invalid_argument("Unrecognized measurement type: " + inst.str()); + } + for (const auto &t : inst.targets) { + assert(t.is_qubit_target()); + auto q = t.qubit_value(); + if (q < num_qubits && ((xs[q] & x_dep) ^ (zs[q] & z_dep))) { + std::stringstream ss; + ss << "The pauli observable '" << *this; + ss << "' doesn't have a well specified value after '" << inst; + ss << "' because it anticommutes with the measurement."; + throw std::invalid_argument(ss.str()); } } } template -void PauliStringRef::after_inplace(const CircuitInstruction &operation, bool inverse) { - const auto &gate_data = GATE_DATA[operation.gate_type]; - if (gate_data.flags & GATE_IS_UNITARY) { - // TODO: use hand-optimized methods instead of the generic tableau method. - std::vector indices; - for (auto t : operation.targets) { - if (!t.is_qubit_target()) { - throw std::invalid_argument("Operation isn't unitary: " + operation.str()); - } - if (t.qubit_value() >= num_qubits) { - throw std::invalid_argument("Operation reaches past end of pauli string: " + operation.str()); - } - indices.push_back(t.qubit_value()); +void PauliStringRef::check_avoids_reset(const CircuitInstruction &inst) { + // Only fail if the pauli string actually touches the reset. + for (const auto &t : inst.targets) { + assert(t.is_qubit_target()); + auto q = t.qubit_value(); + if (q < num_qubits && (xs[q] || zs[q])) { + std::stringstream ss; + ss << "The pauli observable '" << *this; + ss << "' doesn't have a well specified value after '" << inst; + ss << "' because the reset discards information."; + throw std::invalid_argument(ss.str()); } - after_inplace_broadcast(gate_data.tableau(), indices, inverse); - } else if (gate_data.flags & GATE_HAS_NO_EFFECT_ON_QUBITS) { - // Gate can be ignored. - } else if (gate_data.flags & GATE_IS_RESET) { - // Only fail if the pauli string actually touches the reset. - for (const auto &t : operation.targets) { - assert(t.is_qubit_target()); + } +} + +template +void PauliStringRef::check_avoids_MPP(const CircuitInstruction &inst) { + size_t start = 0; + const auto &targets = inst.targets; + while (start < targets.size()) { + size_t end = start + 1; + bool anticommutes = false; + while (true) { + auto t = targets[end - 1]; auto q = t.qubit_value(); - if (q < num_qubits && (xs[q] || zs[q])) { - std::stringstream ss; - ss << "The pauli observable '" << *this << "' doesn't have a well specified value after '" << operation - << "' because the reset discards information."; - throw std::invalid_argument(ss.str()); + if (q < num_qubits) { + anticommutes ^= zs[q] && (t.data & TARGET_PAULI_X_BIT); + anticommutes ^= xs[q] && (t.data & TARGET_PAULI_Z_BIT); } - } - } else if (operation.gate_type == GateType::MPAD) { - // No effect. - } else if (operation.gate_type == GateType::MPP) { - size_t start = 0; - const auto &targets = operation.targets; - while (start < targets.size()) { - size_t end = start + 1; - bool anticommutes = false; - while (true) { - auto t = targets[end - 1]; - auto q = t.qubit_value(); - if (q < num_qubits) { - anticommutes ^= zs[q] && (t.data & TARGET_PAULI_X_BIT); - anticommutes ^= xs[q] && (t.data & TARGET_PAULI_Z_BIT); - } - if (end >= targets.size() || !targets[end].is_combiner()) { - break; - } - end += 2; + if (end >= targets.size() || !targets[end].is_combiner()) { + break; } - if (anticommutes) { - std::stringstream ss; - ss << "The pauli observable '" << *this << "' doesn't have a well specified value after '" << operation - << "' because it anticommutes with the measurement."; - throw std::invalid_argument(ss.str()); - } - start = end; + end += 2; } - } else if (gate_data.flags & GATE_PRODUCES_RESULTS) { - bool x_dep, z_dep; - if (operation.gate_type == GateType::M) { - x_dep = true; - z_dep = false; - } else if (operation.gate_type == GateType::MX) { - x_dep = false; - z_dep = true; - } else if (operation.gate_type == GateType::MY) { - x_dep = true; - z_dep = true; - } else { - throw std::invalid_argument("Unrecognized measurement type: " + operation.str()); + if (anticommutes) { + std::stringstream ss; + ss << "The pauli observable '" << *this; + ss << "' doesn't have a well specified value after '" << inst; + ss << "' because it anticommutes with the measurement."; + throw std::invalid_argument(ss.str()); } - for (const auto &t : operation.targets) { - assert(t.is_qubit_target()); - auto q = t.qubit_value(); - if (q < num_qubits && ((xs[q] & x_dep) ^ (zs[q] & z_dep))) { - std::stringstream ss; - ss << "The pauli observable '" << *this << "' doesn't have a well specified value after '" << operation - << "' because it anticommutes with the measurement."; - throw std::invalid_argument(ss.str()); - } + start = end; + } +} + +template +void PauliStringRef::do_instruction(const CircuitInstruction &inst) { + for (const auto &t : inst.targets) { + if (t.has_qubit_value() && t.qubit_value() >= num_qubits && + !(GATE_DATA[inst.gate_type].flags & GATE_HAS_NO_EFFECT_ON_QUBITS)) { + std::stringstream ss; + ss << "The instruction '" << inst; + ss << "' targets qubits outside the pauli string '" << *this; + ss << "'."; + throw std::invalid_argument(ss.str()); } - } else { - std::stringstream ss; - ss << "The pauli string '" << *this << "' doesn't have a well defined deterministic value after '" << operation - << "'."; - throw std::invalid_argument(ss.str()); + } + + const auto &gate_data = GATE_DATA[inst.gate_type]; + switch (gate_data.id) { + case GateType::H: + do_H_XZ(inst); + break; + case GateType::H_YZ: + do_H_YZ(inst); + break; + case GateType::H_XY: + do_H_XY(inst); + break; + case GateType::C_XYZ: + do_C_XYZ(inst); + break; + case GateType::C_ZYX: + do_C_ZYX(inst); + break; + case GateType::SQRT_X: + do_SQRT_X(inst); + break; + case GateType::SQRT_Y: + do_SQRT_Y(inst); + break; + case GateType::S: + do_SQRT_Z(inst); + break; + case GateType::SQRT_X_DAG: + do_SQRT_X_DAG(inst); + break; + case GateType::SQRT_Y_DAG: + do_SQRT_Y_DAG(inst); + break; + case GateType::S_DAG: + do_SQRT_Z_DAG(inst); + break; + case GateType::SQRT_XX: + do_SQRT_XX(inst); + break; + case GateType::SQRT_XX_DAG: + do_SQRT_XX_DAG(inst); + break; + case GateType::SQRT_YY: + do_SQRT_YY(inst); + break; + case GateType::SQRT_YY_DAG: + do_SQRT_YY_DAG(inst); + break; + case GateType::SQRT_ZZ: + do_SQRT_ZZ(inst); + break; + case GateType::SQRT_ZZ_DAG: + do_SQRT_ZZ_DAG(inst); + break; + case GateType::CX: + do_ZCX(inst); + break; + case GateType::CY: + do_ZCY(inst); + break; + case GateType::CZ: + do_ZCZ(inst); + break; + case GateType::SWAP: + do_SWAP(inst); + break; + case GateType::X: + do_X(inst); + break; + case GateType::Y: + do_Y(inst); + break; + case GateType::Z: + do_Z(inst); + break; + case GateType::ISWAP: + do_ISWAP(inst); + break; + case GateType::ISWAP_DAG: + do_ISWAP_DAG(inst); + break; + case GateType::CXSWAP: + do_CXSWAP(inst); + break; + case GateType::CZSWAP: + do_CZSWAP(inst); + break; + case GateType::SWAPCX: + do_SWAPCX(inst); + break; + case GateType::XCX: + do_XCX(inst); + break; + case GateType::XCY: + do_XCY(inst); + break; + case GateType::XCZ: + do_XCZ(inst); + break; + case GateType::YCX: + do_YCX(inst); + break; + case GateType::YCY: + do_YCY(inst); + break; + case GateType::YCZ: + do_YCZ(inst); + break; + + case GateType::DETECTOR: + case GateType::OBSERVABLE_INCLUDE: + case GateType::TICK: + case GateType::QUBIT_COORDS: + case GateType::SHIFT_COORDS: + case GateType::MPAD: + case GateType::I: + // No effect. + break; + + case GateType::R: + case GateType::RX: + case GateType::RY: + case GateType::MR: + case GateType::MRX: + case GateType::MRY: + check_avoids_reset(inst); + break; + + case GateType::M: + case GateType::MX: + case GateType::MY: + check_avoids_measurement(inst); + break; + + case GateType::MPP: + check_avoids_MPP(inst); + break; + + case GateType::X_ERROR: + case GateType::Y_ERROR: + case GateType::Z_ERROR: + case GateType::DEPOLARIZE1: + case GateType::DEPOLARIZE2: + case GateType::E: + case GateType::ELSE_CORRELATED_ERROR: { + std::stringstream ss; + ss << "The pauli string '" << *this; + ss << "' doesn't have a well defined deterministic value after '" << inst; + ss << "'."; + throw std::invalid_argument(ss.str()); + } + + default: + throw std::invalid_argument("Not implemented in PauliStringRef::do_instruction: " + inst.str()); + } +} + +template +void PauliStringRef::undo_instruction(const CircuitInstruction &inst) { + for (const auto &t : inst.targets) { + if (t.has_qubit_value() && t.qubit_value() >= num_qubits && + !(GATE_DATA[inst.gate_type].flags & GATE_HAS_NO_EFFECT_ON_QUBITS)) { + std::stringstream ss; + ss << "The instruction '" << inst; + ss << "' targets qubits outside the pauli string '" << *this; + ss << "'."; + throw std::invalid_argument(ss.str()); + } + } + + const auto &gate_data = GATE_DATA[inst.gate_type]; + switch (gate_data.id) { + case GateType::H: + do_H_XZ(inst); + break; + case GateType::H_YZ: + do_H_YZ(inst); + break; + case GateType::H_XY: + do_H_XY(inst); + break; + case GateType::C_XYZ: + do_C_ZYX(inst); + break; + case GateType::C_ZYX: + do_C_XYZ(inst); + break; + case GateType::SQRT_X: + do_SQRT_X_DAG(inst); + break; + case GateType::SQRT_Y: + do_SQRT_Y_DAG(inst); + break; + case GateType::S: + do_SQRT_Z_DAG(inst); + break; + case GateType::SQRT_X_DAG: + do_SQRT_X(inst); + break; + case GateType::SQRT_Y_DAG: + do_SQRT_Y(inst); + break; + case GateType::S_DAG: + do_SQRT_Z(inst); + break; + case GateType::SQRT_XX: + do_SQRT_XX_DAG(inst); + break; + case GateType::SQRT_XX_DAG: + do_SQRT_XX(inst); + break; + case GateType::SQRT_YY: + do_SQRT_YY_DAG(inst); + break; + case GateType::SQRT_YY_DAG: + do_SQRT_YY(inst); + break; + case GateType::SQRT_ZZ: + do_SQRT_ZZ_DAG(inst); + break; + case GateType::SQRT_ZZ_DAG: + do_SQRT_ZZ(inst); + break; + case GateType::CX: + do_ZCX(inst); + break; + case GateType::CY: + do_ZCY(inst); + break; + case GateType::CZ: + do_ZCZ(inst); + break; + case GateType::SWAP: + do_SWAP(inst); + break; + case GateType::X: + do_X(inst); + break; + case GateType::Y: + do_Y(inst); + break; + case GateType::Z: + do_Z(inst); + break; + case GateType::ISWAP: + do_ISWAP_DAG(inst); + break; + case GateType::ISWAP_DAG: + do_ISWAP(inst); + break; + case GateType::CXSWAP: + do_SWAPCX(inst); + break; + case GateType::CZSWAP: + do_CZSWAP(inst); + break; + case GateType::SWAPCX: + do_CXSWAP(inst); + break; + case GateType::XCX: + do_XCX(inst); + break; + case GateType::XCY: + do_XCY(inst); + break; + case GateType::XCZ: + do_XCZ(inst); + break; + case GateType::YCX: + do_YCX(inst); + break; + case GateType::YCY: + do_YCY(inst); + break; + case GateType::YCZ: + do_YCZ(inst); + break; + + case GateType::DETECTOR: + case GateType::OBSERVABLE_INCLUDE: + case GateType::TICK: + case GateType::QUBIT_COORDS: + case GateType::SHIFT_COORDS: + case GateType::MPAD: + case GateType::I: + // No effect. + break; + + case GateType::R: + case GateType::RX: + case GateType::RY: + case GateType::MR: + case GateType::MRX: + case GateType::MRY: + check_avoids_reset(inst); + break; + + case GateType::M: + case GateType::MX: + case GateType::MY: + check_avoids_measurement(inst); + break; + + case GateType::MPP: + check_avoids_MPP(inst); + break; + + case GateType::X_ERROR: + case GateType::Y_ERROR: + case GateType::Z_ERROR: + case GateType::DEPOLARIZE1: + case GateType::DEPOLARIZE2: + case GateType::E: + case GateType::ELSE_CORRELATED_ERROR: { + std::stringstream ss; + ss << "The pauli string '" << *this; + ss << "' doesn't have a well defined deterministic value before '" << inst; + ss << "'."; + throw std::invalid_argument(ss.str()); + } + + default: + throw std::invalid_argument("Not implemented in PauliStringRef::undo_instruction: " + inst.str()); } } template PauliString PauliStringRef::after(const Circuit &circuit) const { PauliString result = *this; - result.ref().after_inplace(circuit); + result.ref().do_circuit(circuit); return result; } template PauliString PauliStringRef::after(const Tableau &tableau, SpanRef indices) const { PauliString result = *this; - result.ref().after_inplace_broadcast(tableau, indices, false); + result.ref().do_tableau(tableau, indices, false); return result; } template -PauliString PauliStringRef::after(const CircuitInstruction &operation) const { +PauliString PauliStringRef::after(const CircuitInstruction &inst) const { PauliString result = *this; - result.ref().after_inplace(operation, false); + result.ref().do_instruction(inst); return result; } template PauliString PauliStringRef::before(const Circuit &circuit) const { - // TODO: use hand-optimized methods instead of the generic circuit inverse method. - return after(circuit.inverse(true)); + PauliString result = *this; + result.ref().undo_circuit(circuit); + return result; } template -PauliString PauliStringRef::before(const CircuitInstruction &operation) const { +PauliString PauliStringRef::before(const CircuitInstruction &inst) const { PauliString result = *this; - result.ref().after_inplace(operation, true); + result.ref().undo_instruction(inst); return result; } template PauliString PauliStringRef::before(const Tableau &tableau, SpanRef indices) const { PauliString result = *this; - result.ref().after_inplace_broadcast(tableau, indices, true); + result.ref().do_tableau(tableau, indices, true); return result; } @@ -348,4 +669,491 @@ std::ostream &operator<<(std::ostream &out, const PauliStringRef &ps) { return out; } +template +void PauliStringRef::do_H_XZ(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + xs[q].swap_with(zs[q]); + sign ^= xs[q] && zs[q]; + } +} + +template +void PauliStringRef::do_SQRT_Y(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + xs[q].swap_with(zs[q]); + sign ^= !xs[q] && zs[q]; + } +} + +template +void PauliStringRef::do_SQRT_Y_DAG(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + xs[q].swap_with(zs[q]); + sign ^= xs[q] && !zs[q]; + } +} + +template +void PauliStringRef::do_H_XY(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + zs[q] ^= xs[q]; + sign ^= !xs[q] && zs[q]; + } +} + +template +void PauliStringRef::do_SQRT_Z(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + zs[q] ^= xs[q]; + sign ^= xs[q] && !zs[q]; + } +} + +template +void PauliStringRef::do_SQRT_Z_DAG(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + zs[q] ^= xs[q]; + sign ^= xs[q] && zs[q]; + } +} + +template +void PauliStringRef::do_H_YZ(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + xs[q] ^= zs[q]; + sign ^= xs[q] && !zs[q]; + } +} + +template +void PauliStringRef::do_SQRT_X(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + xs[q] ^= zs[q]; + sign ^= xs[q] && zs[q]; + } +} + +template +void PauliStringRef::do_SQRT_X_DAG(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + xs[q] ^= zs[q]; + sign ^= !xs[q] && zs[q]; + } +} + +template +void PauliStringRef::do_C_XYZ(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + xs[q] ^= zs[q]; + zs[q] ^= xs[q]; + } +} + +template +void PauliStringRef::do_C_ZYX(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + zs[q] ^= xs[q]; + xs[q] ^= zs[q]; + } +} + +template +void PauliStringRef::do_X(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + sign ^= zs[q]; + } +} + +template +void PauliStringRef::do_Y(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + sign ^= xs[q] ^ zs[q]; + } +} + +template +void PauliStringRef::do_Z(const CircuitInstruction &inst) { + for (auto t : inst.targets) { + auto q = t.data; + sign ^= xs[q]; + } +} + +template +void PauliStringRef::do_single_cx(const CircuitInstruction &inst, uint32_t c, uint32_t t) { + c &= ~TARGET_INVERTED_BIT; + t &= ~TARGET_INVERTED_BIT; + if (!((c | t) & (TARGET_RECORD_BIT | TARGET_SWEEP_BIT))) { + bit_ref x1 = xs[c], x2 = xs[t], z1 = zs[c], z2 = zs[t]; + z1 ^= z2; + x2 ^= x1; + sign ^= x1 && z2 && (z1 == x2); + } else if (t & (TARGET_RECORD_BIT | TARGET_SWEEP_BIT)) { + throw std::invalid_argument( + "CX had a bit (" + GateTarget{t}.str() + ") as its target, instead of its control."); + } else { + if (zs[t]) { + std::stringstream ss; + ss << "The pauli observable '" << *this; + ss << "' is affected by a controlled operation in '" << inst; + ss << "' but the controlling measurement result isn't known."; + throw std::invalid_argument(ss.str()); + } + } +} + +template +void PauliStringRef::do_single_cy(const CircuitInstruction &inst, uint32_t c, uint32_t t) { + c &= ~TARGET_INVERTED_BIT; + t &= ~TARGET_INVERTED_BIT; + if (!((c | t) & (TARGET_RECORD_BIT | TARGET_SWEEP_BIT))) { + bit_ref x1 = xs[c], x2 = xs[t], z1 = zs[c], z2 = zs[t]; + z1 ^= x2 ^ z2; + z2 ^= x1; + x2 ^= x1; + sign ^= x1 && !z1 && x2 && !z2; + sign ^= x1 && z1 && !x2 && z2; + } else if (t & (TARGET_RECORD_BIT | TARGET_SWEEP_BIT)) { + throw std::invalid_argument( + "CY had a bit (" + GateTarget{t}.str() + ") as its target, instead of its control."); + } else { + if (xs[t] ^ zs[t]) { + std::stringstream ss; + ss << "The pauli observable '" << *this; + ss << "' is affected by a controlled operation in '" << inst; + ss << "' but the controlling measurement result isn't known."; + throw std::invalid_argument(ss.str()); + } + } +} + +template +void PauliStringRef::do_single_cz(const CircuitInstruction &inst, uint32_t c, uint32_t t) { + c &= ~TARGET_INVERTED_BIT; + t &= ~TARGET_INVERTED_BIT; + if (!((c | t) & (TARGET_RECORD_BIT | TARGET_SWEEP_BIT))) { + bit_ref x1 = xs[c], x2 = xs[t], z1 = zs[c], z2 = zs[t]; + z1 ^= x2; + z2 ^= x1; + sign ^= x1 && x2 && (z1 ^ z2); + } else { + bool bc = !(c & (TARGET_RECORD_BIT | TARGET_SWEEP_BIT)) && xs[c]; + bool bt = !(t & (TARGET_RECORD_BIT | TARGET_SWEEP_BIT)) && xs[t]; + if (bc || bt) { + std::stringstream ss; + ss << "The pauli observable '" << *this; + ss << "' is affected by a controlled operation in '" << inst; + ss << "' but the controlling measurement result isn't known."; + throw std::invalid_argument(ss.str()); + } + } +} + +template +template +void PauliStringRef::do_ZCX(const CircuitInstruction &inst) { + const auto &targets = inst.targets; + assert((targets.size() & 1) == 0); + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + do_single_cx(inst, q1, q2); + } +} + +template +template +void PauliStringRef::do_ZCY(const CircuitInstruction &inst) { + const auto &targets = inst.targets; + assert((targets.size() & 1) == 0); + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + do_single_cy(inst, q1, q2); + } +} + +template +void PauliStringRef::do_ZCZ(const CircuitInstruction &inst) { + const auto &targets = inst.targets; + assert((targets.size() & 1) == 0); + for (size_t k = 0; k < targets.size(); k += 2) { + do_single_cz(inst, targets[k].data, targets[k + 1].data); + } +} + +template +template +void PauliStringRef::do_SWAP(const CircuitInstruction &inst) { + const auto &targets = inst.targets; + assert((targets.size() & 1) == 0); + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + zs[q1].swap_with(zs[q2]); + xs[q1].swap_with(xs[q2]); + } +} + +template +template +void PauliStringRef::do_ISWAP(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + sign ^= x1 && z1 && !x2 && !z2; + sign ^= !x1 && !z1 && x2 && z2; + sign ^= (x1 ^ x2) && z1 && z2; + auto dx = x1 ^ x2; + z1 ^= dx; + z2 ^= dx; + z1.swap_with(z2); + x1.swap_with(x2); + } +} + +template +template +void PauliStringRef::do_ISWAP_DAG(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + auto dx = x1 ^ x2; + z1 ^= dx; + z2 ^= dx; + z1.swap_with(z2); + x1.swap_with(x2); + sign ^= x1 && z1 && !x2 && !z2; + sign ^= !x1 && !z1 && x2 && z2; + sign ^= (x1 ^ x2) && z1 && z2; + } +} + +template +template +void PauliStringRef::do_CXSWAP(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + sign ^= x1 && z2 && (z1 == x2); + z2 ^= z1; + z1 ^= z2; + x1 ^= x2; + x2 ^= x1; + } +} + +template +template +void PauliStringRef::do_CZSWAP(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + z1.swap_with(z2); + x1.swap_with(x2); + z1 ^= x2; + z2 ^= x1; + sign ^= x1 && x2 && (z1 ^ z2); + } +} + +template +template +void PauliStringRef::do_SWAPCX(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + z1 ^= z2; + z2 ^= z1; + x2 ^= x1; + x1 ^= x2; + sign ^= x1 && z2 && (z1 == x2); + } +} + +template +void PauliStringRef::do_SQRT_XX(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t q1 = inst.targets[k].data, q2 = inst.targets[k + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + sign ^= !x1 && z1 && !z2; + sign ^= !x2 && !z1 && z2; + auto dz = z1 ^ z2; + x1 ^= dz; + x2 ^= dz; + } +} + +template +void PauliStringRef::do_SQRT_XX_DAG(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t q1 = inst.targets[k].data, q2 = inst.targets[k + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + auto dz = z1 ^ z2; + x1 ^= dz; + x2 ^= dz; + sign ^= !x1 && z1 && !z2; + sign ^= !x2 && !z1 && z2; + } +} + +template +void PauliStringRef::do_SQRT_YY(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t q1 = inst.targets[k].data, q2 = inst.targets[k + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + sign ^= x1 && z1 && x2 && !z2; + sign ^= x1 && !z1 && x2 && z2; + sign ^= x1 && !z1 && !x2 && !z2; + sign ^= !x1 && !z1 && x2 && !z2; + auto d = x1 ^ z1 ^ x2 ^ z2; + x1 ^= d; + z1 ^= d; + x2 ^= d; + z2 ^= d; + } +} + +template +void PauliStringRef::do_SQRT_YY_DAG(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t q1 = inst.targets[k].data, q2 = inst.targets[k + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + auto d = x1 ^ z1 ^ x2 ^ z2; + x1 ^= d; + z1 ^= d; + x2 ^= d; + z2 ^= d; + sign ^= x1 && z1 && x2 && !z2; + sign ^= x1 && !z1 && x2 && z2; + sign ^= x1 && !z1 && !x2 && !z2; + sign ^= !x1 && !z1 && x2 && !z2; + } +} + +template +void PauliStringRef::do_SQRT_ZZ(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t q1 = inst.targets[k].data, q2 = inst.targets[k + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + auto dx = x1 ^ x2; + z1 ^= dx; + z2 ^= dx; + sign ^= !z1 && x1 && !x2; + sign ^= !z2 && !x1 && x2; + } +} + +template +void PauliStringRef::do_SQRT_ZZ_DAG(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t q1 = inst.targets[k].data, q2 = inst.targets[k + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + sign ^= !z1 && x1 && !x2; + sign ^= !z2 && !x1 && x2; + auto dx = x1 ^ x2; + z1 ^= dx; + z2 ^= dx; + } +} + +template +void PauliStringRef::do_XCX(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t q1 = inst.targets[k].data, q2 = inst.targets[k + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + sign ^= (x1 != x2) && z1 && z2; + x1 ^= z2; + x2 ^= z1; + } +} + +template +template +void PauliStringRef::do_XCY(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + x1 ^= x2 ^ z2; + x2 ^= z1; + z2 ^= z1; + sign ^= !x1 && z1 && !x2 && z2; + sign ^= x1 && z1 && x2 && !z2; + } +} + +template +template +void PauliStringRef::do_XCZ(const CircuitInstruction &inst) { + const auto &targets = inst.targets; + assert((targets.size() & 1) == 0); + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + do_single_cx(inst, q2, q1); + } +} + +template +template +void PauliStringRef::do_YCX(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + x2 ^= x1 ^ z1; + x1 ^= z2; + z1 ^= z2; + sign ^= !x2 && z2 && !x1 && z1; + sign ^= x2 && z2 && x1 && !z1; + } +} + +template +void PauliStringRef::do_YCY(const CircuitInstruction &inst) { + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t q1 = inst.targets[k].data, q2 = inst.targets[k + 1].data; + bit_ref x1 = xs[q1], z1 = zs[q1], x2 = xs[q2], z2 = zs[q2]; + bool y1 = x1 ^ z1; + bool y2 = x2 ^ z2; + x1 ^= y2; + z1 ^= y2; + x2 ^= y1; + z2 ^= y1; + sign ^= x1 && !z1 && !x2 && z2; + sign ^= !x1 && z1 && x2 && !z2; + } +} + +template +template +void PauliStringRef::do_YCZ(const CircuitInstruction &inst) { + const auto &targets = inst.targets; + assert((targets.size() & 1) == 0); + for (size_t k = 0; k < inst.targets.size(); k += 2) { + size_t k2 = reverse_order ? inst.targets.size() - 2 - k : k; + size_t q1 = inst.targets[k2].data, q2 = inst.targets[k2 + 1].data; + do_single_cy(inst, q2, q1); + } +} + } // namespace stim diff --git a/src/stim/stabilizers/pauli_string_ref.test.cc b/src/stim/stabilizers/pauli_string_ref.test.cc new file mode 100644 index 000000000..e730c82f8 --- /dev/null +++ b/src/stim/stabilizers/pauli_string_ref.test.cc @@ -0,0 +1,78 @@ +// Copyright 2021 Google LLC +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +#include "stim/stabilizers/pauli_string_ref.h" + +#include "gtest/gtest.h" + +#include "stim/mem/simd_word.test.h" +#include "stim/simulators/tableau_simulator.h" +#include "stim/stabilizers/pauli_string.h" +#include "stim/test_util.test.h" + +using namespace stim; + +template +void check_pauli_string_do_instruction_agrees_with_tableau_sim(Gate gate, TableauSimulator &sim) { + std::vector targets{ + GateTarget::qubit(3), + GateTarget::qubit(5), + GateTarget::qubit(8), + GateTarget::qubit(5), + }; + CircuitInstruction inst{gate.id, {}, targets}; + + std::vector> before; + for (size_t k = 0; k < 16; k++) { + sim.inv_state.zs[k].sign = false; + before.push_back(sim.inv_state.zs[k]); + } + auto tableau = gate.tableau(); + if (gate.flags & GATE_TARGETS_PAIRS) { + sim.inv_state.inplace_scatter_append(tableau, {3, 5}); + sim.inv_state.inplace_scatter_append(tableau, {8, 5}); + } else { + sim.inv_state.inplace_scatter_append(tableau, {3}); + sim.inv_state.inplace_scatter_append(tableau, {5}); + sim.inv_state.inplace_scatter_append(tableau, {8}); + sim.inv_state.inplace_scatter_append(tableau, {5}); + } + + PauliString v(0); + for (size_t k = 0; k < before.size(); k++) { + v = before[k]; + v.ref().do_instruction(inst); + if (v != sim.inv_state.zs[k]) { + EXPECT_EQ(v, sim.inv_state.zs[k]) << "do_" << gate.name << "\nbefore=" << before[k]; + return; + } + + v.ref().undo_instruction(inst); + if (v != before[k]) { + EXPECT_EQ(v, sim.inv_state.zs[k]) << "undo_" << gate.name; + return; + } + } +} + +TEST_EACH_WORD_SIZE_W(pauli_string, do_instruction_agrees_with_tableau_sim, { + TableauSimulator sim(INDEPENDENT_TEST_RNG(), 16); + sim.inv_state = Tableau::random(sim.inv_state.num_qubits, sim.rng); + + for (const auto &gate : GATE_DATA.items) { + if (gate.flags & GATE_IS_UNITARY) { + check_pauli_string_do_instruction_agrees_with_tableau_sim(gate, sim); + } + } +}) diff --git a/src/stim/stabilizers/tableau.inl b/src/stim/stabilizers/tableau.inl index 88d861599..812500171 100644 --- a/src/stim/stabilizers/tableau.inl +++ b/src/stim/stabilizers/tableau.inl @@ -19,7 +19,7 @@ #include #include -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" #include "stim/simulators/vector_simulator.h" #include "stim/stabilizers/pauli_string.h" #include "stim/stabilizers/tableau.h" diff --git a/src/stim/stabilizers/tableau.test.cc b/src/stim/stabilizers/tableau.test.cc index c224bf253..416cc47fd 100644 --- a/src/stim/stabilizers/tableau.test.cc +++ b/src/stim/stabilizers/tableau.test.cc @@ -18,7 +18,7 @@ #include "gtest/gtest.h" -#include "stim/circuit/gate_data.h" +#include "stim/gates/gates.h" #include "stim/mem/simd_word.test.h" #include "stim/simulators/vector_simulator.h" #include "stim/stabilizers/tableau_transposed_raii.h" diff --git a/testdata/circuit_all_ops_3d.gltf b/testdata/circuit_all_ops_3d.gltf index d20d88a6f..75c0eafc7 100644 --- a/testdata/circuit_all_ops_3d.gltf +++ b/testdata/circuit_all_ops_3d.gltf @@ -1 +1 @@ -{"accessors":[{"bufferView":0,"byteOffset":0,"componentType":5126,"count":12,"max":[0,0.5,0.5],"min":[0,-0.5,-0.5],"name":"cube","type":"VEC3"},{"bufferView":1,"byteOffset":0,"componentType":5126,"count":12,"max":[0.0625,0.4375],"min":[0,0.375],"name":"tex_coords_gate_I","type":"VEC2"},{"bufferView":2,"byteOffset":0,"componentType":5126,"count":12,"max":[0.0625,0.4375],"min":[0,0.375],"name":"tex_coords_gate_X","type":"VEC2"},{"bufferView":3,"byteOffset":0,"componentType":5126,"count":12,"max":[0.0625,0.5],"min":[0,0.4375],"name":"tex_coords_gate_Y","type":"VEC2"},{"bufferView":4,"byteOffset":0,"componentType":5126,"count":12,"max":[0.0625,0.5625],"min":[0,0.5],"name":"tex_coords_gate_Z","type":"VEC2"},{"bufferView":5,"byteOffset":0,"componentType":5126,"count":12,"max":[0.125,0.625],"min":[0.0625,0.5625],"name":"tex_coords_gate_C_XYZ","type":"VEC2"},{"bufferView":6,"byteOffset":0,"componentType":5126,"count":12,"max":[0.1875,0.625],"min":[0.125,0.5625],"name":"tex_coords_gate_C_ZYX","type":"VEC2"},{"bufferView":7,"byteOffset":0,"componentType":5126,"count":12,"max":[0.125,0.5625],"min":[0.0625,0.5],"name":"tex_coords_gate_H_XY","type":"VEC2"},{"bufferView":8,"byteOffset":0,"componentType":5126,"count":12,"max":[0.125,0.5],"min":[0.0625,0.4375],"name":"tex_coords_gate_H","type":"VEC2"},{"bufferView":9,"byteOffset":0,"componentType":5126,"count":12,"max":[0.125,0.4375],"min":[0.0625,0.375],"name":"tex_coords_gate_H_YZ","type":"VEC2"},{"bufferView":10,"byteOffset":0,"componentType":5126,"count":12,"max":[0.1875,0.4375],"min":[0.125,0.375],"name":"tex_coords_gate_SQRT_X","type":"VEC2"},{"bufferView":11,"byteOffset":0,"componentType":5126,"count":12,"max":[0.25,0.4375],"min":[0.1875,0.375],"name":"tex_coords_gate_SQRT_X_DAG","type":"VEC2"},{"bufferView":12,"byteOffset":0,"componentType":5126,"count":12,"max":[0.1875,0.5],"min":[0.125,0.4375],"name":"tex_coords_gate_SQRT_Y","type":"VEC2"},{"bufferView":13,"byteOffset":0,"componentType":5126,"count":12,"max":[0.25,0.5],"min":[0.1875,0.4375],"name":"tex_coords_gate_SQRT_Y_DAG","type":"VEC2"},{"bufferView":14,"byteOffset":0,"componentType":5126,"count":12,"max":[0.1875,0.5625],"min":[0.125,0.5],"name":"tex_coords_gate_S","type":"VEC2"},{"bufferView":15,"byteOffset":0,"componentType":5126,"count":12,"max":[0.25,0.5625],"min":[0.1875,0.5],"name":"tex_coords_gate_S_DAG","type":"VEC2"},{"bufferView":16,"byteOffset":0,"componentType":5126,"count":17,"max":[0,0.400000005960464,0.400000005960464],"min":[0,-0.400000005960464,-0.400000005960464],"name":"circle_loop","type":"VEC3"},{"bufferView":17,"byteOffset":0,"componentType":5126,"count":4,"max":[0,0.45254835486412,0.45254835486412],"min":[0,-0.45254835486412,-0.45254835486412],"name":"control_zswap_line_cross","type":"VEC3"},{"bufferView":18,"byteOffset":0,"componentType":5126,"count":17,"max":[0,0.400000005960464,0.400000005960464],"min":[0,-0.400000005960464,-0.400000005960464],"name":"circle_loop","type":"VEC3"},{"bufferView":19,"byteOffset":0,"componentType":5126,"count":4,"max":[0,0.45254835486412,0.45254835486412],"min":[0,-0.45254835486412,-0.45254835486412],"name":"control_xswap_line_cross","type":"VEC3"},{"bufferView":20,"byteOffset":0,"componentType":5126,"count":12,"max":[0.375,0.625],"min":[0.3125,0.5625],"name":"tex_coords_gate_ISWAP","type":"VEC2"},{"bufferView":21,"byteOffset":0,"componentType":5126,"count":12,"max":[0.4375,0.625],"min":[0.375,0.5625],"name":"tex_coords_gate_ISWAP_DAG","type":"VEC2"},{"bufferView":22,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5,0.625],"min":[0.4375,0.5625],"name":"tex_coords_gate_SWAP","type":"VEC2"},{"bufferView":23,"byteOffset":0,"componentType":5126,"count":12,"max":[0.75,0.4375],"min":[0.6875,0.375],"name":"tex_coords_gate_SQRT_XX","type":"VEC2"},{"bufferView":24,"byteOffset":0,"componentType":5126,"count":12,"max":[0.8125,0.4375],"min":[0.75,0.375],"name":"tex_coords_gate_SQRT_XX_DAG","type":"VEC2"},{"bufferView":25,"byteOffset":0,"componentType":5126,"count":12,"max":[0.75,0.5],"min":[0.6875,0.4375],"name":"tex_coords_gate_SQRT_YY","type":"VEC2"},{"bufferView":26,"byteOffset":0,"componentType":5126,"count":12,"max":[0.8125,0.5],"min":[0.75,0.4375],"name":"tex_coords_gate_SQRT_YY_DAG","type":"VEC2"},{"bufferView":27,"byteOffset":0,"componentType":5126,"count":12,"max":[0.75,0.5625],"min":[0.6875,0.5],"name":"tex_coords_gate_SQRT_ZZ","type":"VEC2"},{"bufferView":28,"byteOffset":0,"componentType":5126,"count":12,"max":[0.8125,0.5625],"min":[0.75,0.5],"name":"tex_coords_gate_SQRT_ZZ_DAG","type":"VEC2"},{"bufferView":29,"byteOffset":0,"componentType":5126,"count":17,"max":[0,0.400000005960464,0.400000005960464],"min":[0,-0.400000005960464,-0.400000005960464],"name":"circle_loop","type":"VEC3"},{"bufferView":30,"byteOffset":0,"componentType":5126,"count":4,"max":[0,0.400000005960464,0.400000005960464],"min":[0,-0.400000005960464,-0.400000005960464],"name":"control_x_line_cross","type":"VEC3"},{"bufferView":31,"byteOffset":0,"componentType":5126,"count":3,"max":[0,0.400000005960464,0.346410155296326],"min":[0,-0.200000032782555,-0.346410185098648],"name":"circle_loop","type":"VEC3"},{"bufferView":32,"byteOffset":0,"componentType":5126,"count":17,"max":[0,0.400000005960464,0.400000005960464],"min":[0,-0.400000005960464,-0.400000005960464],"name":"circle_loop","type":"VEC3"},{"bufferView":33,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5625,0.4375],"min":[0.5,0.375],"name":"tex_coords_gate_E:X","type":"VEC2"},{"bufferView":34,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5625,0.5],"min":[0.5,0.4375],"name":"tex_coords_gate_E:Y","type":"VEC2"},{"bufferView":35,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5625,0.5625],"min":[0.5,0.5],"name":"tex_coords_gate_E:Z","type":"VEC2"},{"bufferView":36,"byteOffset":0,"componentType":5126,"count":12,"max":[0.625,0.4375],"min":[0.5625,0.375],"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:X","type":"VEC2"},{"bufferView":37,"byteOffset":0,"componentType":5126,"count":12,"max":[0.625,0.5],"min":[0.5625,0.4375],"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Y","type":"VEC2"},{"bufferView":38,"byteOffset":0,"componentType":5126,"count":12,"max":[0.625,0.5625],"min":[0.5625,0.5],"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Z","type":"VEC2"},{"bufferView":39,"byteOffset":0,"componentType":5126,"count":12,"max":[0.25,0.625],"min":[0.1875,0.5625],"name":"tex_coords_gate_DEPOLARIZE1","type":"VEC2"},{"bufferView":40,"byteOffset":0,"componentType":5126,"count":12,"max":[0.3125,0.625],"min":[0.25,0.5625],"name":"tex_coords_gate_DEPOLARIZE2","type":"VEC2"},{"bufferView":41,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5625,0.625],"min":[0.5,0.5625],"name":"tex_coords_gate_PAULI_CHANNEL_1","type":"VEC2"},{"bufferView":42,"byteOffset":0,"componentType":5126,"count":12,"max":[0.625,0.625],"min":[0.5625,0.5625],"name":"tex_coords_gate_PAULI_CHANNEL_2","type":"VEC2"},{"bufferView":43,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5,0.4375],"min":[0.4375,0.375],"name":"tex_coords_gate_X_ERROR","type":"VEC2"},{"bufferView":44,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5,0.5],"min":[0.4375,0.4375],"name":"tex_coords_gate_Y_ERROR","type":"VEC2"},{"bufferView":45,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5,0.5625],"min":[0.4375,0.5],"name":"tex_coords_gate_Z_ERROR","type":"VEC2"},{"bufferView":46,"byteOffset":0,"componentType":5126,"count":12,"max":[0.9375,0.625],"min":[0.875,0.5625],"name":"tex_coords_gate_HERALDED_ERASE","type":"VEC2"},{"bufferView":47,"byteOffset":0,"componentType":5126,"count":12,"max":[1,0.625],"min":[0.9375,0.5625],"name":"tex_coords_gate_HERALDED_PAULI_CHANNEL_1","type":"VEC2"},{"bufferView":48,"byteOffset":0,"componentType":5126,"count":12,"max":[0.6875,0.4375],"min":[0.625,0.375],"name":"tex_coords_gate_MPP:X","type":"VEC2"},{"bufferView":49,"byteOffset":0,"componentType":5126,"count":12,"max":[0.6875,0.5],"min":[0.625,0.4375],"name":"tex_coords_gate_MPP:Y","type":"VEC2"},{"bufferView":50,"byteOffset":0,"componentType":5126,"count":12,"max":[0.6875,0.5625],"min":[0.625,0.5],"name":"tex_coords_gate_MPP:Z","type":"VEC2"},{"bufferView":51,"byteOffset":0,"componentType":5126,"count":12,"max":[0.4375,0.4375],"min":[0.375,0.375],"name":"tex_coords_gate_MRX","type":"VEC2"},{"bufferView":52,"byteOffset":0,"componentType":5126,"count":12,"max":[0.4375,0.5],"min":[0.375,0.4375],"name":"tex_coords_gate_MRY","type":"VEC2"},{"bufferView":53,"byteOffset":0,"componentType":5126,"count":12,"max":[0.4375,0.5625],"min":[0.375,0.5],"name":"tex_coords_gate_MR","type":"VEC2"},{"bufferView":54,"byteOffset":0,"componentType":5126,"count":12,"max":[0.3125,0.4375],"min":[0.25,0.375],"name":"tex_coords_gate_MX","type":"VEC2"},{"bufferView":55,"byteOffset":0,"componentType":5126,"count":12,"max":[0.3125,0.5],"min":[0.25,0.4375],"name":"tex_coords_gate_MY","type":"VEC2"},{"bufferView":56,"byteOffset":0,"componentType":5126,"count":12,"max":[0.3125,0.5625],"min":[0.25,0.5],"name":"tex_coords_gate_M","type":"VEC2"},{"bufferView":57,"byteOffset":0,"componentType":5126,"count":12,"max":[0.375,0.4375],"min":[0.3125,0.375],"name":"tex_coords_gate_RX","type":"VEC2"},{"bufferView":58,"byteOffset":0,"componentType":5126,"count":12,"max":[0.375,0.5],"min":[0.3125,0.4375],"name":"tex_coords_gate_RY","type":"VEC2"},{"bufferView":59,"byteOffset":0,"componentType":5126,"count":12,"max":[0.375,0.5625],"min":[0.3125,0.5],"name":"tex_coords_gate_R","type":"VEC2"},{"bufferView":60,"byteOffset":0,"componentType":5126,"count":12,"max":[0.6875,0.625],"min":[0.625,0.5625],"name":"tex_coords_gate_MXX","type":"VEC2"},{"bufferView":61,"byteOffset":0,"componentType":5126,"count":12,"max":[0.75,0.625],"min":[0.6875,0.5625],"name":"tex_coords_gate_MYY","type":"VEC2"},{"bufferView":62,"byteOffset":0,"componentType":5126,"count":12,"max":[0.8125,0.625],"min":[0.75,0.5625],"name":"tex_coords_gate_MZZ","type":"VEC2"},{"bufferView":63,"byteOffset":0,"componentType":5126,"count":12,"max":[0.875,0.625],"min":[0.8125,0.5625],"name":"tex_coords_gate_MPAD","type":"VEC2"},{"bufferView":64,"byteOffset":0,"componentType":5126,"count":12,"max":[0.875,0.4375],"min":[0.8125,0.375],"name":"tex_coords_gate_X:REC","type":"VEC2"},{"bufferView":65,"byteOffset":0,"componentType":5126,"count":12,"max":[0.9375,0.5],"min":[0.875,0.4375],"name":"tex_coords_gate_Y:SWEEP","type":"VEC2"},{"bufferView":66,"byteOffset":0,"componentType":5126,"count":12,"max":[0.875,0.5625],"min":[0.8125,0.5],"name":"tex_coords_gate_Z:REC","type":"VEC2"},{"bufferView":67,"byteOffset":0,"componentType":5126,"count":128,"max":[1,-2,-0],"min":[-26,-34,-32],"name":"buf_scattered_lines","type":"VEC3"},{"bufferView":68,"byteOffset":0,"componentType":5126,"count":30,"max":[0,0.5,1],"min":[-17.25,-35,-33],"name":"buf_red_scattered_lines","type":"VEC3"},{"bufferView":69,"byteOffset":0,"componentType":5126,"count":96,"max":[-0.75,-1.20000004768372,0.5],"min":[-23.25,-1.60000002384186,-32.5],"name":"buf_blue_scattered_lines","type":"VEC3"}],"asset":{"version":"2.0"},"bufferViews":[{"buffer":0,"byteLength":144,"byteOffset":0,"name":"cube","target":34962},{"buffer":1,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_I","target":34962},{"buffer":2,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_X","target":34962},{"buffer":3,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Y","target":34962},{"buffer":4,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Z","target":34962},{"buffer":5,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_C_XYZ","target":34962},{"buffer":6,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_C_ZYX","target":34962},{"buffer":7,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_H_XY","target":34962},{"buffer":8,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_H","target":34962},{"buffer":9,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_H_YZ","target":34962},{"buffer":10,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_X","target":34962},{"buffer":11,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_X_DAG","target":34962},{"buffer":12,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_Y","target":34962},{"buffer":13,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_Y_DAG","target":34962},{"buffer":14,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_S","target":34962},{"buffer":15,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_S_DAG","target":34962},{"buffer":16,"byteLength":204,"byteOffset":0,"name":"circle_loop","target":34962},{"buffer":17,"byteLength":48,"byteOffset":0,"name":"control_zswap_line_cross","target":34962},{"buffer":18,"byteLength":204,"byteOffset":0,"name":"circle_loop","target":34962},{"buffer":19,"byteLength":48,"byteOffset":0,"name":"control_xswap_line_cross","target":34962},{"buffer":20,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_ISWAP","target":34962},{"buffer":21,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_ISWAP_DAG","target":34962},{"buffer":22,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SWAP","target":34962},{"buffer":23,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_XX","target":34962},{"buffer":24,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_XX_DAG","target":34962},{"buffer":25,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_YY","target":34962},{"buffer":26,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_YY_DAG","target":34962},{"buffer":27,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_ZZ","target":34962},{"buffer":28,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_ZZ_DAG","target":34962},{"buffer":29,"byteLength":204,"byteOffset":0,"name":"circle_loop","target":34962},{"buffer":30,"byteLength":48,"byteOffset":0,"name":"control_x_line_cross","target":34962},{"buffer":31,"byteLength":36,"byteOffset":0,"name":"circle_loop","target":34962},{"buffer":32,"byteLength":204,"byteOffset":0,"name":"circle_loop","target":34962},{"buffer":33,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_E:X","target":34962},{"buffer":34,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_E:Y","target":34962},{"buffer":35,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_E:Z","target":34962},{"buffer":36,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:X","target":34962},{"buffer":37,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Y","target":34962},{"buffer":38,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Z","target":34962},{"buffer":39,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_DEPOLARIZE1","target":34962},{"buffer":40,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_DEPOLARIZE2","target":34962},{"buffer":41,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_PAULI_CHANNEL_1","target":34962},{"buffer":42,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_PAULI_CHANNEL_2","target":34962},{"buffer":43,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_X_ERROR","target":34962},{"buffer":44,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Y_ERROR","target":34962},{"buffer":45,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Z_ERROR","target":34962},{"buffer":46,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_HERALDED_ERASE","target":34962},{"buffer":47,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_HERALDED_PAULI_CHANNEL_1","target":34962},{"buffer":48,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MPP:X","target":34962},{"buffer":49,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MPP:Y","target":34962},{"buffer":50,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MPP:Z","target":34962},{"buffer":51,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MRX","target":34962},{"buffer":52,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MRY","target":34962},{"buffer":53,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MR","target":34962},{"buffer":54,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MX","target":34962},{"buffer":55,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MY","target":34962},{"buffer":56,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_M","target":34962},{"buffer":57,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_RX","target":34962},{"buffer":58,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_RY","target":34962},{"buffer":59,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_R","target":34962},{"buffer":60,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MXX","target":34962},{"buffer":61,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MYY","target":34962},{"buffer":62,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MZZ","target":34962},{"buffer":63,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MPAD","target":34962},{"buffer":64,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_X:REC","target":34962},{"buffer":65,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Y:SWEEP","target":34962},{"buffer":66,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Z:REC","target":34962},{"buffer":67,"byteLength":1536,"byteOffset":0,"name":"buf_scattered_lines","target":34962},{"buffer":68,"byteLength":360,"byteOffset":0,"name":"buf_red_scattered_lines","target":34962},{"buffer":69,"byteLength":1152,"byteOffset":0,"name":"buf_blue_scattered_lines","target":34962}],"buffers":[{"byteLength":144,"name":"cube","uri":"data:application/octet-stream;base64,AAAAAAAAAD8AAAA/AAAAAAAAAD8AAAC/AAAAAAAAAL8AAAA/AAAAAAAAAD8AAAC/AAAAAAAAAL8AAAC/AAAAAAAAAL8AAAA/AAAAAAAAAL8AAAC/AAAAAAAAAD8AAAC/AAAAAAAAAL8AAAA/AAAAAAAAAL8AAAA/AAAAAAAAAD8AAAC/AAAAAAAAAD8AAAA/"},{"byteLength":96,"name":"tex_coords_gate_I","uri":"data:application/octet-stream;base64,AACAPQAAwD4AAAAAAADAPgAAgD0AAOA+AAAAAAAAwD4AAAAAAADgPgAAgD0AAOA+AACAPQAA4D4AAIA9AADAPgAAAAAAAOA+AAAAAAAA4D4AAIA9AADAPgAAAAAAAMA+"},{"byteLength":96,"name":"tex_coords_gate_X","uri":"data:application/octet-stream;base64,AACAPQAAwD4AAAAAAADAPgAAgD0AAOA+AAAAAAAAwD4AAAAAAADgPgAAgD0AAOA+AACAPQAA4D4AAIA9AADAPgAAAAAAAOA+AAAAAAAA4D4AAIA9AADAPgAAAAAAAMA+"},{"byteLength":96,"name":"tex_coords_gate_Y","uri":"data:application/octet-stream;base64,AACAPQAA4D4AAAAAAADgPgAAgD0AAAA/AAAAAAAA4D4AAAAAAAAAPwAAgD0AAAA/AACAPQAAAD8AAIA9AADgPgAAAAAAAAA/AAAAAAAAAD8AAIA9AADgPgAAAAAAAOA+"},{"byteLength":96,"name":"tex_coords_gate_Z","uri":"data:application/octet-stream;base64,AACAPQAAAD8AAAAAAAAAPwAAgD0AABA/AAAAAAAAAD8AAAAAAAAQPwAAgD0AABA/AACAPQAAED8AAIA9AAAAPwAAAAAAABA/AAAAAAAAED8AAIA9AAAAPwAAAAAAAAA/"},{"byteLength":96,"name":"tex_coords_gate_C_XYZ","uri":"data:application/octet-stream;base64,AAAAPgAAED8AAIA9AAAQPwAAAD4AACA/AACAPQAAED8AAIA9AAAgPwAAAD4AACA/AAAAPgAAID8AAAA+AAAQPwAAgD0AACA/AACAPQAAID8AAAA+AAAQPwAAgD0AABA/"},{"byteLength":96,"name":"tex_coords_gate_C_ZYX","uri":"data:application/octet-stream;base64,AABAPgAAED8AAAA+AAAQPwAAQD4AACA/AAAAPgAAED8AAAA+AAAgPwAAQD4AACA/AABAPgAAID8AAEA+AAAQPwAAAD4AACA/AAAAPgAAID8AAEA+AAAQPwAAAD4AABA/"},{"byteLength":96,"name":"tex_coords_gate_H_XY","uri":"data:application/octet-stream;base64,AAAAPgAAAD8AAIA9AAAAPwAAAD4AABA/AACAPQAAAD8AAIA9AAAQPwAAAD4AABA/AAAAPgAAED8AAAA+AAAAPwAAgD0AABA/AACAPQAAED8AAAA+AAAAPwAAgD0AAAA/"},{"byteLength":96,"name":"tex_coords_gate_H","uri":"data:application/octet-stream;base64,AAAAPgAA4D4AAIA9AADgPgAAAD4AAAA/AACAPQAA4D4AAIA9AAAAPwAAAD4AAAA/AAAAPgAAAD8AAAA+AADgPgAAgD0AAAA/AACAPQAAAD8AAAA+AADgPgAAgD0AAOA+"},{"byteLength":96,"name":"tex_coords_gate_H_YZ","uri":"data:application/octet-stream;base64,AAAAPgAAwD4AAIA9AADAPgAAAD4AAOA+AACAPQAAwD4AAIA9AADgPgAAAD4AAOA+AAAAPgAA4D4AAAA+AADAPgAAgD0AAOA+AACAPQAA4D4AAAA+AADAPgAAgD0AAMA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_X","uri":"data:application/octet-stream;base64,AABAPgAAwD4AAAA+AADAPgAAQD4AAOA+AAAAPgAAwD4AAAA+AADgPgAAQD4AAOA+AABAPgAA4D4AAEA+AADAPgAAAD4AAOA+AAAAPgAA4D4AAEA+AADAPgAAAD4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_X_DAG","uri":"data:application/octet-stream;base64,AACAPgAAwD4AAEA+AADAPgAAgD4AAOA+AABAPgAAwD4AAEA+AADgPgAAgD4AAOA+AACAPgAA4D4AAIA+AADAPgAAQD4AAOA+AABAPgAA4D4AAIA+AADAPgAAQD4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_Y","uri":"data:application/octet-stream;base64,AABAPgAA4D4AAAA+AADgPgAAQD4AAAA/AAAAPgAA4D4AAAA+AAAAPwAAQD4AAAA/AABAPgAAAD8AAEA+AADgPgAAAD4AAAA/AAAAPgAAAD8AAEA+AADgPgAAAD4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_Y_DAG","uri":"data:application/octet-stream;base64,AACAPgAA4D4AAEA+AADgPgAAgD4AAAA/AABAPgAA4D4AAEA+AAAAPwAAgD4AAAA/AACAPgAAAD8AAIA+AADgPgAAQD4AAAA/AABAPgAAAD8AAIA+AADgPgAAQD4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_S","uri":"data:application/octet-stream;base64,AABAPgAAAD8AAAA+AAAAPwAAQD4AABA/AAAAPgAAAD8AAAA+AAAQPwAAQD4AABA/AABAPgAAED8AAEA+AAAAPwAAAD4AABA/AAAAPgAAED8AAEA+AAAAPwAAAD4AAAA/"},{"byteLength":96,"name":"tex_coords_gate_S_DAG","uri":"data:application/octet-stream;base64,AACAPgAAAD8AAEA+AAAAPwAAgD4AABA/AABAPgAAAD8AAEA+AAAQPwAAgD4AABA/AACAPgAAED8AAIA+AAAAPwAAQD4AABA/AABAPgAAED8AAIA+AAAAPwAAQD4AAAA/"},{"byteLength":204,"name":"circle_loop","uri":"data:application/octet-stream;base64,AAAAAM3MzD4AAAAAAAAAAOU1vT5Fvxw+AAAAAMPQkD7D0JA+AAAAAES/HD7lNb0+AAAAAPIwlrLNzMw+AAAAAEe/HL7lNb0+AAAAAMPQkL7D0JA+AAAAAOc1vb5Avxw+AAAAAM3MzL7yMBazAAAAAOU1vb5Evxy+AAAAAMHQkL7E0JC+AAAAADy/HL7nNb2+AAAAAPLkozHNzMy+AAAAAEm/HD7kNb2+AAAAAMbQkD6/0JC+AAAAAOY1vT5Evxy+AAAAAM3MzD4AAAAA"},{"byteLength":48,"name":"control_zswap_line_cross","uri":"data:application/octet-stream;base64,AAAAAGu0575rtOe+AAAAAGu05z5rtOc+AAAAAGu0575rtOc+AAAAAGu05z5rtOe+"},{"byteLength":204,"name":"circle_loop","uri":"data:application/octet-stream;base64,AAAAAM3MzD4AAAAAAAAAAOU1vT5Fvxw+AAAAAMPQkD7D0JA+AAAAAES/HD7lNb0+AAAAAPIwlrLNzMw+AAAAAEe/HL7lNb0+AAAAAMPQkL7D0JA+AAAAAOc1vb5Avxw+AAAAAM3MzL7yMBazAAAAAOU1vb5Evxy+AAAAAMHQkL7E0JC+AAAAADy/HL7nNb2+AAAAAPLkozHNzMy+AAAAAEm/HD7kNb2+AAAAAMbQkD6/0JC+AAAAAOY1vT5Evxy+AAAAAM3MzD4AAAAA"},{"byteLength":48,"name":"control_xswap_line_cross","uri":"data:application/octet-stream;base64,AAAAAGu0575rtOe+AAAAAGu05z5rtOc+AAAAAGu0575rtOc+AAAAAGu05z5rtOe+"},{"byteLength":96,"name":"tex_coords_gate_ISWAP","uri":"data:application/octet-stream;base64,AADAPgAAED8AAKA+AAAQPwAAwD4AACA/AACgPgAAED8AAKA+AAAgPwAAwD4AACA/AADAPgAAID8AAMA+AAAQPwAAoD4AACA/AACgPgAAID8AAMA+AAAQPwAAoD4AABA/"},{"byteLength":96,"name":"tex_coords_gate_ISWAP_DAG","uri":"data:application/octet-stream;base64,AADgPgAAED8AAMA+AAAQPwAA4D4AACA/AADAPgAAED8AAMA+AAAgPwAA4D4AACA/AADgPgAAID8AAOA+AAAQPwAAwD4AACA/AADAPgAAID8AAOA+AAAQPwAAwD4AABA/"},{"byteLength":96,"name":"tex_coords_gate_SWAP","uri":"data:application/octet-stream;base64,AAAAPwAAED8AAOA+AAAQPwAAAD8AACA/AADgPgAAED8AAOA+AAAgPwAAAD8AACA/AAAAPwAAID8AAAA/AAAQPwAA4D4AACA/AADgPgAAID8AAAA/AAAQPwAA4D4AABA/"},{"byteLength":96,"name":"tex_coords_gate_SQRT_XX","uri":"data:application/octet-stream;base64,AABAPwAAwD4AADA/AADAPgAAQD8AAOA+AAAwPwAAwD4AADA/AADgPgAAQD8AAOA+AABAPwAA4D4AAEA/AADAPgAAMD8AAOA+AAAwPwAA4D4AAEA/AADAPgAAMD8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_XX_DAG","uri":"data:application/octet-stream;base64,AABQPwAAwD4AAEA/AADAPgAAUD8AAOA+AABAPwAAwD4AAEA/AADgPgAAUD8AAOA+AABQPwAA4D4AAFA/AADAPgAAQD8AAOA+AABAPwAA4D4AAFA/AADAPgAAQD8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_YY","uri":"data:application/octet-stream;base64,AABAPwAA4D4AADA/AADgPgAAQD8AAAA/AAAwPwAA4D4AADA/AAAAPwAAQD8AAAA/AABAPwAAAD8AAEA/AADgPgAAMD8AAAA/AAAwPwAAAD8AAEA/AADgPgAAMD8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_YY_DAG","uri":"data:application/octet-stream;base64,AABQPwAA4D4AAEA/AADgPgAAUD8AAAA/AABAPwAA4D4AAEA/AAAAPwAAUD8AAAA/AABQPwAAAD8AAFA/AADgPgAAQD8AAAA/AABAPwAAAD8AAFA/AADgPgAAQD8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_ZZ","uri":"data:application/octet-stream;base64,AABAPwAAAD8AADA/AAAAPwAAQD8AABA/AAAwPwAAAD8AADA/AAAQPwAAQD8AABA/AABAPwAAED8AAEA/AAAAPwAAMD8AABA/AAAwPwAAED8AAEA/AAAAPwAAMD8AAAA/"},{"byteLength":96,"name":"tex_coords_gate_SQRT_ZZ_DAG","uri":"data:application/octet-stream;base64,AABQPwAAAD8AAEA/AAAAPwAAUD8AABA/AABAPwAAAD8AAEA/AAAQPwAAUD8AABA/AABQPwAAED8AAFA/AAAAPwAAQD8AABA/AABAPwAAED8AAFA/AAAAPwAAQD8AAAA/"},{"byteLength":204,"name":"circle_loop","uri":"data:application/octet-stream;base64,AAAAAM3MzD4AAAAAAAAAAOU1vT5Fvxw+AAAAAMPQkD7D0JA+AAAAAES/HD7lNb0+AAAAAPIwlrLNzMw+AAAAAEe/HL7lNb0+AAAAAMPQkL7D0JA+AAAAAOc1vb5Avxw+AAAAAM3MzL7yMBazAAAAAOU1vb5Evxy+AAAAAMHQkL7E0JC+AAAAADy/HL7nNb2+AAAAAPLkozHNzMy+AAAAAEm/HD7kNb2+AAAAAMbQkD6/0JC+AAAAAOY1vT5Evxy+AAAAAM3MzD4AAAAA"},{"byteLength":48,"name":"control_x_line_cross","uri":"data:application/octet-stream;base64,AAAAAM3MzL4AAAAAAAAAAM3MzD4AAAAAAAAAAAAAAADNzMy+AAAAAAAAAADNzMw+"},{"byteLength":36,"name":"circle_loop","uri":"data:application/octet-stream;base64,AAAAAM3MzD4AAAAAAAAAAM/MTL6sXLE+AAAAAMvMTL6tXLG+"},{"byteLength":204,"name":"circle_loop","uri":"data:application/octet-stream;base64,AAAAAM3MzD4AAAAAAAAAAOU1vT5Fvxw+AAAAAMPQkD7D0JA+AAAAAES/HD7lNb0+AAAAAPIwlrLNzMw+AAAAAEe/HL7lNb0+AAAAAMPQkL7D0JA+AAAAAOc1vb5Avxw+AAAAAM3MzL7yMBazAAAAAOU1vb5Evxy+AAAAAMHQkL7E0JC+AAAAADy/HL7nNb2+AAAAAPLkozHNzMy+AAAAAEm/HD7kNb2+AAAAAMbQkD6/0JC+AAAAAOY1vT5Evxy+AAAAAM3MzD4AAAAA"},{"byteLength":96,"name":"tex_coords_gate_E:X","uri":"data:application/octet-stream;base64,AAAQPwAAwD4AAAA/AADAPgAAED8AAOA+AAAAPwAAwD4AAAA/AADgPgAAED8AAOA+AAAQPwAA4D4AABA/AADAPgAAAD8AAOA+AAAAPwAA4D4AABA/AADAPgAAAD8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_E:Y","uri":"data:application/octet-stream;base64,AAAQPwAA4D4AAAA/AADgPgAAED8AAAA/AAAAPwAA4D4AAAA/AAAAPwAAED8AAAA/AAAQPwAAAD8AABA/AADgPgAAAD8AAAA/AAAAPwAAAD8AABA/AADgPgAAAD8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_E:Z","uri":"data:application/octet-stream;base64,AAAQPwAAAD8AAAA/AAAAPwAAED8AABA/AAAAPwAAAD8AAAA/AAAQPwAAED8AABA/AAAQPwAAED8AABA/AAAAPwAAAD8AABA/AAAAPwAAED8AABA/AAAAPwAAAD8AAAA/"},{"byteLength":96,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:X","uri":"data:application/octet-stream;base64,AAAgPwAAwD4AABA/AADAPgAAID8AAOA+AAAQPwAAwD4AABA/AADgPgAAID8AAOA+AAAgPwAA4D4AACA/AADAPgAAED8AAOA+AAAQPwAA4D4AACA/AADAPgAAED8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Y","uri":"data:application/octet-stream;base64,AAAgPwAA4D4AABA/AADgPgAAID8AAAA/AAAQPwAA4D4AABA/AAAAPwAAID8AAAA/AAAgPwAAAD8AACA/AADgPgAAED8AAAA/AAAQPwAAAD8AACA/AADgPgAAED8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Z","uri":"data:application/octet-stream;base64,AAAgPwAAAD8AABA/AAAAPwAAID8AABA/AAAQPwAAAD8AABA/AAAQPwAAID8AABA/AAAgPwAAED8AACA/AAAAPwAAED8AABA/AAAQPwAAED8AACA/AAAAPwAAED8AAAA/"},{"byteLength":96,"name":"tex_coords_gate_DEPOLARIZE1","uri":"data:application/octet-stream;base64,AACAPgAAED8AAEA+AAAQPwAAgD4AACA/AABAPgAAED8AAEA+AAAgPwAAgD4AACA/AACAPgAAID8AAIA+AAAQPwAAQD4AACA/AABAPgAAID8AAIA+AAAQPwAAQD4AABA/"},{"byteLength":96,"name":"tex_coords_gate_DEPOLARIZE2","uri":"data:application/octet-stream;base64,AACgPgAAED8AAIA+AAAQPwAAoD4AACA/AACAPgAAED8AAIA+AAAgPwAAoD4AACA/AACgPgAAID8AAKA+AAAQPwAAgD4AACA/AACAPgAAID8AAKA+AAAQPwAAgD4AABA/"},{"byteLength":96,"name":"tex_coords_gate_PAULI_CHANNEL_1","uri":"data:application/octet-stream;base64,AAAQPwAAED8AAAA/AAAQPwAAED8AACA/AAAAPwAAED8AAAA/AAAgPwAAED8AACA/AAAQPwAAID8AABA/AAAQPwAAAD8AACA/AAAAPwAAID8AABA/AAAQPwAAAD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_PAULI_CHANNEL_2","uri":"data:application/octet-stream;base64,AAAgPwAAED8AABA/AAAQPwAAID8AACA/AAAQPwAAED8AABA/AAAgPwAAID8AACA/AAAgPwAAID8AACA/AAAQPwAAED8AACA/AAAQPwAAID8AACA/AAAQPwAAED8AABA/"},{"byteLength":96,"name":"tex_coords_gate_X_ERROR","uri":"data:application/octet-stream;base64,AAAAPwAAwD4AAOA+AADAPgAAAD8AAOA+AADgPgAAwD4AAOA+AADgPgAAAD8AAOA+AAAAPwAA4D4AAAA/AADAPgAA4D4AAOA+AADgPgAA4D4AAAA/AADAPgAA4D4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_Y_ERROR","uri":"data:application/octet-stream;base64,AAAAPwAA4D4AAOA+AADgPgAAAD8AAAA/AADgPgAA4D4AAOA+AAAAPwAAAD8AAAA/AAAAPwAAAD8AAAA/AADgPgAA4D4AAAA/AADgPgAAAD8AAAA/AADgPgAA4D4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_Z_ERROR","uri":"data:application/octet-stream;base64,AAAAPwAAAD8AAOA+AAAAPwAAAD8AABA/AADgPgAAAD8AAOA+AAAQPwAAAD8AABA/AAAAPwAAED8AAAA/AAAAPwAA4D4AABA/AADgPgAAED8AAAA/AAAAPwAA4D4AAAA/"},{"byteLength":96,"name":"tex_coords_gate_HERALDED_ERASE","uri":"data:application/octet-stream;base64,AABwPwAAED8AAGA/AAAQPwAAcD8AACA/AABgPwAAED8AAGA/AAAgPwAAcD8AACA/AABwPwAAID8AAHA/AAAQPwAAYD8AACA/AABgPwAAID8AAHA/AAAQPwAAYD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_HERALDED_PAULI_CHANNEL_1","uri":"data:application/octet-stream;base64,AACAPwAAED8AAHA/AAAQPwAAgD8AACA/AABwPwAAED8AAHA/AAAgPwAAgD8AACA/AACAPwAAID8AAIA/AAAQPwAAcD8AACA/AABwPwAAID8AAIA/AAAQPwAAcD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_MPP:X","uri":"data:application/octet-stream;base64,AAAwPwAAwD4AACA/AADAPgAAMD8AAOA+AAAgPwAAwD4AACA/AADgPgAAMD8AAOA+AAAwPwAA4D4AADA/AADAPgAAID8AAOA+AAAgPwAA4D4AADA/AADAPgAAID8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_MPP:Y","uri":"data:application/octet-stream;base64,AAAwPwAA4D4AACA/AADgPgAAMD8AAAA/AAAgPwAA4D4AACA/AAAAPwAAMD8AAAA/AAAwPwAAAD8AADA/AADgPgAAID8AAAA/AAAgPwAAAD8AADA/AADgPgAAID8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_MPP:Z","uri":"data:application/octet-stream;base64,AAAwPwAAAD8AACA/AAAAPwAAMD8AABA/AAAgPwAAAD8AACA/AAAQPwAAMD8AABA/AAAwPwAAED8AADA/AAAAPwAAID8AABA/AAAgPwAAED8AADA/AAAAPwAAID8AAAA/"},{"byteLength":96,"name":"tex_coords_gate_MRX","uri":"data:application/octet-stream;base64,AADgPgAAwD4AAMA+AADAPgAA4D4AAOA+AADAPgAAwD4AAMA+AADgPgAA4D4AAOA+AADgPgAA4D4AAOA+AADAPgAAwD4AAOA+AADAPgAA4D4AAOA+AADAPgAAwD4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_MRY","uri":"data:application/octet-stream;base64,AADgPgAA4D4AAMA+AADgPgAA4D4AAAA/AADAPgAA4D4AAMA+AAAAPwAA4D4AAAA/AADgPgAAAD8AAOA+AADgPgAAwD4AAAA/AADAPgAAAD8AAOA+AADgPgAAwD4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_MR","uri":"data:application/octet-stream;base64,AADgPgAAAD8AAMA+AAAAPwAA4D4AABA/AADAPgAAAD8AAMA+AAAQPwAA4D4AABA/AADgPgAAED8AAOA+AAAAPwAAwD4AABA/AADAPgAAED8AAOA+AAAAPwAAwD4AAAA/"},{"byteLength":96,"name":"tex_coords_gate_MX","uri":"data:application/octet-stream;base64,AACgPgAAwD4AAIA+AADAPgAAoD4AAOA+AACAPgAAwD4AAIA+AADgPgAAoD4AAOA+AACgPgAA4D4AAKA+AADAPgAAgD4AAOA+AACAPgAA4D4AAKA+AADAPgAAgD4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_MY","uri":"data:application/octet-stream;base64,AACgPgAA4D4AAIA+AADgPgAAoD4AAAA/AACAPgAA4D4AAIA+AAAAPwAAoD4AAAA/AACgPgAAAD8AAKA+AADgPgAAgD4AAAA/AACAPgAAAD8AAKA+AADgPgAAgD4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_M","uri":"data:application/octet-stream;base64,AACgPgAAAD8AAIA+AAAAPwAAoD4AABA/AACAPgAAAD8AAIA+AAAQPwAAoD4AABA/AACgPgAAED8AAKA+AAAAPwAAgD4AABA/AACAPgAAED8AAKA+AAAAPwAAgD4AAAA/"},{"byteLength":96,"name":"tex_coords_gate_RX","uri":"data:application/octet-stream;base64,AADAPgAAwD4AAKA+AADAPgAAwD4AAOA+AACgPgAAwD4AAKA+AADgPgAAwD4AAOA+AADAPgAA4D4AAMA+AADAPgAAoD4AAOA+AACgPgAA4D4AAMA+AADAPgAAoD4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_RY","uri":"data:application/octet-stream;base64,AADAPgAA4D4AAKA+AADgPgAAwD4AAAA/AACgPgAA4D4AAKA+AAAAPwAAwD4AAAA/AADAPgAAAD8AAMA+AADgPgAAoD4AAAA/AACgPgAAAD8AAMA+AADgPgAAoD4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_R","uri":"data:application/octet-stream;base64,AADAPgAAAD8AAKA+AAAAPwAAwD4AABA/AACgPgAAAD8AAKA+AAAQPwAAwD4AABA/AADAPgAAED8AAMA+AAAAPwAAoD4AABA/AACgPgAAED8AAMA+AAAAPwAAoD4AAAA/"},{"byteLength":96,"name":"tex_coords_gate_MXX","uri":"data:application/octet-stream;base64,AAAwPwAAED8AACA/AAAQPwAAMD8AACA/AAAgPwAAED8AACA/AAAgPwAAMD8AACA/AAAwPwAAID8AADA/AAAQPwAAID8AACA/AAAgPwAAID8AADA/AAAQPwAAID8AABA/"},{"byteLength":96,"name":"tex_coords_gate_MYY","uri":"data:application/octet-stream;base64,AABAPwAAED8AADA/AAAQPwAAQD8AACA/AAAwPwAAED8AADA/AAAgPwAAQD8AACA/AABAPwAAID8AAEA/AAAQPwAAMD8AACA/AAAwPwAAID8AAEA/AAAQPwAAMD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_MZZ","uri":"data:application/octet-stream;base64,AABQPwAAED8AAEA/AAAQPwAAUD8AACA/AABAPwAAED8AAEA/AAAgPwAAUD8AACA/AABQPwAAID8AAFA/AAAQPwAAQD8AACA/AABAPwAAID8AAFA/AAAQPwAAQD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_MPAD","uri":"data:application/octet-stream;base64,AABgPwAAED8AAFA/AAAQPwAAYD8AACA/AABQPwAAED8AAFA/AAAgPwAAYD8AACA/AABgPwAAID8AAGA/AAAQPwAAUD8AACA/AABQPwAAID8AAGA/AAAQPwAAUD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_X:REC","uri":"data:application/octet-stream;base64,AABgPwAAwD4AAFA/AADAPgAAYD8AAOA+AABQPwAAwD4AAFA/AADgPgAAYD8AAOA+AABgPwAA4D4AAGA/AADAPgAAUD8AAOA+AABQPwAA4D4AAGA/AADAPgAAUD8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_Y:SWEEP","uri":"data:application/octet-stream;base64,AABwPwAA4D4AAGA/AADgPgAAcD8AAAA/AABgPwAA4D4AAGA/AAAAPwAAcD8AAAA/AABwPwAAAD8AAHA/AADgPgAAYD8AAAA/AABgPwAAAD8AAHA/AADgPgAAYD8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_Z:REC","uri":"data:application/octet-stream;base64,AABgPwAAAD8AAFA/AAAAPwAAYD8AABA/AABQPwAAAD8AAFA/AAAQPwAAYD8AABA/AABgPwAAED8AAGA/AAAAPwAAUD8AABA/AABQPwAAED8AAGA/AAAAPwAAUD8AAAA/"},{"byteLength":1536,"name":"buf_scattered_lines","uri":"data:application/octet-stream;base64,AABAwAAAAMIAAADCAABQwAAAiMEAAIDBAABQwAAAiMEAAIDBAABAwAAAAMAAAACAAABAwAAAgMAAAACAAABAwAAAwMAAAACAAABAwAAAAMEAAACAAABAwAAAIMEAAACAAABAwAAAQMEAAACAAABAwAAAYMEAAACAAABAwAAAgMEAAACAAABAwAAAkMEAAACAAACAwAAAAMIAAADCAACIwAAAiMEAAIDBAACIwAAAiMEAAIDBAACAwAAAAMAAAACAAACAwAAAgMAAAACAAACAwAAAwMAAAACAAACAwAAAAMEAAACAAACAwAAAIMEAAACAAACAwAAAQMEAAACAAACAwAAAYMEAAACAAACAwAAAgMEAAACAAACAwAAAkMEAAACAAACAwAAAoMEAAACAAACAwAAAsMEAAACAAACgwAAAAMIAAADCAACowAAAiMEAAIDBAACowAAAiMEAAIDBAACgwAAAAMAAAACAAACgwAAAgMAAAACAAACgwAAAwMAAAACAAACgwAAAAMEAAACAAACgwAAAIMEAAACAAACgwAAAQMEAAACAAACgwAAAYMEAAACAAACgwAAAgMEAAACAAACgwAAAkMEAAACAAACgwAAAoMEAAACAAACgwAAAsMEAAACAAACgwAAAwMEAAACAAACgwAAA0MEAAACAAACgwAAA4MEAAACAAACgwAAA8MEAAACAAACgwAAAAMIAAACAAACgwAAACMIAAACAAADAwAAAgMAAAACAAADAwAAAAMAAAACAAADAwAAAwMAAAACAAADAwAAAgMAAAACAAADAwAAAYMEAAACAAADIwAAAMMEAAACAAADIwAAAMMEAAACAAADAwAAAAMEAAACAAADAwAAAQMEAAACAAADAwAAAYMEAAACAAADgwAAAAMAAAACAAADgwAAAgMAAAACAAADgwAAAAMEAAACAAADgwAAAIMEAAACAAAAQwQAAAMAAAACAAAAUwQAAiMEAAIDBAAAUwQAAiMEAAIDBAAAQwQAAAMIAAADCAAAQwQAAgMAAAACAAAAQwQAAAMAAAACAAAAgwQAAAMAAAACAAAAkwQAAiMEAAIDBAAAkwQAAiMEAAIDBAAAgwQAAAMIAAADCAABAwQAAAMIAAADCAABEwQAAiMEAAIDBAABEwQAAiMEAAIDBAABAwQAAAMAAAACAAABAwQAAgMAAAACAAABAwQAAwMAAAACAAABAwQAAAMEAAACAAABAwQAAIMEAAACAAABAwQAAQMEAAACAAABAwQAAYMEAAACAAABwwQAAAMIAAADCAAB0wQAAiMEAAIDBAAB0wQAAiMEAAIDBAABwwQAAAMAAAACAAADAwQAAgMAAAACAAADAwQAAwMAAAACAAADAwQAAAMEAAACAAADAwQAAIMEAAACAAADAwQAAYMEAAACAAADAwQAAQMEAAACAAADAwQAAgMEAAACAAADAwQAAYMEAAACAAACAPwAAAMIAAADCAADQwQAAAMIAAADCAACAPwAAAMAAAACAAADQwQAAAMAAAACAAACAPwAAgMAAAACAAADQwQAAgMAAAACAAACAPwAAwMAAAACAAADQwQAAwMAAAACAAACAPwAAAMEAAACAAADQwQAAAMEAAACAAACAPwAAIMEAAACAAADQwQAAIMEAAACAAACAPwAAQMEAAACAAADQwQAAQMEAAACAAACAPwAAYMEAAACAAADQwQAAYMEAAACAAACAPwAAgMEAAACAAADQwQAAgMEAAACAAACAPwAAkMEAAACAAADQwQAAkMEAAACAAACAPwAAoMEAAACAAADQwQAAoMEAAACAAACAPwAAsMEAAACAAADQwQAAsMEAAACAAACAPwAAwMEAAACAAADQwQAAwMEAAACAAACAPwAA0MEAAACAAADQwQAA0MEAAACAAACAPwAA4MEAAACAAADQwQAA4MEAAACAAACAPwAA8MEAAACAAADQwQAA8MEAAACAAACAPwAAAMIAAACAAADQwQAAAMIAAACAAACAPwAACMIAAACAAADQwQAACMIAAACA"},{"byteLength":360,"name":"buf_red_scattered_lines","uri":"data:application/octet-stream;base64,AAAAAAAAAIAAAIDBAABAwAAAAIAAAIDBAAAgwAAAAL8AAIDBAABAwAAAAIAAAIDBAAAgwAAAAD8AAIDBAABAwAAAAIAAAIDBAABcwQAAgL8AAIA/AABcwQAAgL8AAATCAABcwQAAgL8AAIA/AABcwQAADMIAAIA/AABcwQAAgL8AAIA/AACKwQAAgL8AAIA/AABcwQAAgL8AAATCAABcwQAADMIAAATCAABcwQAAgL8AAATCAACKwQAAgL8AAATCAABcwQAADMIAAIA/AABcwQAADMIAAATCAABcwQAADMIAAIA/AACKwQAADMIAAIA/AABcwQAADMIAAATCAACKwQAADMIAAATCAACKwQAAgL8AAIA/AACKwQAAgL8AAATCAACKwQAAgL8AAIA/AACKwQAADMIAAIA/AACKwQAAgL8AAATCAACKwQAADMIAAATCAACKwQAADMIAAIA/AACKwQAADMIAAATC"},{"byteLength":1152,"name":"buf_blue_scattered_lines","uri":"data:application/octet-stream;base64,AABAv83MzL8AAAA/AABAv5qZmb8AAAA/AABAv83MzL8AAALCAABAv5qZmb8AAALCAABAv5qZmb8AAAA/AABAv5qZmb8AAALCAABAv5qZmb8AAAA/AAAQwJqZmb8AAAA/AABAv5qZmb8AAALCAAAQwJqZmb8AAALCAAAQwM3MzL8AAAA/AAAQwJqZmb8AAAA/AAAQwM3MzL8AAALCAAAQwJqZmb8AAALCAAAQwJqZmb8AAAA/AAAQwJqZmb8AAALCAAAwwM3MzL8AAAA/AAAwwJqZmb8AAAA/AAAwwM3MzL8AAALCAAAwwJqZmb8AAALCAAAwwJqZmb8AAAA/AAAwwJqZmb8AAALCAAAwwJqZmb8AAAA/AACowJqZmb8AAAA/AAAwwJqZmb8AAALCAACowJqZmb8AAALCAACowM3MzL8AAAA/AACowJqZmb8AAAA/AACowM3MzL8AAALCAACowJqZmb8AAALCAACowJqZmb8AAAA/AACowJqZmb8AAALCAAC4wM3MzL8AAAA/AAC4wJqZmb8AAAA/AAC4wM3MzL8AAALCAAC4wJqZmb8AAALCAAC4wJqZmb8AAAA/AAC4wJqZmb8AAALCAAC4wJqZmb8AAAA/AAAEwZqZmb8AAAA/AAC4wJqZmb8AAALCAAAEwZqZmb8AAALCAAAEwc3MzL8AAAA/AAAEwZqZmb8AAAA/AAAEwc3MzL8AAALCAAAEwZqZmb8AAALCAAAEwZqZmb8AAAA/AAAEwZqZmb8AAALCAAAMwc3MzL8AAAA/AAAMwZqZmb8AAAA/AAAMwc3MzL8AAALCAAAMwZqZmb8AAALCAAAMwZqZmb8AAAA/AAAMwZqZmb8AAALCAAAMwZqZmb8AAAA/AAA0wZqZmb8AAAA/AAAMwZqZmb8AAALCAAA0wZqZmb8AAALCAAA0wc3MzL8AAAA/AAA0wZqZmb8AAAA/AAA0wc3MzL8AAALCAAA0wZqZmb8AAALCAAA0wZqZmb8AAAA/AAA0wZqZmb8AAALCAABcwc3MzL8AAAA/AABcwZqZmb8AAAA/AABcwc3MzL8AAALCAABcwZqZmb8AAALCAABcwZqZmb8AAAA/AABcwZqZmb8AAALCAABcwZqZmb8AAAA/AACCwZqZmb8AAAA/AABcwZqZmb8AAALCAACCwZqZmb8AAALCAACCwc3MzL8AAAA/AACCwZqZmb8AAAA/AACCwc3MzL8AAALCAACCwZqZmb8AAALCAACCwZqZmb8AAAA/AACCwZqZmb8AAALCAACWwc3MzL8AAAA/AACWwZqZmb8AAAA/AACWwc3MzL8AAALCAACWwZqZmb8AAALCAACWwZqZmb8AAAA/AACWwZqZmb8AAALCAACWwZqZmb8AAAA/AAC6wZqZmb8AAAA/AACWwZqZmb8AAALCAAC6wZqZmb8AAALCAAC6wc3MzL8AAAA/AAC6wZqZmb8AAAA/AAC6wc3MzL8AAALCAAC6wZqZmb8AAALCAAC6wZqZmb8AAAA/AAC6wZqZmb8AAALC"}],"images":[{"name":"gates_image","uri":""}],"materials":[{"doubleSided":false,"name":"gates_material","pbrMetallicRoughness":{"baseColorFactor":[1,1,1,1],"baseColorTexture":{"index":0,"texCoord":0},"metallicFactor":0.4,"roughnessFactor":0.5}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"white","pbrMetallicRoughness":{"baseColorFactor":[1,1,1,1],"metallicFactor":0.4,"roughnessFactor":0.5}},{"doubleSided":true,"name":"white","pbrMetallicRoughness":{"baseColorFactor":[1,1,1,1],"metallicFactor":0.4,"roughnessFactor":0.5}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"white","pbrMetallicRoughness":{"baseColorFactor":[1,1,1,1],"metallicFactor":0.4,"roughnessFactor":0.5}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"gray","pbrMetallicRoughness":{"baseColorFactor":[0.5,0.5,0.5,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"red","pbrMetallicRoughness":{"baseColorFactor":[1,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"blue","pbrMetallicRoughness":{"baseColorFactor":[0,0,1,1],"metallicFactor":1,"roughnessFactor":1}}],"meshes":[{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":1},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":2},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":3},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":4},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":5},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":6},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":7},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":8},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":9},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":10},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":11},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":12},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":13},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":14},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":15},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":16},"material":1,"mode":6},{"attributes":{"POSITION":17},"material":2,"mode":1}]},{"primitives":[{"attributes":{"POSITION":18},"material":3,"mode":6},{"attributes":{"POSITION":18},"material":4,"mode":3},{"attributes":{"POSITION":19},"material":4,"mode":1}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":20},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":21},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":22},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":23},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":24},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":25},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":26},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":27},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":28},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":29},"material":5,"mode":6},{"attributes":{"POSITION":29},"material":6,"mode":3},{"attributes":{"POSITION":30},"material":6,"mode":1}]},{"primitives":[{"attributes":{"POSITION":31},"material":7,"mode":2},{"attributes":{"POSITION":31},"material":8,"mode":4}]},{"primitives":[{"attributes":{"POSITION":32},"material":9,"mode":6}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":33},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":34},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":35},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":36},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":37},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":38},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":39},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":40},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":41},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":42},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":43},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":44},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":45},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":46},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":47},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":48},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":49},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":50},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":51},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":52},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":53},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":54},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":55},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":56},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":57},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":58},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":59},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":60},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":61},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":62},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":63},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":64},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":65},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":66},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":67},"material":10,"mode":1}]},{"primitives":[{"attributes":{"POSITION":68},"material":11,"mode":1}]},{"primitives":[{"attributes":{"POSITION":69},"material":12,"mode":1}]}],"nodes":[{"mesh":0,"translation":[-0,-32,-32]},{"mesh":1,"translation":[-0,-2,-0]},{"mesh":2,"translation":[-0,-4,-0]},{"mesh":3,"translation":[-0,-6,-0]},{"mesh":4,"translation":[-1,-32,-32]},{"mesh":5,"translation":[-1,-2,-0]},{"mesh":6,"translation":[-1,-4,-0]},{"mesh":7,"translation":[-1,-6,-0]},{"mesh":8,"translation":[-1,-8,-0]},{"mesh":9,"translation":[-2,-32,-32]},{"mesh":10,"translation":[-2,-2,-0]},{"mesh":11,"translation":[-2,-4,-0]},{"mesh":12,"translation":[-2,-6,-0]},{"mesh":13,"translation":[-2,-8,-0]},{"mesh":14,"translation":[-2,-10,-0]},{"mesh":15,"translation":[-3,-32,-32]},{"mesh":16,"translation":[-3,-2,-0]},{"mesh":17,"translation":[-3,-4,-0]},{"mesh":17,"translation":[-3,-6,-0]},{"mesh":18,"translation":[-3,-8,-0]},{"mesh":18,"translation":[-3,-10,-0]},{"mesh":19,"translation":[-3,-12,-0]},{"mesh":19,"translation":[-3,-14,-0]},{"mesh":16,"translation":[-3,-16,-0]},{"mesh":15,"translation":[-3,-18,-0]},{"mesh":20,"translation":[-4,-32,-32]},{"mesh":20,"translation":[-4,-2,-0]},{"mesh":21,"translation":[-4,-4,-0]},{"mesh":21,"translation":[-4,-6,-0]},{"mesh":22,"translation":[-4,-8,-0]},{"mesh":22,"translation":[-4,-10,-0]},{"mesh":23,"translation":[-4,-12,-0]},{"mesh":23,"translation":[-4,-14,-0]},{"mesh":24,"translation":[-4,-16,-0]},{"mesh":24,"translation":[-4,-18,-0]},{"mesh":25,"translation":[-4,-20,-0]},{"mesh":25,"translation":[-4,-22,-0]},{"mesh":26,"translation":[-5,-32,-32]},{"mesh":26,"translation":[-5,-2,-0]},{"mesh":26,"translation":[-5,-4,-0]},{"mesh":27,"translation":[-5,-6,-0]},{"mesh":26,"translation":[-5,-8,-0]},{"mesh":28,"translation":[-5,-10,-0]},{"mesh":27,"translation":[-5,-12,-0]},{"mesh":26,"translation":[-5,-14,-0]},{"mesh":27,"translation":[-5,-16,-0]},{"mesh":27,"translation":[-5,-18,-0]},{"mesh":27,"translation":[-5,-20,-0]},{"mesh":28,"translation":[-5,-22,-0]},{"mesh":28,"translation":[-5,-24,-0]},{"mesh":26,"translation":[-5,-26,-0]},{"mesh":28,"translation":[-5,-28,-0]},{"mesh":27,"translation":[-5,-30,-0]},{"mesh":28,"translation":[-5,-32,-0]},{"mesh":28,"translation":[-5,-34,-0]},{"mesh":29,"translation":[-6,-2,-0]},{"mesh":30,"translation":[-6,-4,-0]},{"mesh":31,"translation":[-6,-6,-0]},{"mesh":32,"translation":[-6,-8,-0]},{"mesh":33,"translation":[-6,-14,-0]},{"mesh":34,"translation":[-6,-12,-0]},{"mesh":35,"translation":[-6,-32,-32]},{"mesh":36,"translation":[-7,-2,-0]},{"mesh":36,"translation":[-7,-4,-0]},{"mesh":37,"translation":[-7,-6,-0]},{"mesh":38,"translation":[-7,-8,-0]},{"mesh":38,"translation":[-7,-10,-0]},{"mesh":39,"translation":[-7,-32,-32]},{"mesh":40,"translation":[-8,-2,-0]},{"mesh":41,"translation":[-8,-4,-0]},{"mesh":42,"translation":[-8,-6,-0]},{"mesh":43,"translation":[-8,-12,-0]},{"mesh":44,"translation":[-9,-32,-32]},{"mesh":45,"translation":[-9,-2,-0]},{"mesh":46,"translation":[-9,-4,-0]},{"mesh":46,"translation":[-10,-32,-32]},{"mesh":46,"translation":[-10,-2,-0]},{"mesh":47,"translation":[-11,-32,-32]},{"mesh":48,"translation":[-11,-2,-0]},{"mesh":49,"translation":[-11,-4,-0]},{"mesh":50,"translation":[-11,-6,-0]},{"mesh":51,"translation":[-11,-8,-0]},{"mesh":52,"translation":[-11,-10,-0]},{"mesh":52,"translation":[-11,-12,-0]},{"mesh":53,"translation":[-11,-14,-0]},{"mesh":54,"translation":[-11,-16,-0]},{"mesh":55,"translation":[-11,-18,-0]},{"mesh":56,"translation":[-12,-32,-32]},{"mesh":56,"translation":[-12,-2,-0]},{"mesh":56,"translation":[-12,-4,-0]},{"mesh":56,"translation":[-12,-6,-0]},{"mesh":57,"translation":[-12,-8,-0]},{"mesh":57,"translation":[-12,-10,-0]},{"mesh":58,"translation":[-12,-12,-0]},{"mesh":58,"translation":[-12,-14,-0]},{"mesh":7,"translation":[-14,-32,-32]},{"mesh":28,"translation":[-15,-32,-32]},{"mesh":26,"translation":[-15,-2,-0]},{"mesh":13,"translation":[-16,-2,-0]},{"mesh":49,"translation":[-19,-32,-32]},{"mesh":39,"translation":[-20,-32,-32]},{"mesh":49,"translation":[-21,-32,-32]},{"mesh":59,"translation":[-22,-32,-32]},{"mesh":59,"translation":[-22,-2,-0]},{"mesh":59,"translation":[-23,-32,-32]},{"mesh":47,"translation":[-24,-32,-32]},{"mesh":51,"translation":[-24,-2,-0]},{"mesh":58,"translation":[-24,-4,-0]},{"mesh":58,"translation":[-24,-6,-0]},{"mesh":57,"translation":[-24,-8,-0]},{"mesh":57,"translation":[-24,-10,-0]},{"mesh":44,"translation":[-24,-12,-0]},{"mesh":45,"translation":[-24,-14,-0]},{"mesh":46,"translation":[-24,-16,-0]},{"mesh":60,"translation":[-25,-32,-32]},{"mesh":61,"translation":[-25,-2,-0]},{"mesh":62,"translation":[-25,-4,-0]},{"mesh":63,"translation":[0,0,0]},{"mesh":64,"translation":[0,0,0]},{"mesh":65,"translation":[0,0,0]}],"samplers":[{"magFilter":9728,"minFilter":9728,"wrapS":33071,"wrapT":33071}],"scene":0,"scenes":[{"nodes":[0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,118,119]}],"textures":[{"name":"gates_texture","sampler":0,"source":0}]} \ No newline at end of file +{"accessors":[{"bufferView":0,"byteOffset":0,"componentType":5126,"count":12,"max":[0,0.5,0.5],"min":[0,-0.5,-0.5],"name":"cube","type":"VEC3"},{"bufferView":1,"byteOffset":0,"componentType":5126,"count":12,"max":[0.0625,0.4375],"min":[0,0.375],"name":"tex_coords_gate_I","type":"VEC2"},{"bufferView":2,"byteOffset":0,"componentType":5126,"count":12,"max":[0.0625,0.4375],"min":[0,0.375],"name":"tex_coords_gate_X","type":"VEC2"},{"bufferView":3,"byteOffset":0,"componentType":5126,"count":12,"max":[0.0625,0.5],"min":[0,0.4375],"name":"tex_coords_gate_Y","type":"VEC2"},{"bufferView":4,"byteOffset":0,"componentType":5126,"count":12,"max":[0.0625,0.5625],"min":[0,0.5],"name":"tex_coords_gate_Z","type":"VEC2"},{"bufferView":5,"byteOffset":0,"componentType":5126,"count":12,"max":[0.125,0.625],"min":[0.0625,0.5625],"name":"tex_coords_gate_C_XYZ","type":"VEC2"},{"bufferView":6,"byteOffset":0,"componentType":5126,"count":12,"max":[0.1875,0.625],"min":[0.125,0.5625],"name":"tex_coords_gate_C_ZYX","type":"VEC2"},{"bufferView":7,"byteOffset":0,"componentType":5126,"count":12,"max":[0.125,0.5625],"min":[0.0625,0.5],"name":"tex_coords_gate_H_XY","type":"VEC2"},{"bufferView":8,"byteOffset":0,"componentType":5126,"count":12,"max":[0.125,0.5],"min":[0.0625,0.4375],"name":"tex_coords_gate_H","type":"VEC2"},{"bufferView":9,"byteOffset":0,"componentType":5126,"count":12,"max":[0.125,0.4375],"min":[0.0625,0.375],"name":"tex_coords_gate_H_YZ","type":"VEC2"},{"bufferView":10,"byteOffset":0,"componentType":5126,"count":12,"max":[0.1875,0.4375],"min":[0.125,0.375],"name":"tex_coords_gate_SQRT_X","type":"VEC2"},{"bufferView":11,"byteOffset":0,"componentType":5126,"count":12,"max":[0.25,0.4375],"min":[0.1875,0.375],"name":"tex_coords_gate_SQRT_X_DAG","type":"VEC2"},{"bufferView":12,"byteOffset":0,"componentType":5126,"count":12,"max":[0.1875,0.5],"min":[0.125,0.4375],"name":"tex_coords_gate_SQRT_Y","type":"VEC2"},{"bufferView":13,"byteOffset":0,"componentType":5126,"count":12,"max":[0.25,0.5],"min":[0.1875,0.4375],"name":"tex_coords_gate_SQRT_Y_DAG","type":"VEC2"},{"bufferView":14,"byteOffset":0,"componentType":5126,"count":12,"max":[0.1875,0.5625],"min":[0.125,0.5],"name":"tex_coords_gate_S","type":"VEC2"},{"bufferView":15,"byteOffset":0,"componentType":5126,"count":12,"max":[0.25,0.5625],"min":[0.1875,0.5],"name":"tex_coords_gate_S_DAG","type":"VEC2"},{"bufferView":16,"byteOffset":0,"componentType":5126,"count":17,"max":[0,0.400000005960464,0.400000005960464],"min":[0,-0.400000005960464,-0.400000005960464],"name":"circle_loop","type":"VEC3"},{"bufferView":17,"byteOffset":0,"componentType":5126,"count":4,"max":[0,0.45254835486412,0.45254835486412],"min":[0,-0.45254835486412,-0.45254835486412],"name":"control_zswap_line_cross","type":"VEC3"},{"bufferView":18,"byteOffset":0,"componentType":5126,"count":17,"max":[0,0.400000005960464,0.400000005960464],"min":[0,-0.400000005960464,-0.400000005960464],"name":"circle_loop","type":"VEC3"},{"bufferView":19,"byteOffset":0,"componentType":5126,"count":4,"max":[0,0.45254835486412,0.45254835486412],"min":[0,-0.45254835486412,-0.45254835486412],"name":"control_xswap_line_cross","type":"VEC3"},{"bufferView":20,"byteOffset":0,"componentType":5126,"count":12,"max":[0.375,0.625],"min":[0.3125,0.5625],"name":"tex_coords_gate_ISWAP","type":"VEC2"},{"bufferView":21,"byteOffset":0,"componentType":5126,"count":12,"max":[0.4375,0.625],"min":[0.375,0.5625],"name":"tex_coords_gate_ISWAP_DAG","type":"VEC2"},{"bufferView":22,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5,0.625],"min":[0.4375,0.5625],"name":"tex_coords_gate_SWAP","type":"VEC2"},{"bufferView":23,"byteOffset":0,"componentType":5126,"count":12,"max":[0.75,0.4375],"min":[0.6875,0.375],"name":"tex_coords_gate_SQRT_XX","type":"VEC2"},{"bufferView":24,"byteOffset":0,"componentType":5126,"count":12,"max":[0.8125,0.4375],"min":[0.75,0.375],"name":"tex_coords_gate_SQRT_XX_DAG","type":"VEC2"},{"bufferView":25,"byteOffset":0,"componentType":5126,"count":12,"max":[0.75,0.5],"min":[0.6875,0.4375],"name":"tex_coords_gate_SQRT_YY","type":"VEC2"},{"bufferView":26,"byteOffset":0,"componentType":5126,"count":12,"max":[0.8125,0.5],"min":[0.75,0.4375],"name":"tex_coords_gate_SQRT_YY_DAG","type":"VEC2"},{"bufferView":27,"byteOffset":0,"componentType":5126,"count":12,"max":[0.75,0.5625],"min":[0.6875,0.5],"name":"tex_coords_gate_SQRT_ZZ","type":"VEC2"},{"bufferView":28,"byteOffset":0,"componentType":5126,"count":12,"max":[0.8125,0.5625],"min":[0.75,0.5],"name":"tex_coords_gate_SQRT_ZZ_DAG","type":"VEC2"},{"bufferView":29,"byteOffset":0,"componentType":5126,"count":17,"max":[0,0.400000005960464,0.400000005960464],"min":[0,-0.400000005960464,-0.400000005960464],"name":"circle_loop","type":"VEC3"},{"bufferView":30,"byteOffset":0,"componentType":5126,"count":4,"max":[0,0.400000005960464,0.400000005960464],"min":[0,-0.400000005960464,-0.400000005960464],"name":"control_x_line_cross","type":"VEC3"},{"bufferView":31,"byteOffset":0,"componentType":5126,"count":3,"max":[0,0.400000005960464,0.346410155296326],"min":[0,-0.200000032782555,-0.346410185098648],"name":"circle_loop","type":"VEC3"},{"bufferView":32,"byteOffset":0,"componentType":5126,"count":17,"max":[0,0.400000005960464,0.400000005960464],"min":[0,-0.400000005960464,-0.400000005960464],"name":"circle_loop","type":"VEC3"},{"bufferView":33,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5625,0.4375],"min":[0.5,0.375],"name":"tex_coords_gate_E:X","type":"VEC2"},{"bufferView":34,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5625,0.5],"min":[0.5,0.4375],"name":"tex_coords_gate_E:Y","type":"VEC2"},{"bufferView":35,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5625,0.5625],"min":[0.5,0.5],"name":"tex_coords_gate_E:Z","type":"VEC2"},{"bufferView":36,"byteOffset":0,"componentType":5126,"count":12,"max":[0.625,0.4375],"min":[0.5625,0.375],"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:X","type":"VEC2"},{"bufferView":37,"byteOffset":0,"componentType":5126,"count":12,"max":[0.625,0.5],"min":[0.5625,0.4375],"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Y","type":"VEC2"},{"bufferView":38,"byteOffset":0,"componentType":5126,"count":12,"max":[0.625,0.5625],"min":[0.5625,0.5],"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Z","type":"VEC2"},{"bufferView":39,"byteOffset":0,"componentType":5126,"count":12,"max":[0.25,0.625],"min":[0.1875,0.5625],"name":"tex_coords_gate_DEPOLARIZE1","type":"VEC2"},{"bufferView":40,"byteOffset":0,"componentType":5126,"count":12,"max":[0.3125,0.625],"min":[0.25,0.5625],"name":"tex_coords_gate_DEPOLARIZE2","type":"VEC2"},{"bufferView":41,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5625,0.625],"min":[0.5,0.5625],"name":"tex_coords_gate_PAULI_CHANNEL_1","type":"VEC2"},{"bufferView":42,"byteOffset":0,"componentType":5126,"count":12,"max":[0.625,0.625],"min":[0.5625,0.5625],"name":"tex_coords_gate_PAULI_CHANNEL_2","type":"VEC2"},{"bufferView":43,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5,0.4375],"min":[0.4375,0.375],"name":"tex_coords_gate_X_ERROR","type":"VEC2"},{"bufferView":44,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5,0.5],"min":[0.4375,0.4375],"name":"tex_coords_gate_Y_ERROR","type":"VEC2"},{"bufferView":45,"byteOffset":0,"componentType":5126,"count":12,"max":[0.5,0.5625],"min":[0.4375,0.5],"name":"tex_coords_gate_Z_ERROR","type":"VEC2"},{"bufferView":46,"byteOffset":0,"componentType":5126,"count":12,"max":[0.9375,0.625],"min":[0.875,0.5625],"name":"tex_coords_gate_HERALDED_ERASE","type":"VEC2"},{"bufferView":47,"byteOffset":0,"componentType":5126,"count":12,"max":[1,0.625],"min":[0.9375,0.5625],"name":"tex_coords_gate_HERALDED_PAULI_CHANNEL_1","type":"VEC2"},{"bufferView":48,"byteOffset":0,"componentType":5126,"count":12,"max":[0.6875,0.4375],"min":[0.625,0.375],"name":"tex_coords_gate_MPP:X","type":"VEC2"},{"bufferView":49,"byteOffset":0,"componentType":5126,"count":12,"max":[0.6875,0.5],"min":[0.625,0.4375],"name":"tex_coords_gate_MPP:Y","type":"VEC2"},{"bufferView":50,"byteOffset":0,"componentType":5126,"count":12,"max":[0.6875,0.5625],"min":[0.625,0.5],"name":"tex_coords_gate_MPP:Z","type":"VEC2"},{"bufferView":51,"byteOffset":0,"componentType":5126,"count":12,"max":[0.4375,0.4375],"min":[0.375,0.375],"name":"tex_coords_gate_MRX","type":"VEC2"},{"bufferView":52,"byteOffset":0,"componentType":5126,"count":12,"max":[0.4375,0.5],"min":[0.375,0.4375],"name":"tex_coords_gate_MRY","type":"VEC2"},{"bufferView":53,"byteOffset":0,"componentType":5126,"count":12,"max":[0.4375,0.5625],"min":[0.375,0.5],"name":"tex_coords_gate_MR","type":"VEC2"},{"bufferView":54,"byteOffset":0,"componentType":5126,"count":12,"max":[0.3125,0.4375],"min":[0.25,0.375],"name":"tex_coords_gate_MX","type":"VEC2"},{"bufferView":55,"byteOffset":0,"componentType":5126,"count":12,"max":[0.3125,0.5],"min":[0.25,0.4375],"name":"tex_coords_gate_MY","type":"VEC2"},{"bufferView":56,"byteOffset":0,"componentType":5126,"count":12,"max":[0.3125,0.5625],"min":[0.25,0.5],"name":"tex_coords_gate_M","type":"VEC2"},{"bufferView":57,"byteOffset":0,"componentType":5126,"count":12,"max":[0.375,0.4375],"min":[0.3125,0.375],"name":"tex_coords_gate_RX","type":"VEC2"},{"bufferView":58,"byteOffset":0,"componentType":5126,"count":12,"max":[0.375,0.5],"min":[0.3125,0.4375],"name":"tex_coords_gate_RY","type":"VEC2"},{"bufferView":59,"byteOffset":0,"componentType":5126,"count":12,"max":[0.375,0.5625],"min":[0.3125,0.5],"name":"tex_coords_gate_R","type":"VEC2"},{"bufferView":60,"byteOffset":0,"componentType":5126,"count":12,"max":[0.6875,0.625],"min":[0.625,0.5625],"name":"tex_coords_gate_MXX","type":"VEC2"},{"bufferView":61,"byteOffset":0,"componentType":5126,"count":12,"max":[0.75,0.625],"min":[0.6875,0.5625],"name":"tex_coords_gate_MYY","type":"VEC2"},{"bufferView":62,"byteOffset":0,"componentType":5126,"count":12,"max":[0.8125,0.625],"min":[0.75,0.5625],"name":"tex_coords_gate_MZZ","type":"VEC2"},{"bufferView":63,"byteOffset":0,"componentType":5126,"count":12,"max":[0.875,0.625],"min":[0.8125,0.5625],"name":"tex_coords_gate_MPAD","type":"VEC2"},{"bufferView":64,"byteOffset":0,"componentType":5126,"count":12,"max":[0.875,0.4375],"min":[0.8125,0.375],"name":"tex_coords_gate_X:REC","type":"VEC2"},{"bufferView":65,"byteOffset":0,"componentType":5126,"count":12,"max":[0.9375,0.5],"min":[0.875,0.4375],"name":"tex_coords_gate_Y:SWEEP","type":"VEC2"},{"bufferView":66,"byteOffset":0,"componentType":5126,"count":12,"max":[0.875,0.5625],"min":[0.8125,0.5],"name":"tex_coords_gate_Z:REC","type":"VEC2"},{"bufferView":67,"byteOffset":0,"componentType":5126,"count":130,"max":[1,-2,-0],"min":[-26,-34,-32],"name":"buf_scattered_lines","type":"VEC3"},{"bufferView":68,"byteOffset":0,"componentType":5126,"count":30,"max":[0,0.5,1],"min":[-17.25,-35,-33],"name":"buf_red_scattered_lines","type":"VEC3"},{"bufferView":69,"byteOffset":0,"componentType":5126,"count":96,"max":[-0.75,-1.20000004768372,0.5],"min":[-23.25,-1.60000002384186,-32.5],"name":"buf_blue_scattered_lines","type":"VEC3"}],"asset":{"version":"2.0"},"bufferViews":[{"buffer":0,"byteLength":144,"byteOffset":0,"name":"cube","target":34962},{"buffer":1,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_I","target":34962},{"buffer":2,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_X","target":34962},{"buffer":3,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Y","target":34962},{"buffer":4,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Z","target":34962},{"buffer":5,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_C_XYZ","target":34962},{"buffer":6,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_C_ZYX","target":34962},{"buffer":7,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_H_XY","target":34962},{"buffer":8,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_H","target":34962},{"buffer":9,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_H_YZ","target":34962},{"buffer":10,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_X","target":34962},{"buffer":11,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_X_DAG","target":34962},{"buffer":12,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_Y","target":34962},{"buffer":13,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_Y_DAG","target":34962},{"buffer":14,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_S","target":34962},{"buffer":15,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_S_DAG","target":34962},{"buffer":16,"byteLength":204,"byteOffset":0,"name":"circle_loop","target":34962},{"buffer":17,"byteLength":48,"byteOffset":0,"name":"control_zswap_line_cross","target":34962},{"buffer":18,"byteLength":204,"byteOffset":0,"name":"circle_loop","target":34962},{"buffer":19,"byteLength":48,"byteOffset":0,"name":"control_xswap_line_cross","target":34962},{"buffer":20,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_ISWAP","target":34962},{"buffer":21,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_ISWAP_DAG","target":34962},{"buffer":22,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SWAP","target":34962},{"buffer":23,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_XX","target":34962},{"buffer":24,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_XX_DAG","target":34962},{"buffer":25,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_YY","target":34962},{"buffer":26,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_YY_DAG","target":34962},{"buffer":27,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_ZZ","target":34962},{"buffer":28,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_SQRT_ZZ_DAG","target":34962},{"buffer":29,"byteLength":204,"byteOffset":0,"name":"circle_loop","target":34962},{"buffer":30,"byteLength":48,"byteOffset":0,"name":"control_x_line_cross","target":34962},{"buffer":31,"byteLength":36,"byteOffset":0,"name":"circle_loop","target":34962},{"buffer":32,"byteLength":204,"byteOffset":0,"name":"circle_loop","target":34962},{"buffer":33,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_E:X","target":34962},{"buffer":34,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_E:Y","target":34962},{"buffer":35,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_E:Z","target":34962},{"buffer":36,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:X","target":34962},{"buffer":37,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Y","target":34962},{"buffer":38,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Z","target":34962},{"buffer":39,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_DEPOLARIZE1","target":34962},{"buffer":40,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_DEPOLARIZE2","target":34962},{"buffer":41,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_PAULI_CHANNEL_1","target":34962},{"buffer":42,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_PAULI_CHANNEL_2","target":34962},{"buffer":43,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_X_ERROR","target":34962},{"buffer":44,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Y_ERROR","target":34962},{"buffer":45,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Z_ERROR","target":34962},{"buffer":46,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_HERALDED_ERASE","target":34962},{"buffer":47,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_HERALDED_PAULI_CHANNEL_1","target":34962},{"buffer":48,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MPP:X","target":34962},{"buffer":49,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MPP:Y","target":34962},{"buffer":50,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MPP:Z","target":34962},{"buffer":51,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MRX","target":34962},{"buffer":52,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MRY","target":34962},{"buffer":53,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MR","target":34962},{"buffer":54,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MX","target":34962},{"buffer":55,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MY","target":34962},{"buffer":56,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_M","target":34962},{"buffer":57,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_RX","target":34962},{"buffer":58,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_RY","target":34962},{"buffer":59,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_R","target":34962},{"buffer":60,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MXX","target":34962},{"buffer":61,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MYY","target":34962},{"buffer":62,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MZZ","target":34962},{"buffer":63,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_MPAD","target":34962},{"buffer":64,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_X:REC","target":34962},{"buffer":65,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Y:SWEEP","target":34962},{"buffer":66,"byteLength":96,"byteOffset":0,"name":"tex_coords_gate_Z:REC","target":34962},{"buffer":67,"byteLength":1560,"byteOffset":0,"name":"buf_scattered_lines","target":34962},{"buffer":68,"byteLength":360,"byteOffset":0,"name":"buf_red_scattered_lines","target":34962},{"buffer":69,"byteLength":1152,"byteOffset":0,"name":"buf_blue_scattered_lines","target":34962}],"buffers":[{"byteLength":144,"name":"cube","uri":"data:application/octet-stream;base64,AAAAAAAAAD8AAAA/AAAAAAAAAD8AAAC/AAAAAAAAAL8AAAA/AAAAAAAAAD8AAAC/AAAAAAAAAL8AAAC/AAAAAAAAAL8AAAA/AAAAAAAAAL8AAAC/AAAAAAAAAD8AAAC/AAAAAAAAAL8AAAA/AAAAAAAAAL8AAAA/AAAAAAAAAD8AAAC/AAAAAAAAAD8AAAA/"},{"byteLength":96,"name":"tex_coords_gate_I","uri":"data:application/octet-stream;base64,AACAPQAAwD4AAAAAAADAPgAAgD0AAOA+AAAAAAAAwD4AAAAAAADgPgAAgD0AAOA+AACAPQAA4D4AAIA9AADAPgAAAAAAAOA+AAAAAAAA4D4AAIA9AADAPgAAAAAAAMA+"},{"byteLength":96,"name":"tex_coords_gate_X","uri":"data:application/octet-stream;base64,AACAPQAAwD4AAAAAAADAPgAAgD0AAOA+AAAAAAAAwD4AAAAAAADgPgAAgD0AAOA+AACAPQAA4D4AAIA9AADAPgAAAAAAAOA+AAAAAAAA4D4AAIA9AADAPgAAAAAAAMA+"},{"byteLength":96,"name":"tex_coords_gate_Y","uri":"data:application/octet-stream;base64,AACAPQAA4D4AAAAAAADgPgAAgD0AAAA/AAAAAAAA4D4AAAAAAAAAPwAAgD0AAAA/AACAPQAAAD8AAIA9AADgPgAAAAAAAAA/AAAAAAAAAD8AAIA9AADgPgAAAAAAAOA+"},{"byteLength":96,"name":"tex_coords_gate_Z","uri":"data:application/octet-stream;base64,AACAPQAAAD8AAAAAAAAAPwAAgD0AABA/AAAAAAAAAD8AAAAAAAAQPwAAgD0AABA/AACAPQAAED8AAIA9AAAAPwAAAAAAABA/AAAAAAAAED8AAIA9AAAAPwAAAAAAAAA/"},{"byteLength":96,"name":"tex_coords_gate_C_XYZ","uri":"data:application/octet-stream;base64,AAAAPgAAED8AAIA9AAAQPwAAAD4AACA/AACAPQAAED8AAIA9AAAgPwAAAD4AACA/AAAAPgAAID8AAAA+AAAQPwAAgD0AACA/AACAPQAAID8AAAA+AAAQPwAAgD0AABA/"},{"byteLength":96,"name":"tex_coords_gate_C_ZYX","uri":"data:application/octet-stream;base64,AABAPgAAED8AAAA+AAAQPwAAQD4AACA/AAAAPgAAED8AAAA+AAAgPwAAQD4AACA/AABAPgAAID8AAEA+AAAQPwAAAD4AACA/AAAAPgAAID8AAEA+AAAQPwAAAD4AABA/"},{"byteLength":96,"name":"tex_coords_gate_H_XY","uri":"data:application/octet-stream;base64,AAAAPgAAAD8AAIA9AAAAPwAAAD4AABA/AACAPQAAAD8AAIA9AAAQPwAAAD4AABA/AAAAPgAAED8AAAA+AAAAPwAAgD0AABA/AACAPQAAED8AAAA+AAAAPwAAgD0AAAA/"},{"byteLength":96,"name":"tex_coords_gate_H","uri":"data:application/octet-stream;base64,AAAAPgAA4D4AAIA9AADgPgAAAD4AAAA/AACAPQAA4D4AAIA9AAAAPwAAAD4AAAA/AAAAPgAAAD8AAAA+AADgPgAAgD0AAAA/AACAPQAAAD8AAAA+AADgPgAAgD0AAOA+"},{"byteLength":96,"name":"tex_coords_gate_H_YZ","uri":"data:application/octet-stream;base64,AAAAPgAAwD4AAIA9AADAPgAAAD4AAOA+AACAPQAAwD4AAIA9AADgPgAAAD4AAOA+AAAAPgAA4D4AAAA+AADAPgAAgD0AAOA+AACAPQAA4D4AAAA+AADAPgAAgD0AAMA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_X","uri":"data:application/octet-stream;base64,AABAPgAAwD4AAAA+AADAPgAAQD4AAOA+AAAAPgAAwD4AAAA+AADgPgAAQD4AAOA+AABAPgAA4D4AAEA+AADAPgAAAD4AAOA+AAAAPgAA4D4AAEA+AADAPgAAAD4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_X_DAG","uri":"data:application/octet-stream;base64,AACAPgAAwD4AAEA+AADAPgAAgD4AAOA+AABAPgAAwD4AAEA+AADgPgAAgD4AAOA+AACAPgAA4D4AAIA+AADAPgAAQD4AAOA+AABAPgAA4D4AAIA+AADAPgAAQD4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_Y","uri":"data:application/octet-stream;base64,AABAPgAA4D4AAAA+AADgPgAAQD4AAAA/AAAAPgAA4D4AAAA+AAAAPwAAQD4AAAA/AABAPgAAAD8AAEA+AADgPgAAAD4AAAA/AAAAPgAAAD8AAEA+AADgPgAAAD4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_Y_DAG","uri":"data:application/octet-stream;base64,AACAPgAA4D4AAEA+AADgPgAAgD4AAAA/AABAPgAA4D4AAEA+AAAAPwAAgD4AAAA/AACAPgAAAD8AAIA+AADgPgAAQD4AAAA/AABAPgAAAD8AAIA+AADgPgAAQD4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_S","uri":"data:application/octet-stream;base64,AABAPgAAAD8AAAA+AAAAPwAAQD4AABA/AAAAPgAAAD8AAAA+AAAQPwAAQD4AABA/AABAPgAAED8AAEA+AAAAPwAAAD4AABA/AAAAPgAAED8AAEA+AAAAPwAAAD4AAAA/"},{"byteLength":96,"name":"tex_coords_gate_S_DAG","uri":"data:application/octet-stream;base64,AACAPgAAAD8AAEA+AAAAPwAAgD4AABA/AABAPgAAAD8AAEA+AAAQPwAAgD4AABA/AACAPgAAED8AAIA+AAAAPwAAQD4AABA/AABAPgAAED8AAIA+AAAAPwAAQD4AAAA/"},{"byteLength":204,"name":"circle_loop","uri":"data:application/octet-stream;base64,AAAAAM3MzD4AAAAAAAAAAOU1vT5Fvxw+AAAAAMPQkD7D0JA+AAAAAES/HD7lNb0+AAAAAPIwlrLNzMw+AAAAAEe/HL7lNb0+AAAAAMPQkL7D0JA+AAAAAOc1vb5Avxw+AAAAAM3MzL7yMBazAAAAAOU1vb5Evxy+AAAAAMHQkL7E0JC+AAAAADy/HL7nNb2+AAAAAPLkozHNzMy+AAAAAEm/HD7kNb2+AAAAAMbQkD6/0JC+AAAAAOY1vT5Evxy+AAAAAM3MzD4AAAAA"},{"byteLength":48,"name":"control_zswap_line_cross","uri":"data:application/octet-stream;base64,AAAAAGu0575rtOe+AAAAAGu05z5rtOc+AAAAAGu0575rtOc+AAAAAGu05z5rtOe+"},{"byteLength":204,"name":"circle_loop","uri":"data:application/octet-stream;base64,AAAAAM3MzD4AAAAAAAAAAOU1vT5Fvxw+AAAAAMPQkD7D0JA+AAAAAES/HD7lNb0+AAAAAPIwlrLNzMw+AAAAAEe/HL7lNb0+AAAAAMPQkL7D0JA+AAAAAOc1vb5Avxw+AAAAAM3MzL7yMBazAAAAAOU1vb5Evxy+AAAAAMHQkL7E0JC+AAAAADy/HL7nNb2+AAAAAPLkozHNzMy+AAAAAEm/HD7kNb2+AAAAAMbQkD6/0JC+AAAAAOY1vT5Evxy+AAAAAM3MzD4AAAAA"},{"byteLength":48,"name":"control_xswap_line_cross","uri":"data:application/octet-stream;base64,AAAAAGu0575rtOe+AAAAAGu05z5rtOc+AAAAAGu0575rtOc+AAAAAGu05z5rtOe+"},{"byteLength":96,"name":"tex_coords_gate_ISWAP","uri":"data:application/octet-stream;base64,AADAPgAAED8AAKA+AAAQPwAAwD4AACA/AACgPgAAED8AAKA+AAAgPwAAwD4AACA/AADAPgAAID8AAMA+AAAQPwAAoD4AACA/AACgPgAAID8AAMA+AAAQPwAAoD4AABA/"},{"byteLength":96,"name":"tex_coords_gate_ISWAP_DAG","uri":"data:application/octet-stream;base64,AADgPgAAED8AAMA+AAAQPwAA4D4AACA/AADAPgAAED8AAMA+AAAgPwAA4D4AACA/AADgPgAAID8AAOA+AAAQPwAAwD4AACA/AADAPgAAID8AAOA+AAAQPwAAwD4AABA/"},{"byteLength":96,"name":"tex_coords_gate_SWAP","uri":"data:application/octet-stream;base64,AAAAPwAAED8AAOA+AAAQPwAAAD8AACA/AADgPgAAED8AAOA+AAAgPwAAAD8AACA/AAAAPwAAID8AAAA/AAAQPwAA4D4AACA/AADgPgAAID8AAAA/AAAQPwAA4D4AABA/"},{"byteLength":96,"name":"tex_coords_gate_SQRT_XX","uri":"data:application/octet-stream;base64,AABAPwAAwD4AADA/AADAPgAAQD8AAOA+AAAwPwAAwD4AADA/AADgPgAAQD8AAOA+AABAPwAA4D4AAEA/AADAPgAAMD8AAOA+AAAwPwAA4D4AAEA/AADAPgAAMD8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_XX_DAG","uri":"data:application/octet-stream;base64,AABQPwAAwD4AAEA/AADAPgAAUD8AAOA+AABAPwAAwD4AAEA/AADgPgAAUD8AAOA+AABQPwAA4D4AAFA/AADAPgAAQD8AAOA+AABAPwAA4D4AAFA/AADAPgAAQD8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_YY","uri":"data:application/octet-stream;base64,AABAPwAA4D4AADA/AADgPgAAQD8AAAA/AAAwPwAA4D4AADA/AAAAPwAAQD8AAAA/AABAPwAAAD8AAEA/AADgPgAAMD8AAAA/AAAwPwAAAD8AAEA/AADgPgAAMD8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_YY_DAG","uri":"data:application/octet-stream;base64,AABQPwAA4D4AAEA/AADgPgAAUD8AAAA/AABAPwAA4D4AAEA/AAAAPwAAUD8AAAA/AABQPwAAAD8AAFA/AADgPgAAQD8AAAA/AABAPwAAAD8AAFA/AADgPgAAQD8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_SQRT_ZZ","uri":"data:application/octet-stream;base64,AABAPwAAAD8AADA/AAAAPwAAQD8AABA/AAAwPwAAAD8AADA/AAAQPwAAQD8AABA/AABAPwAAED8AAEA/AAAAPwAAMD8AABA/AAAwPwAAED8AAEA/AAAAPwAAMD8AAAA/"},{"byteLength":96,"name":"tex_coords_gate_SQRT_ZZ_DAG","uri":"data:application/octet-stream;base64,AABQPwAAAD8AAEA/AAAAPwAAUD8AABA/AABAPwAAAD8AAEA/AAAQPwAAUD8AABA/AABQPwAAED8AAFA/AAAAPwAAQD8AABA/AABAPwAAED8AAFA/AAAAPwAAQD8AAAA/"},{"byteLength":204,"name":"circle_loop","uri":"data:application/octet-stream;base64,AAAAAM3MzD4AAAAAAAAAAOU1vT5Fvxw+AAAAAMPQkD7D0JA+AAAAAES/HD7lNb0+AAAAAPIwlrLNzMw+AAAAAEe/HL7lNb0+AAAAAMPQkL7D0JA+AAAAAOc1vb5Avxw+AAAAAM3MzL7yMBazAAAAAOU1vb5Evxy+AAAAAMHQkL7E0JC+AAAAADy/HL7nNb2+AAAAAPLkozHNzMy+AAAAAEm/HD7kNb2+AAAAAMbQkD6/0JC+AAAAAOY1vT5Evxy+AAAAAM3MzD4AAAAA"},{"byteLength":48,"name":"control_x_line_cross","uri":"data:application/octet-stream;base64,AAAAAM3MzL4AAAAAAAAAAM3MzD4AAAAAAAAAAAAAAADNzMy+AAAAAAAAAADNzMw+"},{"byteLength":36,"name":"circle_loop","uri":"data:application/octet-stream;base64,AAAAAM3MzD4AAAAAAAAAAM/MTL6sXLE+AAAAAMvMTL6tXLG+"},{"byteLength":204,"name":"circle_loop","uri":"data:application/octet-stream;base64,AAAAAM3MzD4AAAAAAAAAAOU1vT5Fvxw+AAAAAMPQkD7D0JA+AAAAAES/HD7lNb0+AAAAAPIwlrLNzMw+AAAAAEe/HL7lNb0+AAAAAMPQkL7D0JA+AAAAAOc1vb5Avxw+AAAAAM3MzL7yMBazAAAAAOU1vb5Evxy+AAAAAMHQkL7E0JC+AAAAADy/HL7nNb2+AAAAAPLkozHNzMy+AAAAAEm/HD7kNb2+AAAAAMbQkD6/0JC+AAAAAOY1vT5Evxy+AAAAAM3MzD4AAAAA"},{"byteLength":96,"name":"tex_coords_gate_E:X","uri":"data:application/octet-stream;base64,AAAQPwAAwD4AAAA/AADAPgAAED8AAOA+AAAAPwAAwD4AAAA/AADgPgAAED8AAOA+AAAQPwAA4D4AABA/AADAPgAAAD8AAOA+AAAAPwAA4D4AABA/AADAPgAAAD8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_E:Y","uri":"data:application/octet-stream;base64,AAAQPwAA4D4AAAA/AADgPgAAED8AAAA/AAAAPwAA4D4AAAA/AAAAPwAAED8AAAA/AAAQPwAAAD8AABA/AADgPgAAAD8AAAA/AAAAPwAAAD8AABA/AADgPgAAAD8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_E:Z","uri":"data:application/octet-stream;base64,AAAQPwAAAD8AAAA/AAAAPwAAED8AABA/AAAAPwAAAD8AAAA/AAAQPwAAED8AABA/AAAQPwAAED8AABA/AAAAPwAAAD8AABA/AAAAPwAAED8AABA/AAAAPwAAAD8AAAA/"},{"byteLength":96,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:X","uri":"data:application/octet-stream;base64,AAAgPwAAwD4AABA/AADAPgAAID8AAOA+AAAQPwAAwD4AABA/AADgPgAAID8AAOA+AAAgPwAA4D4AACA/AADAPgAAED8AAOA+AAAQPwAA4D4AACA/AADAPgAAED8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Y","uri":"data:application/octet-stream;base64,AAAgPwAA4D4AABA/AADgPgAAID8AAAA/AAAQPwAA4D4AABA/AAAAPwAAID8AAAA/AAAgPwAAAD8AACA/AADgPgAAED8AAAA/AAAQPwAAAD8AACA/AADgPgAAED8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_ELSE_CORRELATED_ERROR:Z","uri":"data:application/octet-stream;base64,AAAgPwAAAD8AABA/AAAAPwAAID8AABA/AAAQPwAAAD8AABA/AAAQPwAAID8AABA/AAAgPwAAED8AACA/AAAAPwAAED8AABA/AAAQPwAAED8AACA/AAAAPwAAED8AAAA/"},{"byteLength":96,"name":"tex_coords_gate_DEPOLARIZE1","uri":"data:application/octet-stream;base64,AACAPgAAED8AAEA+AAAQPwAAgD4AACA/AABAPgAAED8AAEA+AAAgPwAAgD4AACA/AACAPgAAID8AAIA+AAAQPwAAQD4AACA/AABAPgAAID8AAIA+AAAQPwAAQD4AABA/"},{"byteLength":96,"name":"tex_coords_gate_DEPOLARIZE2","uri":"data:application/octet-stream;base64,AACgPgAAED8AAIA+AAAQPwAAoD4AACA/AACAPgAAED8AAIA+AAAgPwAAoD4AACA/AACgPgAAID8AAKA+AAAQPwAAgD4AACA/AACAPgAAID8AAKA+AAAQPwAAgD4AABA/"},{"byteLength":96,"name":"tex_coords_gate_PAULI_CHANNEL_1","uri":"data:application/octet-stream;base64,AAAQPwAAED8AAAA/AAAQPwAAED8AACA/AAAAPwAAED8AAAA/AAAgPwAAED8AACA/AAAQPwAAID8AABA/AAAQPwAAAD8AACA/AAAAPwAAID8AABA/AAAQPwAAAD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_PAULI_CHANNEL_2","uri":"data:application/octet-stream;base64,AAAgPwAAED8AABA/AAAQPwAAID8AACA/AAAQPwAAED8AABA/AAAgPwAAID8AACA/AAAgPwAAID8AACA/AAAQPwAAED8AACA/AAAQPwAAID8AACA/AAAQPwAAED8AABA/"},{"byteLength":96,"name":"tex_coords_gate_X_ERROR","uri":"data:application/octet-stream;base64,AAAAPwAAwD4AAOA+AADAPgAAAD8AAOA+AADgPgAAwD4AAOA+AADgPgAAAD8AAOA+AAAAPwAA4D4AAAA/AADAPgAA4D4AAOA+AADgPgAA4D4AAAA/AADAPgAA4D4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_Y_ERROR","uri":"data:application/octet-stream;base64,AAAAPwAA4D4AAOA+AADgPgAAAD8AAAA/AADgPgAA4D4AAOA+AAAAPwAAAD8AAAA/AAAAPwAAAD8AAAA/AADgPgAA4D4AAAA/AADgPgAAAD8AAAA/AADgPgAA4D4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_Z_ERROR","uri":"data:application/octet-stream;base64,AAAAPwAAAD8AAOA+AAAAPwAAAD8AABA/AADgPgAAAD8AAOA+AAAQPwAAAD8AABA/AAAAPwAAED8AAAA/AAAAPwAA4D4AABA/AADgPgAAED8AAAA/AAAAPwAA4D4AAAA/"},{"byteLength":96,"name":"tex_coords_gate_HERALDED_ERASE","uri":"data:application/octet-stream;base64,AABwPwAAED8AAGA/AAAQPwAAcD8AACA/AABgPwAAED8AAGA/AAAgPwAAcD8AACA/AABwPwAAID8AAHA/AAAQPwAAYD8AACA/AABgPwAAID8AAHA/AAAQPwAAYD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_HERALDED_PAULI_CHANNEL_1","uri":"data:application/octet-stream;base64,AACAPwAAED8AAHA/AAAQPwAAgD8AACA/AABwPwAAED8AAHA/AAAgPwAAgD8AACA/AACAPwAAID8AAIA/AAAQPwAAcD8AACA/AABwPwAAID8AAIA/AAAQPwAAcD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_MPP:X","uri":"data:application/octet-stream;base64,AAAwPwAAwD4AACA/AADAPgAAMD8AAOA+AAAgPwAAwD4AACA/AADgPgAAMD8AAOA+AAAwPwAA4D4AADA/AADAPgAAID8AAOA+AAAgPwAA4D4AADA/AADAPgAAID8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_MPP:Y","uri":"data:application/octet-stream;base64,AAAwPwAA4D4AACA/AADgPgAAMD8AAAA/AAAgPwAA4D4AACA/AAAAPwAAMD8AAAA/AAAwPwAAAD8AADA/AADgPgAAID8AAAA/AAAgPwAAAD8AADA/AADgPgAAID8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_MPP:Z","uri":"data:application/octet-stream;base64,AAAwPwAAAD8AACA/AAAAPwAAMD8AABA/AAAgPwAAAD8AACA/AAAQPwAAMD8AABA/AAAwPwAAED8AADA/AAAAPwAAID8AABA/AAAgPwAAED8AADA/AAAAPwAAID8AAAA/"},{"byteLength":96,"name":"tex_coords_gate_MRX","uri":"data:application/octet-stream;base64,AADgPgAAwD4AAMA+AADAPgAA4D4AAOA+AADAPgAAwD4AAMA+AADgPgAA4D4AAOA+AADgPgAA4D4AAOA+AADAPgAAwD4AAOA+AADAPgAA4D4AAOA+AADAPgAAwD4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_MRY","uri":"data:application/octet-stream;base64,AADgPgAA4D4AAMA+AADgPgAA4D4AAAA/AADAPgAA4D4AAMA+AAAAPwAA4D4AAAA/AADgPgAAAD8AAOA+AADgPgAAwD4AAAA/AADAPgAAAD8AAOA+AADgPgAAwD4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_MR","uri":"data:application/octet-stream;base64,AADgPgAAAD8AAMA+AAAAPwAA4D4AABA/AADAPgAAAD8AAMA+AAAQPwAA4D4AABA/AADgPgAAED8AAOA+AAAAPwAAwD4AABA/AADAPgAAED8AAOA+AAAAPwAAwD4AAAA/"},{"byteLength":96,"name":"tex_coords_gate_MX","uri":"data:application/octet-stream;base64,AACgPgAAwD4AAIA+AADAPgAAoD4AAOA+AACAPgAAwD4AAIA+AADgPgAAoD4AAOA+AACgPgAA4D4AAKA+AADAPgAAgD4AAOA+AACAPgAA4D4AAKA+AADAPgAAgD4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_MY","uri":"data:application/octet-stream;base64,AACgPgAA4D4AAIA+AADgPgAAoD4AAAA/AACAPgAA4D4AAIA+AAAAPwAAoD4AAAA/AACgPgAAAD8AAKA+AADgPgAAgD4AAAA/AACAPgAAAD8AAKA+AADgPgAAgD4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_M","uri":"data:application/octet-stream;base64,AACgPgAAAD8AAIA+AAAAPwAAoD4AABA/AACAPgAAAD8AAIA+AAAQPwAAoD4AABA/AACgPgAAED8AAKA+AAAAPwAAgD4AABA/AACAPgAAED8AAKA+AAAAPwAAgD4AAAA/"},{"byteLength":96,"name":"tex_coords_gate_RX","uri":"data:application/octet-stream;base64,AADAPgAAwD4AAKA+AADAPgAAwD4AAOA+AACgPgAAwD4AAKA+AADgPgAAwD4AAOA+AADAPgAA4D4AAMA+AADAPgAAoD4AAOA+AACgPgAA4D4AAMA+AADAPgAAoD4AAMA+"},{"byteLength":96,"name":"tex_coords_gate_RY","uri":"data:application/octet-stream;base64,AADAPgAA4D4AAKA+AADgPgAAwD4AAAA/AACgPgAA4D4AAKA+AAAAPwAAwD4AAAA/AADAPgAAAD8AAMA+AADgPgAAoD4AAAA/AACgPgAAAD8AAMA+AADgPgAAoD4AAOA+"},{"byteLength":96,"name":"tex_coords_gate_R","uri":"data:application/octet-stream;base64,AADAPgAAAD8AAKA+AAAAPwAAwD4AABA/AACgPgAAAD8AAKA+AAAQPwAAwD4AABA/AADAPgAAED8AAMA+AAAAPwAAoD4AABA/AACgPgAAED8AAMA+AAAAPwAAoD4AAAA/"},{"byteLength":96,"name":"tex_coords_gate_MXX","uri":"data:application/octet-stream;base64,AAAwPwAAED8AACA/AAAQPwAAMD8AACA/AAAgPwAAED8AACA/AAAgPwAAMD8AACA/AAAwPwAAID8AADA/AAAQPwAAID8AACA/AAAgPwAAID8AADA/AAAQPwAAID8AABA/"},{"byteLength":96,"name":"tex_coords_gate_MYY","uri":"data:application/octet-stream;base64,AABAPwAAED8AADA/AAAQPwAAQD8AACA/AAAwPwAAED8AADA/AAAgPwAAQD8AACA/AABAPwAAID8AAEA/AAAQPwAAMD8AACA/AAAwPwAAID8AAEA/AAAQPwAAMD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_MZZ","uri":"data:application/octet-stream;base64,AABQPwAAED8AAEA/AAAQPwAAUD8AACA/AABAPwAAED8AAEA/AAAgPwAAUD8AACA/AABQPwAAID8AAFA/AAAQPwAAQD8AACA/AABAPwAAID8AAFA/AAAQPwAAQD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_MPAD","uri":"data:application/octet-stream;base64,AABgPwAAED8AAFA/AAAQPwAAYD8AACA/AABQPwAAED8AAFA/AAAgPwAAYD8AACA/AABgPwAAID8AAGA/AAAQPwAAUD8AACA/AABQPwAAID8AAGA/AAAQPwAAUD8AABA/"},{"byteLength":96,"name":"tex_coords_gate_X:REC","uri":"data:application/octet-stream;base64,AABgPwAAwD4AAFA/AADAPgAAYD8AAOA+AABQPwAAwD4AAFA/AADgPgAAYD8AAOA+AABgPwAA4D4AAGA/AADAPgAAUD8AAOA+AABQPwAA4D4AAGA/AADAPgAAUD8AAMA+"},{"byteLength":96,"name":"tex_coords_gate_Y:SWEEP","uri":"data:application/octet-stream;base64,AABwPwAA4D4AAGA/AADgPgAAcD8AAAA/AABgPwAA4D4AAGA/AAAAPwAAcD8AAAA/AABwPwAAAD8AAHA/AADgPgAAYD8AAAA/AABgPwAAAD8AAHA/AADgPgAAYD8AAOA+"},{"byteLength":96,"name":"tex_coords_gate_Z:REC","uri":"data:application/octet-stream;base64,AABgPwAAAD8AAFA/AAAAPwAAYD8AABA/AABQPwAAAD8AAFA/AAAQPwAAYD8AABA/AABgPwAAED8AAGA/AAAAPwAAUD8AABA/AABQPwAAED8AAGA/AAAAPwAAUD8AAAA/"},{"byteLength":1560,"name":"buf_scattered_lines","uri":"data:application/octet-stream;base64,AABAwAAAAMIAAADCAABQwAAAiMEAAIDBAABQwAAAiMEAAIDBAABAwAAAAMAAAACAAABAwAAAgMAAAACAAABAwAAAwMAAAACAAABAwAAAAMEAAACAAABAwAAAIMEAAACAAABAwAAAQMEAAACAAABAwAAAYMEAAACAAABAwAAAgMEAAACAAABAwAAAkMEAAACAAABAwAAAoMEAAACAAABAwAAAsMEAAACAAACAwAAAAMIAAADCAACIwAAAiMEAAIDBAACIwAAAiMEAAIDBAACAwAAAAMAAAACAAACAwAAAgMAAAACAAACAwAAAwMAAAACAAACAwAAAAMEAAACAAACAwAAAIMEAAACAAACAwAAAQMEAAACAAACAwAAAYMEAAACAAACAwAAAgMEAAACAAACAwAAAkMEAAACAAACAwAAAoMEAAACAAACAwAAAsMEAAACAAACgwAAAAMIAAADCAACowAAAiMEAAIDBAACowAAAiMEAAIDBAACgwAAAAMAAAACAAACgwAAAgMAAAACAAACgwAAAwMAAAACAAACgwAAAAMEAAACAAACgwAAAIMEAAACAAACgwAAAQMEAAACAAACgwAAAYMEAAACAAACgwAAAgMEAAACAAACgwAAAkMEAAACAAACgwAAAoMEAAACAAACgwAAAsMEAAACAAACgwAAAwMEAAACAAACgwAAA0MEAAACAAACgwAAA4MEAAACAAACgwAAA8MEAAACAAACgwAAAAMIAAACAAACgwAAACMIAAACAAADAwAAAgMAAAACAAADAwAAAAMAAAACAAADAwAAAwMAAAACAAADAwAAAgMAAAACAAADAwAAAYMEAAACAAADIwAAAMMEAAACAAADIwAAAMMEAAACAAADAwAAAAMEAAACAAADAwAAAQMEAAACAAADAwAAAYMEAAACAAADgwAAAAMAAAACAAADgwAAAgMAAAACAAADgwAAAAMEAAACAAADgwAAAIMEAAACAAAAQwQAAAMAAAACAAAAUwQAAiMEAAIDBAAAUwQAAiMEAAIDBAAAQwQAAAMIAAADCAAAQwQAAgMAAAACAAAAQwQAAAMAAAACAAAAgwQAAAMAAAACAAAAkwQAAiMEAAIDBAAAkwQAAiMEAAIDBAAAgwQAAAMIAAADCAABAwQAAAMIAAADCAABEwQAAiMEAAIDBAABEwQAAiMEAAIDBAABAwQAAAMAAAACAAABAwQAAgMAAAACAAABAwQAAwMAAAACAAABAwQAAAMEAAACAAABAwQAAIMEAAACAAABAwQAAQMEAAACAAABAwQAAYMEAAACAAABwwQAAAMIAAADCAAB0wQAAiMEAAIDBAAB0wQAAiMEAAIDBAABwwQAAAMAAAACAAADAwQAAgMAAAACAAADAwQAAwMAAAACAAADAwQAAAMEAAACAAADAwQAAIMEAAACAAADAwQAAYMEAAACAAADAwQAAQMEAAACAAADAwQAAgMEAAACAAADAwQAAYMEAAACAAACAPwAAAMIAAADCAADQwQAAAMIAAADCAACAPwAAAMAAAACAAADQwQAAAMAAAACAAACAPwAAgMAAAACAAADQwQAAgMAAAACAAACAPwAAwMAAAACAAADQwQAAwMAAAACAAACAPwAAAMEAAACAAADQwQAAAMEAAACAAACAPwAAIMEAAACAAADQwQAAIMEAAACAAACAPwAAQMEAAACAAADQwQAAQMEAAACAAACAPwAAYMEAAACAAADQwQAAYMEAAACAAACAPwAAgMEAAACAAADQwQAAgMEAAACAAACAPwAAkMEAAACAAADQwQAAkMEAAACAAACAPwAAoMEAAACAAADQwQAAoMEAAACAAACAPwAAsMEAAACAAADQwQAAsMEAAACAAACAPwAAwMEAAACAAADQwQAAwMEAAACAAACAPwAA0MEAAACAAADQwQAA0MEAAACAAACAPwAA4MEAAACAAADQwQAA4MEAAACAAACAPwAA8MEAAACAAADQwQAA8MEAAACAAACAPwAAAMIAAACAAADQwQAAAMIAAACAAACAPwAACMIAAACAAADQwQAACMIAAACA"},{"byteLength":360,"name":"buf_red_scattered_lines","uri":"data:application/octet-stream;base64,AAAAAAAAAIAAAIDBAABAwAAAAIAAAIDBAAAgwAAAAL8AAIDBAABAwAAAAIAAAIDBAAAgwAAAAD8AAIDBAABAwAAAAIAAAIDBAABcwQAAgL8AAIA/AABcwQAAgL8AAATCAABcwQAAgL8AAIA/AABcwQAADMIAAIA/AABcwQAAgL8AAIA/AACKwQAAgL8AAIA/AABcwQAAgL8AAATCAABcwQAADMIAAATCAABcwQAAgL8AAATCAACKwQAAgL8AAATCAABcwQAADMIAAIA/AABcwQAADMIAAATCAABcwQAADMIAAIA/AACKwQAADMIAAIA/AABcwQAADMIAAATCAACKwQAADMIAAATCAACKwQAAgL8AAIA/AACKwQAAgL8AAATCAACKwQAAgL8AAIA/AACKwQAADMIAAIA/AACKwQAAgL8AAATCAACKwQAADMIAAATCAACKwQAADMIAAIA/AACKwQAADMIAAATC"},{"byteLength":1152,"name":"buf_blue_scattered_lines","uri":"data:application/octet-stream;base64,AABAv83MzL8AAAA/AABAv5qZmb8AAAA/AABAv83MzL8AAALCAABAv5qZmb8AAALCAABAv5qZmb8AAAA/AABAv5qZmb8AAALCAABAv5qZmb8AAAA/AAAQwJqZmb8AAAA/AABAv5qZmb8AAALCAAAQwJqZmb8AAALCAAAQwM3MzL8AAAA/AAAQwJqZmb8AAAA/AAAQwM3MzL8AAALCAAAQwJqZmb8AAALCAAAQwJqZmb8AAAA/AAAQwJqZmb8AAALCAAAwwM3MzL8AAAA/AAAwwJqZmb8AAAA/AAAwwM3MzL8AAALCAAAwwJqZmb8AAALCAAAwwJqZmb8AAAA/AAAwwJqZmb8AAALCAAAwwJqZmb8AAAA/AACowJqZmb8AAAA/AAAwwJqZmb8AAALCAACowJqZmb8AAALCAACowM3MzL8AAAA/AACowJqZmb8AAAA/AACowM3MzL8AAALCAACowJqZmb8AAALCAACowJqZmb8AAAA/AACowJqZmb8AAALCAAC4wM3MzL8AAAA/AAC4wJqZmb8AAAA/AAC4wM3MzL8AAALCAAC4wJqZmb8AAALCAAC4wJqZmb8AAAA/AAC4wJqZmb8AAALCAAC4wJqZmb8AAAA/AAAEwZqZmb8AAAA/AAC4wJqZmb8AAALCAAAEwZqZmb8AAALCAAAEwc3MzL8AAAA/AAAEwZqZmb8AAAA/AAAEwc3MzL8AAALCAAAEwZqZmb8AAALCAAAEwZqZmb8AAAA/AAAEwZqZmb8AAALCAAAMwc3MzL8AAAA/AAAMwZqZmb8AAAA/AAAMwc3MzL8AAALCAAAMwZqZmb8AAALCAAAMwZqZmb8AAAA/AAAMwZqZmb8AAALCAAAMwZqZmb8AAAA/AAA0wZqZmb8AAAA/AAAMwZqZmb8AAALCAAA0wZqZmb8AAALCAAA0wc3MzL8AAAA/AAA0wZqZmb8AAAA/AAA0wc3MzL8AAALCAAA0wZqZmb8AAALCAAA0wZqZmb8AAAA/AAA0wZqZmb8AAALCAABcwc3MzL8AAAA/AABcwZqZmb8AAAA/AABcwc3MzL8AAALCAABcwZqZmb8AAALCAABcwZqZmb8AAAA/AABcwZqZmb8AAALCAABcwZqZmb8AAAA/AACCwZqZmb8AAAA/AABcwZqZmb8AAALCAACCwZqZmb8AAALCAACCwc3MzL8AAAA/AACCwZqZmb8AAAA/AACCwc3MzL8AAALCAACCwZqZmb8AAALCAACCwZqZmb8AAAA/AACCwZqZmb8AAALCAACWwc3MzL8AAAA/AACWwZqZmb8AAAA/AACWwc3MzL8AAALCAACWwZqZmb8AAALCAACWwZqZmb8AAAA/AACWwZqZmb8AAALCAACWwZqZmb8AAAA/AAC6wZqZmb8AAAA/AACWwZqZmb8AAALCAAC6wZqZmb8AAALCAAC6wc3MzL8AAAA/AAC6wZqZmb8AAAA/AAC6wc3MzL8AAALCAAC6wZqZmb8AAALCAAC6wZqZmb8AAAA/AAC6wZqZmb8AAALC"}],"images":[{"name":"gates_image","uri":""}],"materials":[{"doubleSided":false,"name":"gates_material","pbrMetallicRoughness":{"baseColorFactor":[1,1,1,1],"baseColorTexture":{"index":0,"texCoord":0},"metallicFactor":0.4,"roughnessFactor":0.5}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"white","pbrMetallicRoughness":{"baseColorFactor":[1,1,1,1],"metallicFactor":0.4,"roughnessFactor":0.5}},{"doubleSided":true,"name":"white","pbrMetallicRoughness":{"baseColorFactor":[1,1,1,1],"metallicFactor":0.4,"roughnessFactor":0.5}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"white","pbrMetallicRoughness":{"baseColorFactor":[1,1,1,1],"metallicFactor":0.4,"roughnessFactor":0.5}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"gray","pbrMetallicRoughness":{"baseColorFactor":[0.5,0.5,0.5,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"black","pbrMetallicRoughness":{"baseColorFactor":[0,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"red","pbrMetallicRoughness":{"baseColorFactor":[1,0,0,1],"metallicFactor":1,"roughnessFactor":1}},{"doubleSided":true,"name":"blue","pbrMetallicRoughness":{"baseColorFactor":[0,0,1,1],"metallicFactor":1,"roughnessFactor":1}}],"meshes":[{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":1},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":2},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":3},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":4},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":5},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":6},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":7},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":8},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":9},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":10},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":11},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":12},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":13},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":14},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":15},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":16},"material":1,"mode":6},{"attributes":{"POSITION":17},"material":2,"mode":1}]},{"primitives":[{"attributes":{"POSITION":18},"material":3,"mode":6},{"attributes":{"POSITION":18},"material":4,"mode":3},{"attributes":{"POSITION":19},"material":4,"mode":1}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":20},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":21},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":22},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":23},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":24},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":25},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":26},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":27},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":28},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":29},"material":5,"mode":6},{"attributes":{"POSITION":29},"material":6,"mode":3},{"attributes":{"POSITION":30},"material":6,"mode":1}]},{"primitives":[{"attributes":{"POSITION":31},"material":7,"mode":2},{"attributes":{"POSITION":31},"material":8,"mode":4}]},{"primitives":[{"attributes":{"POSITION":32},"material":9,"mode":6}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":33},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":34},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":35},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":36},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":37},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":38},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":39},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":40},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":41},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":42},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":43},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":44},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":45},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":46},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":47},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":48},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":49},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":50},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":51},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":52},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":53},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":54},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":55},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":56},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":57},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":58},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":59},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":60},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":61},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":62},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":63},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":64},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":65},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":0,"TEXCOORD_0":66},"material":0,"mode":4}]},{"primitives":[{"attributes":{"POSITION":67},"material":10,"mode":1}]},{"primitives":[{"attributes":{"POSITION":68},"material":11,"mode":1}]},{"primitives":[{"attributes":{"POSITION":69},"material":12,"mode":1}]}],"nodes":[{"mesh":0,"translation":[-0,-32,-32]},{"mesh":1,"translation":[-0,-2,-0]},{"mesh":2,"translation":[-0,-4,-0]},{"mesh":3,"translation":[-0,-6,-0]},{"mesh":4,"translation":[-1,-32,-32]},{"mesh":5,"translation":[-1,-2,-0]},{"mesh":6,"translation":[-1,-4,-0]},{"mesh":7,"translation":[-1,-6,-0]},{"mesh":8,"translation":[-1,-8,-0]},{"mesh":9,"translation":[-2,-32,-32]},{"mesh":10,"translation":[-2,-2,-0]},{"mesh":11,"translation":[-2,-4,-0]},{"mesh":12,"translation":[-2,-6,-0]},{"mesh":13,"translation":[-2,-8,-0]},{"mesh":14,"translation":[-2,-10,-0]},{"mesh":15,"translation":[-3,-32,-32]},{"mesh":16,"translation":[-3,-2,-0]},{"mesh":17,"translation":[-3,-4,-0]},{"mesh":17,"translation":[-3,-6,-0]},{"mesh":18,"translation":[-3,-8,-0]},{"mesh":18,"translation":[-3,-10,-0]},{"mesh":19,"translation":[-3,-12,-0]},{"mesh":19,"translation":[-3,-14,-0]},{"mesh":16,"translation":[-3,-16,-0]},{"mesh":15,"translation":[-3,-18,-0]},{"mesh":15,"translation":[-3,-20,-0]},{"mesh":15,"translation":[-3,-22,-0]},{"mesh":20,"translation":[-4,-32,-32]},{"mesh":20,"translation":[-4,-2,-0]},{"mesh":21,"translation":[-4,-4,-0]},{"mesh":21,"translation":[-4,-6,-0]},{"mesh":22,"translation":[-4,-8,-0]},{"mesh":22,"translation":[-4,-10,-0]},{"mesh":23,"translation":[-4,-12,-0]},{"mesh":23,"translation":[-4,-14,-0]},{"mesh":24,"translation":[-4,-16,-0]},{"mesh":24,"translation":[-4,-18,-0]},{"mesh":25,"translation":[-4,-20,-0]},{"mesh":25,"translation":[-4,-22,-0]},{"mesh":26,"translation":[-5,-32,-32]},{"mesh":26,"translation":[-5,-2,-0]},{"mesh":26,"translation":[-5,-4,-0]},{"mesh":27,"translation":[-5,-6,-0]},{"mesh":26,"translation":[-5,-8,-0]},{"mesh":28,"translation":[-5,-10,-0]},{"mesh":27,"translation":[-5,-12,-0]},{"mesh":26,"translation":[-5,-14,-0]},{"mesh":27,"translation":[-5,-16,-0]},{"mesh":27,"translation":[-5,-18,-0]},{"mesh":27,"translation":[-5,-20,-0]},{"mesh":28,"translation":[-5,-22,-0]},{"mesh":28,"translation":[-5,-24,-0]},{"mesh":26,"translation":[-5,-26,-0]},{"mesh":28,"translation":[-5,-28,-0]},{"mesh":27,"translation":[-5,-30,-0]},{"mesh":28,"translation":[-5,-32,-0]},{"mesh":28,"translation":[-5,-34,-0]},{"mesh":29,"translation":[-6,-2,-0]},{"mesh":30,"translation":[-6,-4,-0]},{"mesh":31,"translation":[-6,-6,-0]},{"mesh":32,"translation":[-6,-8,-0]},{"mesh":33,"translation":[-6,-14,-0]},{"mesh":34,"translation":[-6,-12,-0]},{"mesh":35,"translation":[-6,-32,-32]},{"mesh":36,"translation":[-7,-2,-0]},{"mesh":36,"translation":[-7,-4,-0]},{"mesh":37,"translation":[-7,-6,-0]},{"mesh":38,"translation":[-7,-8,-0]},{"mesh":38,"translation":[-7,-10,-0]},{"mesh":39,"translation":[-7,-32,-32]},{"mesh":40,"translation":[-8,-2,-0]},{"mesh":41,"translation":[-8,-4,-0]},{"mesh":42,"translation":[-8,-6,-0]},{"mesh":43,"translation":[-8,-12,-0]},{"mesh":44,"translation":[-9,-32,-32]},{"mesh":45,"translation":[-9,-2,-0]},{"mesh":46,"translation":[-9,-4,-0]},{"mesh":46,"translation":[-10,-32,-32]},{"mesh":46,"translation":[-10,-2,-0]},{"mesh":47,"translation":[-11,-32,-32]},{"mesh":48,"translation":[-11,-2,-0]},{"mesh":49,"translation":[-11,-4,-0]},{"mesh":50,"translation":[-11,-6,-0]},{"mesh":51,"translation":[-11,-8,-0]},{"mesh":52,"translation":[-11,-10,-0]},{"mesh":52,"translation":[-11,-12,-0]},{"mesh":53,"translation":[-11,-14,-0]},{"mesh":54,"translation":[-11,-16,-0]},{"mesh":55,"translation":[-11,-18,-0]},{"mesh":56,"translation":[-12,-32,-32]},{"mesh":56,"translation":[-12,-2,-0]},{"mesh":56,"translation":[-12,-4,-0]},{"mesh":56,"translation":[-12,-6,-0]},{"mesh":57,"translation":[-12,-8,-0]},{"mesh":57,"translation":[-12,-10,-0]},{"mesh":58,"translation":[-12,-12,-0]},{"mesh":58,"translation":[-12,-14,-0]},{"mesh":7,"translation":[-14,-32,-32]},{"mesh":28,"translation":[-15,-32,-32]},{"mesh":26,"translation":[-15,-2,-0]},{"mesh":13,"translation":[-16,-2,-0]},{"mesh":49,"translation":[-19,-32,-32]},{"mesh":39,"translation":[-20,-32,-32]},{"mesh":49,"translation":[-21,-32,-32]},{"mesh":59,"translation":[-22,-32,-32]},{"mesh":59,"translation":[-22,-2,-0]},{"mesh":59,"translation":[-23,-32,-32]},{"mesh":47,"translation":[-24,-32,-32]},{"mesh":51,"translation":[-24,-2,-0]},{"mesh":58,"translation":[-24,-4,-0]},{"mesh":58,"translation":[-24,-6,-0]},{"mesh":57,"translation":[-24,-8,-0]},{"mesh":57,"translation":[-24,-10,-0]},{"mesh":44,"translation":[-24,-12,-0]},{"mesh":45,"translation":[-24,-14,-0]},{"mesh":46,"translation":[-24,-16,-0]},{"mesh":60,"translation":[-25,-32,-32]},{"mesh":61,"translation":[-25,-2,-0]},{"mesh":62,"translation":[-25,-4,-0]},{"mesh":63,"translation":[0,0,0]},{"mesh":64,"translation":[0,0,0]},{"mesh":65,"translation":[0,0,0]}],"samplers":[{"magFilter":9728,"minFilter":9728,"wrapS":33071,"wrapT":33071}],"scene":0,"scenes":[{"nodes":[0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,118,119,120,121]}],"textures":[{"name":"gates_texture","sampler":0,"source":0}]} \ No newline at end of file diff --git a/testdata/circuit_all_ops_detslice.svg b/testdata/circuit_all_ops_detslice.svg index d96f0bf5b..8d5706f6e 100644 --- a/testdata/circuit_all_ops_detslice.svg +++ b/testdata/circuit_all_ops_detslice.svg @@ -290,6 +290,11 @@ + + + + + √XX diff --git a/testdata/circuit_all_ops_timeline.svg b/testdata/circuit_all_ops_timeline.svg index 70b707457..1c9563c2f 100644 --- a/testdata/circuit_all_ops_timeline.svg +++ b/testdata/circuit_all_ops_timeline.svg @@ -98,6 +98,11 @@ + + + + + √XX diff --git a/testdata/circuit_all_ops_timeslice.svg b/testdata/circuit_all_ops_timeslice.svg index d96f0bf5b..8d5706f6e 100644 --- a/testdata/circuit_all_ops_timeslice.svg +++ b/testdata/circuit_all_ops_timeslice.svg @@ -290,6 +290,11 @@ + + + + + √XX