From 427656544e9c700a3f3d2529b97d55f449ee7d3a Mon Sep 17 00:00:00 2001 From: "Documenter.jl" Date: Fri, 22 Mar 2024 05:13:04 +0000 Subject: [PATCH] build based on 7be5818 --- dev/.documenter-siteinfo.json | 2 +- dev/API/index.html | 104 ++++++++++---------- dev/ECC_API/index.html | 6 +- dev/ECC_evaluating/987beb83.png | Bin 0 -> 42702 bytes dev/ECC_evaluating/a228365f.png | Bin 56139 -> 0 bytes dev/ECC_evaluating/ae335f84.png | Bin 42907 -> 0 bytes dev/ECC_evaluating/b5cc84d2.png | Bin 0 -> 55751 bytes dev/ECC_evaluating/index.html | 4 +- dev/allops/index.html | 2 +- dev/canonicalization/03a5321d.png | Bin 9447 -> 0 bytes dev/canonicalization/18569f0a.png | Bin 0 -> 9497 bytes dev/canonicalization/2aca2462.png | Bin 9317 -> 0 bytes dev/canonicalization/2c1e3a21.png | Bin 0 -> 9704 bytes dev/canonicalization/b9c71a95.png | Bin 9703 -> 0 bytes dev/canonicalization/bb99d1e1.png | Bin 9379 -> 0 bytes dev/canonicalization/ec31a5b5.png | Bin 0 -> 9379 bytes dev/canonicalization/ed23db7b.png | Bin 0 -> 9302 bytes dev/canonicalization/index.html | 8 +- dev/commonstates/index.html | 2 +- dev/datastructures/index.html | 2 +- dev/ecc_example_sim/index.html | 8 +- dev/graphs/index.html | 2 +- dev/index.html | 2 +- dev/mixed/index.html | 2 +- dev/noise/index.html | 2 +- dev/noisycircuits/index.html | 2 +- dev/noisycircuits_API/index.html | 2 +- dev/noisycircuits_mc/index.html | 4 +- dev/noisycircuits_ops/index.html | 2 +- dev/noisycircuits_perturb/index.html | 4 +- dev/plotting/{8c14f637.svg => 02c64905.svg} | 12 +-- dev/plotting/{176f5a9c.svg => 0f1d3571.svg} | 12 +-- dev/plotting/{ec99b703.svg => 429aa586.svg} | 12 +-- dev/plotting/{871713f6.svg => 68cd9931.svg} | 12 +-- dev/plotting/{68f88281.svg => b39b13b5.svg} | 12 +-- dev/plotting/index.html | 12 +-- dev/references.bib | 47 +++++++++ dev/references/index.html | 2 +- dev/search_index.js | 2 +- dev/stab-algebra-manual/index.html | 2 +- dev/tutandpub/index.html | 2 +- 41 files changed, 167 insertions(+), 120 deletions(-) create mode 100644 dev/ECC_evaluating/987beb83.png delete mode 100644 dev/ECC_evaluating/a228365f.png delete mode 100644 dev/ECC_evaluating/ae335f84.png create mode 100644 dev/ECC_evaluating/b5cc84d2.png delete mode 100644 dev/canonicalization/03a5321d.png create mode 100644 dev/canonicalization/18569f0a.png delete mode 100644 dev/canonicalization/2aca2462.png create mode 100644 dev/canonicalization/2c1e3a21.png delete mode 100644 dev/canonicalization/b9c71a95.png delete mode 100644 dev/canonicalization/bb99d1e1.png create mode 100644 dev/canonicalization/ec31a5b5.png create mode 100644 dev/canonicalization/ed23db7b.png rename dev/plotting/{8c14f637.svg => 02c64905.svg} (99%) rename dev/plotting/{176f5a9c.svg => 0f1d3571.svg} (98%) rename dev/plotting/{ec99b703.svg => 429aa586.svg} (99%) rename dev/plotting/{871713f6.svg => 68cd9931.svg} (98%) rename dev/plotting/{68f88281.svg => b39b13b5.svg} (98%) diff --git a/dev/.documenter-siteinfo.json b/dev/.documenter-siteinfo.json index 2c1cf1129..68f880db9 100644 --- a/dev/.documenter-siteinfo.json +++ b/dev/.documenter-siteinfo.json @@ -1 +1 @@ -{"documenter":{"julia_version":"1.10.2","generation_timestamp":"2024-03-20T04:14:31","documenter_version":"1.3.0"}} \ No newline at end of file +{"documenter":{"julia_version":"1.10.2","generation_timestamp":"2024-03-22T05:12:57","documenter_version":"1.3.0"}} \ No newline at end of file diff --git a/dev/API/index.html b/dev/API/index.html index b72fffae0..f88bb6257 100644 --- a/dev/API/index.html +++ b/dev/API/index.html @@ -4,7 +4,7 @@ padding-top: 0 !important; padding-bottom: 0 !important; } -

States

Stabilizer states can be represented with the Stabilizer, Destabilizer, MixedStabilizer, and MixedDestabilizer tableau data structures. You probably want to use MixedDestabilizer which supports the widest set of operations.

Moreover, a MixedDestabilizer can be stored inside a Register together with a set of classical bits in which measurement results can be written.

Lastly, for Pauli frame simulations there is the PauliFrame type, a tableau in which each row represents a different Pauli frame.

There are convenience constructors for common types of states and operators.

Operations

Acting on quantum states can be performed either:

See the full list of operations for a list of implemented operations.

Autogenerated API list

QuantumClifford.QuantumCliffordModule

A module for using the Stabilizer formalism and simulating Clifford circuits.

source
QuantumClifford.continue_statConstant

Returned by applywstatus! if the circuit simulation should continue.

source
QuantumClifford.failure_statConstant

Returned by applywstatus! if the circuit reports a failure.

See also: VerifyOp, BellMeasurement.

source
QuantumClifford.false_success_statConstant

Returned by applywstatus! if the circuit reports a success, but it is a false positive (i.e., there was an undetected error).

See also: VerifyOp, BellMeasurement.

source
QuantumClifford.true_success_statConstant

Returned by applywstatus! if the circuit reports a success and there is no undetected error.

See also: VerifyOp, BellMeasurement.

source
QuantumClifford.AbstractSingleQubitOperatorType

Supertype of all single-qubit symbolic operators.

source
QuantumClifford.AbstractSymbolicOperatorType

Supertype of all symbolic operators. Subtype of AbstractCliffordOperator

source
QuantumClifford.AbstractTwoQubitOperatorType

Supertype of all two-qubit symbolic operators.

source
QuantumClifford.BellMeasurementType

A Bell measurement performing the correlation measurement corresponding to the given pauli projections on the qubits at the selected indices.

source
QuantumClifford.CircuitStatusType

A convenience struct to represent the status of a circuit simulated by mctrajectories

source
QuantumClifford.ClassicalXORType

Applies an XOR gate to classical bits. Currently only implemented for functionality with pauli frames.

source
QuantumClifford.CliffordOperatorType

Clifford Operator specified by the mapping of the basis generators.

julia> tCNOT
+

States

Stabilizer states can be represented with the Stabilizer, Destabilizer, MixedStabilizer, and MixedDestabilizer tableau data structures. You probably want to use MixedDestabilizer which supports the widest set of operations.

Moreover, a MixedDestabilizer can be stored inside a Register together with a set of classical bits in which measurement results can be written.

Lastly, for Pauli frame simulations there is the PauliFrame type, a tableau in which each row represents a different Pauli frame.

There are convenience constructors for common types of states and operators.

Operations

Acting on quantum states can be performed either:

  • In a "linear algebra" language where unitaries, measurements, and other operations have separate interfaces. This is an explicitly deterministic lower-level interface, which provides a great deal of control over how tableaux are manipulated. See the Stabilizer Tableau Algebra Manual as a primer on these approaches.
  • Or in a "circuit" language, where the operators (and measurements and noise) are represented as circuit gates. This is a higher-level interface in which the outcome of an operation can be stochastic. The API for it is centered around the apply! function. Particularly useful for Monte Carlo simulations and Perturbative Expansion Symbolic Results.

See the full list of operations for a list of implemented operations.

Autogenerated API list

QuantumClifford.BellMeasurementType

A Bell measurement performing the correlation measurement corresponding to the given pauli projections on the qubits at the selected indices.

source
QuantumClifford.CliffordOperatorType

Clifford Operator specified by the mapping of the basis generators.

julia> tCNOT
 X₁ ⟼ + XX
 X₂ ⟼ + _X
 Z₁ ⟼ + Z_
@@ -33,12 +33,12 @@
 
 julia> CliffordOperator(d)
 X₁ ⟼ + Z
-Z₁ ⟼ + Y
source
QuantumClifford.DestabilizerType

A tableau representation of a pure stabilizer state. The tableau tracks the destabilizers as well, for efficient projections. On initialization there are no checks that the provided state is indeed pure. This enables the use of this data structure for mixed stabilizer state, but a better choice would be to use MixedDestabilizer.

source
QuantumClifford.MixedDestabilizerType

A tableau representation for mixed stabilizer states that keeps track of the destabilizers in order to provide efficient projection operations.

The rank r of the n-qubit tableau is tracked, either so that it can be used to represent a mixed stabilizer state, or so that it can be used to represent an n-r logical-qubit code over n physical qubits. The "logical" operators are tracked as well.

When the constructor is called on an incomplete Stabilizer it automatically calculates the destabilizers and logical operators, following chapter 4 of (Gottesman, 1997). Under the hood the conversion uses the canonicalize_gott! canonicalization. That canonicalization permutes the columns of the tableau, but we automatically undo the column permutation in the preparation of a MixedDestabilizer so that qubits are not reindexed. The boolean keyword arguments undoperm and reportperm can be used to control this behavior and to report the permutations explicitly.

See also: stabilizerview, destabilizerview, logicalxview, logicalzview

source
QuantumClifford.PauliFrameType
struct PauliFrame{T, S} <: QuantumClifford.AbstractQCState

This is a wrapper around a tableau. This "frame" tableau is not to be viewed as a normal stabilizer tableau, although it does conjugate the same under Clifford operations. Each row in the tableau refers to a single frame. The row represents the Pauli operation by which the frame and the reference differ.

source
QuantumClifford.DestabilizerType

A tableau representation of a pure stabilizer state. The tableau tracks the destabilizers as well, for efficient projections. On initialization there are no checks that the provided state is indeed pure. This enables the use of this data structure for mixed stabilizer state, but a better choice would be to use MixedDestabilizer.

source
QuantumClifford.MixedDestabilizerType

A tableau representation for mixed stabilizer states that keeps track of the destabilizers in order to provide efficient projection operations.

The rank r of the n-qubit tableau is tracked, either so that it can be used to represent a mixed stabilizer state, or so that it can be used to represent an n-r logical-qubit code over n physical qubits. The "logical" operators are tracked as well.

When the constructor is called on an incomplete Stabilizer it automatically calculates the destabilizers and logical operators, following chapter 4 of (Gottesman, 1997). Under the hood the conversion uses the canonicalize_gott! canonicalization. That canonicalization permutes the columns of the tableau, but we automatically undo the column permutation in the preparation of a MixedDestabilizer so that qubits are not reindexed. The boolean keyword arguments undoperm and reportperm can be used to control this behavior and to report the permutations explicitly.

See also: stabilizerview, destabilizerview, logicalxview, logicalzview

source
QuantumClifford.PauliFrameType
struct PauliFrame{T, S} <: QuantumClifford.AbstractQCState

This is a wrapper around a tableau. This "frame" tableau is not to be viewed as a normal stabilizer tableau, although it does conjugate the same under Clifford operations. Each row in the tableau refers to a single frame. The row represents the Pauli operation by which the frame and the reference differ.

source
QuantumClifford.PauliFrameMethod
PauliFrame(
     frames,
     qubits,
     measurements
 ) -> PauliFrame{Stabilizer{QuantumClifford.Tableau{Vector{UInt8}, LinearAlgebra.Adjoint{UInt64, Matrix{UInt64}}}}}
-

Prepare an empty set of Pauli frames with the given number of frames and qubits. Preallocates spaces for measurement number of measurements.

source
QuantumClifford.PauliOperatorType

A multi-qubit Pauli operator ($±\{1,i\}\{I,Z,X,Y\}^{\otimes n}$).

A Pauli can be constructed with the P custom string macro or by building up one through products and tensor products of smaller operators.

julia> pauli3 = P"-iXYZ"
+

Prepare an empty set of Pauli frames with the given number of frames and qubits. Preallocates spaces for measurement number of measurements.

source
QuantumClifford.PauliOperatorType

A multi-qubit Pauli operator ($±\{1,i\}\{I,Z,X,Y\}^{\otimes n}$).

A Pauli can be constructed with the P custom string macro or by building up one through products and tensor products of smaller operators.

julia> pauli3 = P"-iXYZ"
 -iXYZ
 
 julia> pauli4 = 1im * pauli3 ⊗ X
@@ -55,7 +55,7 @@
 (true, false)
 
 julia> p[1] = (true, true); p
-+ YYZ
source
QuantumClifford.RegisterType

A register, representing the state of a computer including both a tableaux and an array of classical bits (e.g. for storing measurement results)

source
QuantumClifford.ResetType

Reset the specified qubits to the given state.

Be careful, this operation implies first tracing out the qubits, which can lead to mixed states if these qubits were entangled with the rest of the system.

See also: sMRZ

source
QuantumClifford.SingleQubitOperatorType

A "symbolic" general single-qubit operator which permits faster multiplication than an operator expressed as an explicit tableau.

julia> op = SingleQubitOperator(2, true, true, true, false, true, true) # Tableau components and phases
++ YYZ
source
QuantumClifford.RegisterType

A register, representing the state of a computer including both a tableaux and an array of classical bits (e.g. for storing measurement results)

source
QuantumClifford.ResetType

Reset the specified qubits to the given state.

Be careful, this operation implies first tracing out the qubits, which can lead to mixed states if these qubits were entangled with the rest of the system.

See also: sMRZ

source
QuantumClifford.SingleQubitOperatorType

A "symbolic" general single-qubit operator which permits faster multiplication than an operator expressed as an explicit tableau.

julia> op = SingleQubitOperator(2, true, true, true, false, true, true) # Tableau components and phases
 SingleQubitOperator on qubit 2
 X₁ ⟼ - Y
 Z₁ ⟼ - X
@@ -76,7 +76,7 @@
 
 julia> CliffordOperator(op, 1, compact=true) # You can also extract just the non-trivial part of the tableau
 X₁ ⟼ - Y
-Z₁ ⟼ - X

See also: sHadamard, sPhase, sId1, sX, sY, sZ, CliffordOperator

Or simply consult subtypes(QuantumClifford.AbstractSingleQubitOperator) and subtypes(QuantumClifford.AbstractTwoQubitOperator) for a full list. You can think of the s prefix as "symbolic" or "sparse".

source
QuantumClifford.SparseGateType

A Clifford gate, applying the given cliff operator to the qubits at the selected indices.

apply!(state, cliff, indices) and apply!(state, SparseGate(cliff, indices)) give the same result.

source
QuantumClifford.StabMixtureType
mutable struct StabMixture{T, F}

Represents mixture ∑ ϕᵢⱼ Pᵢ ρ Pⱼ† where ρ is a pure stabilizer state.

julia> StabMixture(S"-X")
+Z₁ ⟼ - X

See also: sHadamard, sPhase, sId1, sX, sY, sZ, CliffordOperator

Or simply consult subtypes(QuantumClifford.AbstractSingleQubitOperator) and subtypes(QuantumClifford.AbstractTwoQubitOperator) for a full list. You can think of the s prefix as "symbolic" or "sparse".

source
QuantumClifford.SparseGateType

A Clifford gate, applying the given cliff operator to the qubits at the selected indices.

apply!(state, cliff, indices) and apply!(state, SparseGate(cliff, indices)) give the same result.

source
QuantumClifford.StabMixtureType
mutable struct StabMixture{T, F}

Represents mixture ∑ ϕᵢⱼ Pᵢ ρ Pⱼ† where ρ is a pure stabilizer state.

julia> StabMixture(S"-X")
 A mixture ∑ ϕᵢⱼ Pᵢ ρ Pⱼ† where ρ is
 𝒟ℯ𝓈𝓉𝒶𝒷
 + Z
@@ -101,7 +101,7 @@
  0.0+0.353553im | + _ | + Z
  0.0-0.353553im | + Z | + _
  0.853553+0.0im | + _ | + _
- 0.146447+0.0im | + Z | + Z

See also: PauliChannel

source
QuantumClifford.StabilizerType

Stabilizer, i.e. a list of commuting multi-qubit Hermitian Pauli operators.

Instances can be created with the S custom string macro or as direct sum of other stabilizers.

Stabilizers and Destabilizers

In many cases you probably would prefer to use the MixedDestabilizer data structure, as it caries a lot of useful additional information, like tracking rank and destabilizer operators. Stabilizer has mostly a pedagogical value, and it is also used for slightly faster simulation of a particular subset of Clifford operations.

julia> s = S"XXX
+ 0.146447+0.0im | + Z | + Z

See also: PauliChannel

source
QuantumClifford.StabilizerType

Stabilizer, i.e. a list of commuting multi-qubit Hermitian Pauli operators.

Instances can be created with the S custom string macro or as direct sum of other stabilizers.

Stabilizers and Destabilizers

In many cases you probably would prefer to use the MixedDestabilizer data structure, as it caries a lot of useful additional information, like tracking rank and destabilizer operators. Stabilizer has mostly a pedagogical value, and it is also used for slightly faster simulation of a particular subset of Clifford operations.

julia> s = S"XXX
              ZZI
              IZZ"
 + XXX
@@ -134,7 +134,7 @@
 
 julia> s[1,1] = (true, false); s
 + X_
-+ __

There are no automatic checks for correctness (i.e. independence of all rows, commutativity of all rows, hermiticity of all rows). The rank (number of rows) is permitted to be less than the number of qubits (number of columns): canonilization, projection, etc. continue working in that case. To great extent this library uses the Stabilizer data structure simply as a tableau. This might be properly abstracted away in future versions.

See also: PauliOperator, canonicalize!

source
QuantumClifford.UnitaryPauliChannelType

A Pauli channel datastructure, mainly for use with StabMixture.

More convenient to use than PauliChannel when you know your Pauli channel is unitary.

julia> Tgate = UnitaryPauliChannel(
++ __

There are no automatic checks for correctness (i.e. independence of all rows, commutativity of all rows, hermiticity of all rows). The rank (number of rows) is permitted to be less than the number of qubits (number of columns): canonilization, projection, etc. continue working in that case. To great extent this library uses the Stabilizer data structure simply as a tableau. This might be properly abstracted away in future versions.

See also: PauliOperator, canonicalize!

source
QuantumClifford.UnitaryPauliChannelType

A Pauli channel datastructure, mainly for use with StabMixture.

More convenient to use than PauliChannel when you know your Pauli channel is unitary.

julia> Tgate = UnitaryPauliChannel(
            (I, Z),
            ((1+exp(im*π/4))/2, (1-exp(im*π/4))/2)
        )
@@ -149,7 +149,7 @@
  0.853553+0.0im | + _ | + _
  0.0+0.353553im | + _ | + Z
  0.0-0.353553im | + Z | + _
- 0.146447+0.0im | + Z | + Z
source
QuantumClifford.VerifyOpType

A "probe" to verify that the state of the qubits corresponds to a desired good_state, e.g. at the end of the execution of a circuit.

source
QuantumClifford.sMRZType

Measure a qubit in the Z basis and reset to the |0⟩ state.

It does not trace out the qubit!

As described below there is a difference between measuring the qubit (followed by setting it to a given known state) and "tracing out" the qubit. By reset here we mean "measuring and setting to a known state", not "tracing out".

julia> s = MixedDestabilizer(S"XXX ZZI IZZ") # |000⟩+|111⟩
+ 0.146447+0.0im | + Z | + Z
source
QuantumClifford.VerifyOpType

A "probe" to verify that the state of the qubits corresponds to a desired good_state, e.g. at the end of the execution of a circuit.

source
QuantumClifford.sMRZType

Measure a qubit in the Z basis and reset to the |0⟩ state.

It does not trace out the qubit!

As described below there is a difference between measuring the qubit (followed by setting it to a given known state) and "tracing out" the qubit. By reset here we mean "measuring and setting to a known state", not "tracing out".

julia> s = MixedDestabilizer(S"XXX ZZI IZZ") # |000⟩+|111⟩
 𝒟ℯ𝓈𝓉𝒶𝒷
 + Z__
 + _X_
@@ -199,7 +199,7 @@
 𝒮𝓉𝒶𝒷━
 + Z__
 - ZZ_
-- Z_Z

See also: Reset, sMZ

source
QuantumClifford.PauliErrorFunction

A convenient constructor for various types of Pauli errors, that can be used as circuit gates in simulations. Returns more specific types when necessary.

source
QuantumClifford.applybranchesFunction

Compute all possible new states after the application of the given operator. Reports the probability of each one of them. Deterministic (as it reports all branches of potentially random processes), part of the Perturbative Expansion interface.

source
QuantumClifford.applynoise!Function

A method modifying a given state by applying the corresponding noise model. It is non-deterministic, part of the Noise interface.

source
QuantumClifford.PauliErrorFunction

A convenient constructor for various types of Pauli errors, that can be used as circuit gates in simulations. Returns more specific types when necessary.

source
QuantumClifford.applybranchesFunction

Compute all possible new states after the application of the given operator. Reports the probability of each one of them. Deterministic (as it reports all branches of potentially random processes), part of the Perturbative Expansion interface.

source
QuantumClifford.applynoise!Function

A method modifying a given state by applying the corresponding noise model. It is non-deterministic, part of the Noise interface.

source
QuantumClifford.bellFunction

Prepare one or more Bell pairs (with optional phases).

julia> bell()
 + XX
 + ZZ
 
@@ -217,11 +217,11 @@
 - XX__
 + ZZ__
 - __XX
-- __ZZ
source
QuantumClifford.bigramMethod
bigram(
     state::QuantumClifford.AbstractStabilizer;
     clip
 ) -> Matrix{Int64}
-

Get the bigram of a tableau.

It is the list of endpoints of a tableau in the clipped gauge.

If clip=true (the default) the tableau is converted to the clipped gauge in-place before calculating the bigram. Otherwise, the clip gauge conversion is skipped (for cases where the input is already known to be in the correct gauge).

Introduced in (Nahum et al., 2017), with a more detailed explanation of the algorithm in (Li et al., 2019) and (Gullans et al., 2020).

See also: canonicalize_clip!

source
QuantumClifford.canonicalize!Method
canonicalize!(
+

Get the bigram of a tableau.

It is the list of endpoints of a tableau in the clipped gauge.

If clip=true (the default) the tableau is converted to the clipped gauge in-place before calculating the bigram. Otherwise, the clip gauge conversion is skipped (for cases where the input is already known to be in the correct gauge).

Introduced in (Nahum et al., 2017), with a more detailed explanation of the algorithm in (Li et al., 2019) and (Gullans et al., 2020).

See also: canonicalize_clip!

source
QuantumClifford.canonicalize_clip!Method
canonicalize_clip!(
     state::QuantumClifford.AbstractStabilizer;
     phases
 ) -> QuantumClifford.AbstractStabilizer
@@ -294,25 +294,25 @@
 + _XZX__
 - _ZYX_Z
 - __YZ_X
-- ____Z_

If phases=false is set, the canonicalization does not track the phases in the tableau, leading to a significant speedup.

Introduced in (Nahum et al., 2017), with a more detailed explanation of the algorithm in Appendix A of (Li et al., 2019)

See also: canonicalize!, canonicalize_rref!, canonicalize_gott!.

source
QuantumClifford.canonicalize_gott!Method

Inplace Gottesman canonicalization of a tableau.

This uses different canonical form from canonicalize!. It is used in the computation of the logical X and Z operators of a MixedDestabilizer.

It returns the (in place) modified state, the indices of the last pivot of both Gaussian elimination steps, and the permutations that have been used to put the X and Z tableaux in standard form.

Based on (Gottesman, 1997).

See also: canonicalize!, canonicalize_rref!

source
QuantumClifford.canonicalize_gott!Method

Inplace Gottesman canonicalization of a tableau.

This uses different canonical form from canonicalize!. It is used in the computation of the logical X and Z operators of a MixedDestabilizer.

It returns the (in place) modified state, the indices of the last pivot of both Gaussian elimination steps, and the permutations that have been used to put the X and Z tableaux in standard form.

Based on (Gottesman, 1997).

See also: canonicalize!, canonicalize_rref!

source
QuantumClifford.canonicalize_rref!Method
canonicalize_rref!(
     state::QuantumClifford.AbstractStabilizer,
     colindices;
     phases
 ) -> Tuple{QuantumClifford.AbstractStabilizer, Any}
-

Canonicalize a stabilizer (in place) along only some columns.

This uses different canonical form from canonicalize!. It also indexes in reverse in order to make its use in traceout! more efficient. Its use in traceout! is its main application.

It returns the (in place) modified state and the index of the last pivot.

Based on (Audenaert and Plenio, 2005).

See also: canonicalize!, canonicalize_gott!

source
QuantumClifford.commFunction

Check whether two operators commute.

0x0 if they commute, 0x1 if they anticommute.

julia> P"XX"*P"ZZ", P"ZZ"*P"XX"
+
source
QuantumClifford.commFunction

Check whether two operators commute.

0x0 if they commute, 0x1 if they anticommute.

julia> P"XX"*P"ZZ", P"ZZ"*P"XX"
 (- YY, - YY)
 
 julia> comm(P"ZZ", P"XX")
 0x00
 
 julia> comm(P"IZ", P"XX")
-0x01

See also: comm!

source
QuantumClifford.compactify_circuitMethod

Convert a list of gates to a more optimized "sum type" format which permits faster dispatch.

Generally, this should be called on a circuit before it is used in a simulation.

source
QuantumClifford.fastcolumnFunction

Convert a tableau to a memory layout that is fast for column operations.

In this layout a column of the tableau is stored (mostly) contiguously in memory. Due to bitpacking, e.g., packing 64 bits into a single UInt64, the memory layout is not perfectly contiguous, but it is still optimal given that some bitwrangling is required to extract a given bit.

See also: fastrow

source
QuantumClifford.fastrowFunction

Convert a tableau to a memory layout that is fast for row operations.

In this layout a Pauli string (a row of the tableau) is stored contiguously in memory.

See also: fastrow

source
QuantumClifford.generate!Method

Generate a Pauli operator by using operators from a given the Stabilizer.

It assumes the stabilizer is already canonicalized. It modifies the Pauli operator in place, generating it in reverse, up to a phase. That phase is left in the modified operator, which should be the identity up to a phase. Returns the new operator and the list of indices denoting the elements of stabilizer that were used for the generation.

julia> ghz = S"XXXX
+0x01

See also: comm!

source
QuantumClifford.compactify_circuitMethod

Convert a list of gates to a more optimized "sum type" format which permits faster dispatch.

Generally, this should be called on a circuit before it is used in a simulation.

source
QuantumClifford.fastcolumnFunction

Convert a tableau to a memory layout that is fast for column operations.

In this layout a column of the tableau is stored (mostly) contiguously in memory. Due to bitpacking, e.g., packing 64 bits into a single UInt64, the memory layout is not perfectly contiguous, but it is still optimal given that some bitwrangling is required to extract a given bit.

See also: fastrow

source
QuantumClifford.fastrowFunction

Convert a tableau to a memory layout that is fast for row operations.

In this layout a Pauli string (a row of the tableau) is stored contiguously in memory.

See also: fastrow

source
QuantumClifford.generate!Method

Generate a Pauli operator by using operators from a given the Stabilizer.

It assumes the stabilizer is already canonicalized. It modifies the Pauli operator in place, generating it in reverse, up to a phase. That phase is left in the modified operator, which should be the identity up to a phase. Returns the new operator and the list of indices denoting the elements of stabilizer that were used for the generation.

julia> ghz = S"XXXX
                ZZII
                IZZI
                IIZZ";
@@ -329,7 +329,7 @@
 true
 
 julia> generate!(P"XII",canonicalize!(S"XII")) === nothing
-false
source
QuantumClifford.ghzFunction

Prepare a GHZ state of n qubits.

julia> ghz()
 + XXX
 + ZZ_
 + _ZZ
@@ -342,7 +342,7 @@
 + XXXX
 + ZZ__
 + _ZZ_
-+ __ZZ
source
QuantumClifford.graphstateMethod

Convert any stabilizer state to a graph state

Graph states are a special type of entangled stabilizer states that can be represented by a graph. For a graph $G=(V,E)$ the corresponding stabilizers are $S_v = X_v \prod_{u ∈ N(v)} Z_u$. Notice that such tableau rows contain only a single X operator. There is a set of single qubit gates that converts any stabilizer state to a graph state.

This function returns the graph state corresponding to a stabilizer and the gates that might be necessary to convert the stabilizer into a state representable as a graph.

For a tableau stab you can convert it with:

graph, hadamard_idx, iphase_idx, flips_idx = graphstate()

where graph is the graph representation of stab, and the rest specifies the single-qubit gates converting stab to graph: hadamard_idx are the qubits that require a Hadamard gate (mapping X ↔ Z), iphase_idx are (different) qubits that require an inverse Phase gate (Y → X), and flips_idx are the qubits that require a phase flip (Pauli Z gate), after the previous two sets of gates.

julia> using Graphs
+true

See also: graph_gatesequence

source
QuantumClifford.graphstateMethod

Convert any stabilizer state to a graph state

Graph states are a special type of entangled stabilizer states that can be represented by a graph. For a graph $G=(V,E)$ the corresponding stabilizers are $S_v = X_v \prod_{u ∈ N(v)} Z_u$. Notice that such tableau rows contain only a single X operator. There is a set of single qubit gates that converts any stabilizer state to a graph state.

This function returns the graph state corresponding to a stabilizer and the gates that might be necessary to convert the stabilizer into a state representable as a graph.

For a tableau stab you can convert it with:

graph, hadamard_idx, iphase_idx, flips_idx = graphstate()

where graph is the graph representation of stab, and the rest specifies the single-qubit gates converting stab to graph: hadamard_idx are the qubits that require a Hadamard gate (mapping X ↔ Z), iphase_idx are (different) qubits that require an inverse Phase gate (Y → X), and flips_idx are the qubits that require a phase flip (Pauli Z gate), after the previous two sets of gates.

julia> using Graphs
 
 julia> s = S" XXX
               ZZ_
@@ -406,32 +406,32 @@
 1-element Vector{Int64}:
  3

The Graphs.jl library provides many graph-theory tools and the MakieGraphs.jl library provides plotting utilities for graphs.

You can directly call the graph constructor on a stabilizer, if you just want the graph and do not care about the Clifford operation necessary to convert an arbitrary state to a state representable as a graph:

julia> collect(edges( Graph(bell()) ))
 1-element Vector{Graphs.SimpleGraphs.SimpleEdge{Int64}}:
- Edge 1 => 2

For a version that does not copy the stabilizer, but rather performs transformations in-place, use graphstate!. It would perform canonicalize_gott! on its argument as it finds a way to convert it to a graph state.

source
QuantumClifford.logdotMethod

Logarithm of the inner product between to Stabilizer states.

If the result is nothing, the dot inner product is zero. Otherwise the inner product is 2^(-logdot/2).

The actual inner product can be computed with LinearAlgebra.dot.

Based on (Garcia et al., 2012).

source
QuantumClifford.pfmeasurementsMethod
pfmeasurements(frame::PauliFrame) -> Any
-

Returns the measurement results for each frame in the PauliFrame instance.

Relative measurements

The return measurements are relative to the reference measurements, i.e. they only say whether the reference measurements have been flipped in the given frame.

source
QuantumClifford.pfmeasurementsMethod
pfmeasurements(register::Register, frame::PauliFrame) -> Any
-

Takes the references measurements from the given Register and applies the flips as prescribed by the PauliFrame relative measurements. The result is the actual (non-relative) measurement results for each frame.

source
QuantumClifford.pftrajectoriesMethod
pftrajectories(
+ Edge 1 => 2

For a version that does not copy the stabilizer, but rather performs transformations in-place, use graphstate!. It would perform canonicalize_gott! on its argument as it finds a way to convert it to a graph state.

source
QuantumClifford.logdotMethod

Logarithm of the inner product between to Stabilizer states.

If the result is nothing, the dot inner product is zero. Otherwise the inner product is 2^(-logdot/2).

The actual inner product can be computed with LinearAlgebra.dot.

Based on (Garcia et al., 2012).

source
QuantumClifford.pfmeasurementsMethod
pfmeasurements(frame::PauliFrame) -> Any
+

Returns the measurement results for each frame in the PauliFrame instance.

Relative measurements

The return measurements are relative to the reference measurements, i.e. they only say whether the reference measurements have been flipped in the given frame.

source
QuantumClifford.pfmeasurementsMethod
pfmeasurements(register::Register, frame::PauliFrame) -> Any
+

Takes the references measurements from the given Register and applies the flips as prescribed by the PauliFrame relative measurements. The result is the actual (non-relative) measurement results for each frame.

source
QuantumClifford.pftrajectoriesMethod
pftrajectories(
     circuit;
     trajectories,
     threads
 ) -> PauliFrame{Stabilizer{QuantumClifford.Tableau{Vector{UInt8}, LinearAlgebra.Adjoint{UInt64, Matrix{UInt64}}}}, Matrix{Bool}}
-

The main method for running Pauli frame simulations of circuits. See the other methods for lower level access.

Multithreading is enabled by default, but can be disabled by setting threads=false. Do not forget to launch Julia with multiple threads enabled, e.g. julia -t4, if you want to use multithreading.

See also: mctrajectories, petrajectories

source
QuantumClifford.pftrajectoriesMethod
pftrajectories(
+

The main method for running Pauli frame simulations of circuits. See the other methods for lower level access.

Multithreading is enabled by default, but can be disabled by setting threads=false. Do not forget to launch Julia with multiple threads enabled, e.g. julia -t4, if you want to use multithreading.

See also: mctrajectories, petrajectories

source
QuantumClifford.pftrajectoriesMethod
pftrajectories(
     register::Register,
     circuit;
     trajectories
 ) -> Tuple{Register, PauliFrame{Stabilizer{QuantumClifford.Tableau{Vector{UInt8}, LinearAlgebra.Adjoint{UInt64, Matrix{UInt64}}}}, Matrix{Bool}}}
-

For a given Register and circuit, simulates the reference circuit acting on the register and then also simulate numerous PauliFrame trajectories. Returns the register and the PauliFrame instance.

Use pfmeasurements to get the measurement results.

source
QuantumClifford.phasesMethod

The phases of a given tableau. It is a view, i.e. if you modify this array, the original tableau caries these changes.

source
QuantumClifford.prodphaseMethod

Get the phase of the product of two Pauli operators.

Phase is encoded as F(4) in the low qubits of an UInt8.

julia> P"ZZZ"*P"XXX"
+

For a given Register and circuit, simulates the reference circuit acting on the register and then also simulate numerous PauliFrame trajectories. Returns the register and the PauliFrame instance.

Use pfmeasurements to get the measurement results.

source
QuantumClifford.phasesMethod

The phases of a given tableau. It is a view, i.e. if you modify this array, the original tableau caries these changes.

source
QuantumClifford.prodphaseMethod

Get the phase of the product of two Pauli operators.

Phase is encoded as F(4) in the low qubits of an UInt8.

julia> P"ZZZ"*P"XXX"
 -iYYY
 
 julia> prodphase(P"ZZZ", P"XXX")
 0x03
 
 julia> prodphase(P"XXX", P"ZZZ")
-0x01
source
QuantumClifford.random_pauliFunction

A random Pauli operator on n qubits.

Use nophase=false to randomize the phase. Use realphase=false to get operators with phases including ±i.

Optionally, a "flip" probability p can be provided specified, in which case each bit is set to I with probability 1-p and to X or Y or Z with probability p. Useful for simulating unbiased Pauli noise.

See also random_pauli!

source
QuantumClifford.stabilizerplotFunction

A Makie.jl recipe for pictorial representation of a tableau.

Requires a Makie.jl backend to be loaded, e.g. using CairoMakie.

Alternatively, you can use the Plots.jl plotting ecosystem, e.g. using Plots; plot(S"XXX ZZZ").

Consult the documentation for more details on visualization options.

source
QuantumClifford.stabilizerplot_axisFunction

A Makie.jl recipe for pictorial representation of a tableau.

Requires a Makie.jl backend to be loaded, e.g. using CairoMakie.

Alternatively, you can use the Plots.jl plotting ecosystem, e.g. using Plots; plot(S"XXX ZZZ").

Consult the documentation for more details on visualization options.

source
QuantumClifford.random_pauliFunction

A random Pauli operator on n qubits.

Use nophase=false to randomize the phase. Use realphase=false to get operators with phases including ±i.

Optionally, a "flip" probability p can be provided specified, in which case each bit is set to I with probability 1-p and to X or Y or Z with probability p. Useful for simulating unbiased Pauli noise.

See also random_pauli!

source
QuantumClifford.stabilizerplotFunction

A Makie.jl recipe for pictorial representation of a tableau.

Requires a Makie.jl backend to be loaded, e.g. using CairoMakie.

Alternatively, you can use the Plots.jl plotting ecosystem, e.g. using Plots; plot(S"XXX ZZZ").

Consult the documentation for more details on visualization options.

source
QuantumClifford.stabilizerplot_axisFunction

A Makie.jl recipe for pictorial representation of a tableau.

Requires a Makie.jl backend to be loaded, e.g. using CairoMakie.

Alternatively, you can use the Plots.jl plotting ecosystem, e.g. using Plots; plot(S"XXX ZZZ").

Consult the documentation for more details on visualization options.

source
QuantumClifford.xbitMethod

Extract as a new bit array the X part of the UInt array of packed qubits of a given Pauli operator.

source
QuantumClifford.zbitMethod

Extract as a new bit array the Z part of the UInt array of packed qubits of a given Pauli operator.

source
QuantumInterface.apply!Function

In QuantumClifford the apply! function is used to apply any quantum operation to a stabilizer state, including unitary Clifford operations, Pauli measurements, and noise. Thus, this function may result in a random/stochastic result (e.g. with measurements or noise).

source
QuantumClifford.xbitMethod

Extract as a new bit array the X part of the UInt array of packed qubits of a given Pauli operator.

source
QuantumClifford.zbitMethod

Extract as a new bit array the Z part of the UInt array of packed qubits of a given Pauli operator.

source
QuantumInterface.apply!Function

In QuantumClifford the apply! function is used to apply any quantum operation to a stabilizer state, including unitary Clifford operations, Pauli measurements, and noise. Thus, this function may result in a random/stochastic result (e.g. with measurements or noise).

source
QuantumInterface.embedMethod

Embed a Pauli operator in a larger Pauli operator.

julia> embed(5, 3, P"-Y")
 - __Y__
 
 julia> embed(5, (3,5), P"-YX")
-- __Y_X
source
QuantumInterface.entanglement_entropyFunction

Get bipartite entanglement entropy of a subsystem

Defined as entropy of the reduced density matrix.

It can be calculated with multiple different algorithms, the most performant one depending on the particular case.

Currently implemented are the :clip (clipped gauge), :graph (graph state), and :rref (Gaussian elimination) algorithms. Benchmark your particular case to choose the best one.

source
QuantumInterface.entanglement_entropyMethod

Get bipartite entanglement entropy by first converting the state to a graph and computing the rank of the adjacency matrix.

Based on "Entanglement in graph states and its applications".

source
QuantumInterface.expectMethod
expect(p::PauliOperator, st::AbstractStabilizer)

Compute the expectation value of a Pauli operator p on a stabilizer state st. This function will allocate a temporary copy of the stabilizer state st.

source
QuantumInterface.entanglement_entropyFunction

Get bipartite entanglement entropy of a subsystem

Defined as entropy of the reduced density matrix.

It can be calculated with multiple different algorithms, the most performant one depending on the particular case.

Currently implemented are the :clip (clipped gauge), :graph (graph state), and :rref (Gaussian elimination) algorithms. Benchmark your particular case to choose the best one.

source
QuantumInterface.entanglement_entropyMethod

Get bipartite entanglement entropy by first converting the state to a graph and computing the rank of the adjacency matrix.

Based on "Entanglement in graph states and its applications".

source
QuantumInterface.expectMethod
expect(p::PauliOperator, st::AbstractStabilizer)

Compute the expectation value of a Pauli operator p on a stabilizer state st. This function will allocate a temporary copy of the stabilizer state st.

source
QuantumInterface.project!Method
project!(
     state::MixedStabilizer,
     pauli::PauliOperator;
     phases
@@ -556,47 +556,47 @@
 julia> project!(ms, P"IIY")[1]
 + X__
 + _Z_
-+ __Y

Similarly to project! on Stabilizer, this function has cubic complexity when the Pauli operator commutes with all rows of the tableau. Most of the time it is better to simply use MixedDestabilizer representation.

Unlike other project! methods, this one does not allow for keep_result=false, as the correct rank or anticommutation index can not be calculated without the expensive (cubic) canonicalization operation required by keep_result=true.

See the "Datastructure Choice" section in the documentation for more details.

See also: projectX!, projectY!, projectZ!.

source
QuantumInterface.reset_qubits!Method
reset_qubits!(
++ __Y

Similarly to project! on Stabilizer, this function has cubic complexity when the Pauli operator commutes with all rows of the tableau. Most of the time it is better to simply use MixedDestabilizer representation.

Unlike other project! methods, this one does not allow for keep_result=false, as the correct rank or anticommutation index can not be calculated without the expensive (cubic) canonicalization operation required by keep_result=true.

See the "Datastructure Choice" section in the documentation for more details.

See also: projectX!, projectY!, projectZ!.

source
QuantumInterface.reset_qubits!Method
reset_qubits!(
     s::Stabilizer,
     newstate,
     qubits;
     phases
 ) -> Any
-

Reset a given set of qubits to be in the state newstate. These qubits are traced out first, which could lead to "nonlocal" changes in the tableau.

source
QuantumInterface.traceout!Method
traceout!(
+

Reset a given set of qubits to be in the state newstate. These qubits are traced out first, which could lead to "nonlocal" changes in the tableau.

source

Private API

Private Implementation Details

These functions are used internally by the library and might be drastically modified or deleted without warning or deprecation.

QuantumClifford.TableauType

Internal Tableau type for storing a list of Pauli operators in a compact form. No special semantic meaning is attached to this type, it is just a convenient way to store a list of Pauli operators. E.g. it is not used to represent a stabilizer state, or a stabilizer group, or a Clifford circuit.

source

Private API

Private Implementation Details

These functions are used internally by the library and might be drastically modified or deleted without warning or deprecation.

QuantumClifford.TableauType

Internal Tableau type for storing a list of Pauli operators in a compact form. No special semantic meaning is attached to this type, it is just a convenient way to store a list of Pauli operators. E.g. it is not used to represent a stabilizer state, or a stabilizer group, or a Clifford circuit.

source
Base.invMethod
inv(
     c::CliffordOperator;
     phases
 ) -> CliffordOperator{QuantumClifford.Tableau{Vector{UInt8}, Matrix{UInt64}}}
-

Inverse of a CliffordOperator

source
QuantumClifford._remove_rowcol!Method

Unexported low-level function that removes a row (by shifting all rows up as necessary)

Because MixedDestabilizer is not mutable we return a new MixedDestabilizer with the same (modified) xzs array.

Used on its own, this function will break invariants. Meant to be used with projectremove!.

source
QuantumClifford._rowmove!Method

Unexported low-level function that moves row i to row j.

Used on its own, this function will break invariants. Meant to be used in _remove_rowcol!.

source
QuantumClifford.applynoise_branchesFunction

Compute all possible new states after the application of the given noise model. Reports the probability of each one of them. Deterministic (as it reports all branches of potentially random processes), part of the Noise interface.

source
QuantumClifford.initZ!Method
initZ!(frame::PauliFrame) -> PauliFrame
-

Inject random Z errors over all frames and qubits for the supplied PauliFrame with probability 0.5.

Calling this after initialization is essential for simulating any non-deterministic circuit. It is done automatically by most PauliFrame constructors.

source
QuantumClifford._remove_rowcol!Method

Unexported low-level function that removes a row (by shifting all rows up as necessary)

Because MixedDestabilizer is not mutable we return a new MixedDestabilizer with the same (modified) xzs array.

Used on its own, this function will break invariants. Meant to be used with projectremove!.

source
QuantumClifford._rowmove!Method

Unexported low-level function that moves row i to row j.

Used on its own, this function will break invariants. Meant to be used in _remove_rowcol!.

source
QuantumClifford.applynoise_branchesFunction

Compute all possible new states after the application of the given noise model. Reports the probability of each one of them. Deterministic (as it reports all branches of potentially random processes), part of the Noise interface.

source
QuantumClifford.initZ!Method
initZ!(frame::PauliFrame) -> PauliFrame
+

Inject random Z errors over all frames and qubits for the supplied PauliFrame with probability 0.5.

Calling this after initialization is essential for simulating any non-deterministic circuit. It is done automatically by most PauliFrame constructors.

source
QuantumClifford.make_sumtype_methodFunction

``` julia> makesumtypemethod([sCNOT], :apply!, (:s,)) quote function QuantumClifford.apply!(s, g::CompactifiedGate) @cases g begin sCNOT(q1, q2) => apply!(s, sCNOT(q1, q2)) end end end

source
QuantumClifford.make_sumtype_methodFunction

``` julia> makesumtypemethod([sCNOT], :apply!, (:s,)) quote function QuantumClifford.apply!(s, g::CompactifiedGate) @cases g begin sCNOT(q1, q2) => apply!(s, sCNOT(q1, q2)) end end end

source
QuantumClifford.projectremoverand!Method

Unexported low-level function that projects a qubit and returns the result while making the tableau smaller by a qubit.

Because MixedDestabilizer is not mutable we return a new MixedDestabilizer with the same (modified) xzs array.

source
QuantumClifford.remove_column!Method

Unexported low-level function that removes a column (by shifting all columns to the right of the target by one step to the left)

Because Tableau is not mutable we return a new Tableau with the same (modified) xzs array.

source
QuantumClifford.rowdecomposeMethod

Decompose a Pauli $P$ in terms of stabilizer and destabilizer rows from a given tableaux.

For given tableaux of rows destabilizer rows $\{d_i\}$ and stabilizer rows $\{s_i\}$, there are boolean vectors $b$ and $c$ such that $P = i^p \prod_i d_i^{b_i} \prod_i s_i^{c_i}$.

This function returns p, b, c.

julia> s = MixedDestabilizer(ghz(2))
+end)
source
QuantumClifford.projectremoverand!Method

Unexported low-level function that projects a qubit and returns the result while making the tableau smaller by a qubit.

Because MixedDestabilizer is not mutable we return a new MixedDestabilizer with the same (modified) xzs array.

source
QuantumClifford.remove_column!Method

Unexported low-level function that removes a column (by shifting all columns to the right of the target by one step to the left)

Because Tableau is not mutable we return a new Tableau with the same (modified) xzs array.

source
QuantumClifford.rowdecomposeMethod

Decompose a Pauli $P$ in terms of stabilizer and destabilizer rows from a given tableaux.

For given tableaux of rows destabilizer rows $\{d_i\}$ and stabilizer rows $\{s_i\}$, there are boolean vectors $b$ and $c$ such that $P = i^p \prod_i d_i^{b_i} \prod_i s_i^{c_i}$.

This function returns p, b, c.

julia> s = MixedDestabilizer(ghz(2))
 𝒟ℯ𝓈𝓉𝒶𝒷
 + Z_
 + _X
@@ -608,7 +608,7 @@
 (3, Bool[1, 0], Bool[1, 1])
 
 julia> im^3 * P"Z_" * P"XX" * P"ZZ"
-+ XY
source
QuantumClifford.to_cpuFunction

copies the memory content of the object to CPU

You can only use this function if CUDA.jl is imported

For more advanced users to_cpu(data, element_type) will reinterpret elements of data and converts them to element_type. For example based on your CPU architecture, if working with matrices of UInt32 is faster than UInt64, you can use to_cpu(data, UInt32)

julia> using QuantumClifford: to_cpu, to_gpu
++ XY
source
QuantumClifford.to_cpuFunction

copies the memory content of the object to CPU

You can only use this function if CUDA.jl is imported

For more advanced users to_cpu(data, element_type) will reinterpret elements of data and converts them to element_type. For example based on your CPU architecture, if working with matrices of UInt32 is faster than UInt64, you can use to_cpu(data, UInt32)

julia> using QuantumClifford: to_cpu, to_gpu
 
 julia> using CUDA # without this import, to_cpu, to_gpu are just function
 
@@ -631,7 +631,7 @@
 julia> pf_gpu = to_gpu(PauliFrame(1000, 2, 2));
 julia> circuit = [sMZ(1, 1), sHadamard(2), sMZ(2, 2)];
 julia> pftrajectories(pf_gpu, circuit);
-julia> measurements = to_cpu(pf_gpu.measurements);

See also: to_gpu

source
QuantumClifford.to_gpuFunction

copies the memory content of the object to GPU

You can only use this function if CUDA.jl is imported

For more advanced users to_gpu(data, element_type) will reinterpret elements of data and converts them to element_type. For example based on your GPU architecture, if working with matrices of UInt64 is faster than UInt32, you can use to_gpu(data, UInt64)

julia> using QuantumClifford: to_cpu, to_gpu
+julia> measurements = to_cpu(pf_gpu.measurements);

See also: to_gpu

source
QuantumClifford.to_gpuFunction

copies the memory content of the object to GPU

You can only use this function if CUDA.jl is imported

For more advanced users to_gpu(data, element_type) will reinterpret elements of data and converts them to element_type. For example based on your GPU architecture, if working with matrices of UInt64 is faster than UInt32, you can use to_gpu(data, UInt64)

julia> using QuantumClifford: to_cpu, to_gpu
 
 julia> using CUDA # without this import, to_cpu, to_gpu are just function
 
@@ -654,4 +654,4 @@
 julia> pf_gpu = to_gpu(PauliFrame(1000, 2, 2));
 julia> circuit = [sMZ(1, 1), sHadamard(2), sMZ(2, 2)];
 julia> pftrajectories(pf_gpu, circuit);
-julia> measurements = to_cpu(pf_gpu.measurements);

See also: to_cpu

source
+julia> measurements = to_cpu(pf_gpu.measurements);

See also: to_cpu

source
QuantumClifford.trusted_rankFunction

A "trusted" rank which returns rank(state) for Mixed[De]Stabilizer and length(state) for [De]Stabilizer.

source
QuantumClifford.zero!Method

Zero-out a given row of a Tableau

source
QuantumClifford.zero!Method

Zero-out the phases and single-qubit operators in a PauliOperator

source
QuantumClifford.@qubitop1Macro

Macro used to define single qubit symbolic gates and their qubit_kernel methods.

source
QuantumClifford.@qubitop2Macro

Macro used to define 2-qubit symbolic gates and their qubit_kernel methods.

source
QuantumClifford.@valbooldispatchMacro

Turns f(Val(x)) into x ? f(Val(true)) : f(Val(false)) in order to avoid dynamic dispatch

See discourse discussion

source
diff --git a/dev/ECC_API/index.html b/dev/ECC_API/index.html index 7d97a9cfd..625f3e85a 100644 --- a/dev/ECC_API/index.html +++ b/dev/ECC_API/index.html @@ -2,7 +2,7 @@ API · QuantumClifford.jl

Full ECC API (autogenerated)

QuantumClifford.ECC.CSSType

An arbitrary CSS error correcting code defined by its X and Z checks.

julia> CSS([0 1 1 0; 1 1 0 0], [1 1 1 1]) |> parity_checks
 + _XX_
 + XX__
-+ ZZZZ
source
QuantumClifford.ECC.ShorSyndromeECCSetupType

Configuration for ECC evaluators that simulate the Shor-style syndrome measurement (without a flag qubit).

The simulated circuit includes:

  • perfect noiseless encoding (encoding and its fault tolerance are not being studied here)
  • one round of "memory noise" after the encoding but before the syndrome measurement
  • perfect preparation of entangled ancillary qubits
  • noisy Shor-style syndrome measurement (only two-qubit gate noise)
  • noiseless "logical state measurement" (providing the comparison data when evaluating the decoder)

See also: CommutationCheckECCSetup, NaiveSyndromeECCSetup

source
QuantumClifford.ECC.TableDecoderType

A simple look-up table decoder for error correcting codes.

The lookup table contains only weight=1 errors, thus it is small, but at best it provides only for distance=3 decoding.

The size of the lookup table would grow exponentially quickly for higher distances.

source
QuantumClifford.ECC.evaluate_decoderMethod

Evaluate the performance of an error-correcting circuit.

This method requires you give the circuit that performs both syndrome measurements and (probably noiseless) logical state measurements. The faults matrix that translates an error vector into corresponding logical errors is necessary as well.

This is a relatively barebones method that assumes the user prepares necessary circuits, etc. It is a method that is used internally by more user-frienly methods providing automatic conversion of codes and noise models to the necessary noisy circuits.

source
QuantumClifford.ECC.faults_matrixMethod

Error-to-logical-observable map (a.k.a. fault matrix) of a code.

For a code with n physical qubits and k logical qubits this function returns a 2k × 2n binary matrix O such that O[i,j] is true if the logical observable of index i is flipped by the single physical qubit error of index j. Indexing is such that:

  • O[1:k,:] is the error-to-logical-X-observable map (logical X observable, i.e. triggered by logical Z errors)
  • O[k+1:2k,:] is the error-to-logical-Z-observable map
  • O[:,1:n] is the X-physical-error-to-logical-observable map
  • O[n+1:2n,:] is the Z-physical-error-to-logical-observable map

E.g. for k=1, n=10, then if O[2,5] is true, then the logical Z observable is flipped by a X₅ error; and if O[1,12] is true, then the logical X observable is flipped by a Z₂ error.

Of note is that there is a lot of freedom in choosing the logical operations! A logical operator multiplied by a stabilizer operator is still a logical operator. Similarly there is a different fault matrix for each choice of logical operators. But once the logical operators are picked, the fault matrix is fixed.

Below we show an example that uses the Shor code. While it is not the smallest code, it is a convenient choice to showcase the importance of the fault matrix when dealing with degenerate codes where a correction operation and an error do not need to be the same.

First, consider a single-qubit error, potential correction operations, and their effect on the Shor code:

julia> using QuantumClifford.ECC: faults_matrix, Shor9
++ ZZZZ
source
QuantumClifford.ECC.ShorSyndromeECCSetupType

Configuration for ECC evaluators that simulate the Shor-style syndrome measurement (without a flag qubit).

The simulated circuit includes:

  • perfect noiseless encoding (encoding and its fault tolerance are not being studied here)
  • one round of "memory noise" after the encoding but before the syndrome measurement
  • perfect preparation of entangled ancillary qubits
  • noisy Shor-style syndrome measurement (only two-qubit gate noise)
  • noiseless "logical state measurement" (providing the comparison data when evaluating the decoder)

See also: CommutationCheckECCSetup, NaiveSyndromeECCSetup

source
QuantumClifford.ECC.TableDecoderType

A simple look-up table decoder for error correcting codes.

The lookup table contains only weight=1 errors, thus it is small, but at best it provides only for distance=3 decoding.

The size of the lookup table would grow exponentially quickly for higher distances.

source
QuantumClifford.ECC.evaluate_decoderMethod

Evaluate the performance of an error-correcting circuit.

This method requires you give the circuit that performs both syndrome measurements and (probably noiseless) logical state measurements. The faults matrix that translates an error vector into corresponding logical errors is necessary as well.

This is a relatively barebones method that assumes the user prepares necessary circuits, etc. It is a method that is used internally by more user-frienly methods providing automatic conversion of codes and noise models to the necessary noisy circuits.

source
QuantumClifford.ECC.faults_matrixMethod

Error-to-logical-observable map (a.k.a. fault matrix) of a code.

For a code with n physical qubits and k logical qubits this function returns a 2k × 2n binary matrix O such that O[i,j] is true if the logical observable of index i is flipped by the single physical qubit error of index j. Indexing is such that:

  • O[1:k,:] is the error-to-logical-X-observable map (logical X observable, i.e. triggered by logical Z errors)
  • O[k+1:2k,:] is the error-to-logical-Z-observable map
  • O[:,1:n] is the X-physical-error-to-logical-observable map
  • O[n+1:2n,:] is the Z-physical-error-to-logical-observable map

E.g. for k=1, n=10, then if O[2,5] is true, then the logical Z observable is flipped by a X₅ error; and if O[1,12] is true, then the logical X observable is flipped by a Z₂ error.

Of note is that there is a lot of freedom in choosing the logical operations! A logical operator multiplied by a stabilizer operator is still a logical operator. Similarly there is a different fault matrix for each choice of logical operators. But once the logical operators are picked, the fault matrix is fixed.

Below we show an example that uses the Shor code. While it is not the smallest code, it is a convenient choice to showcase the importance of the fault matrix when dealing with degenerate codes where a correction operation and an error do not need to be the same.

First, consider a single-qubit error, potential correction operations, and their effect on the Shor code:

julia> using QuantumClifford.ECC: faults_matrix, Shor9
 
 julia> state = MixedDestabilizer(Shor9())
 𝒟ℯ𝓈𝓉𝒶𝒷━━━━━
@@ -125,7 +125,7 @@
 julia> O * stab_to_gf2(bad_Z₆Z₉)
 2-element Vector{Int64}:
  1
- 0

While its use in this situation is rather contrived, the fault matrix is incredibly useful when running large scale simulations in which we want a separate fast error sampling process, (e.g. with Pauli frames) and a syndrome decoding process, without coupling between them. We just gather all our syndrome measurement and logical observables from the Pauli frame simulations, and then use them with the fault matrix in the syndrome decoding simulation.

source
QuantumClifford.ECC.isdegenerateFunction

Check if the code is degenerate with respect to a given set of error or with respect to all "up to d physical-qubit" errors (defaulting to d=1).

julia> using QuantumClifford.ECC
+ 0

While its use in this situation is rather contrived, the fault matrix is incredibly useful when running large scale simulations in which we want a separate fast error sampling process, (e.g. with Pauli frames) and a syndrome decoding process, without coupling between them. We just gather all our syndrome measurement and logical observables from the Pauli frame simulations, and then use them with the fault matrix in the syndrome decoding simulation.

source
QuantumClifford.ECC.isdegenerateFunction

Check if the code is degenerate with respect to a given set of error or with respect to all "up to d physical-qubit" errors (defaulting to d=1).

julia> using QuantumClifford.ECC
 
 julia> isdegenerate(Shor9(), [single_z(9,1), single_z(9,2)])
 true
@@ -137,4 +137,4 @@
 false
 
 julia> isdegenerate(Steane7(), 2)
-true
source
QuantumClifford.ECC.naive_encoding_circuitMethod

Encoding physical qubits into a larger logical code.

The initial physical qubits to be encoded have to be at indices n-k+1:n.

Encoding circuits are not fault-tolerant

Encoding circuits are not fault-tolerant, and thus should not be used in practice. Instead, you should measure the stabilizers of the code and the logical observables, thus projecting into the code space (which can be fault-tolerant).

The canonicalization operation performed on the code may permute the qubits (see canonicalize_gott!). That permutation is corrected for with SWAP gates by default (controlled by the undoperm keyword argument).

Based on (Cleve and Gottesman, 1997) and (Gottesman, 1997), however it seems the published algorithm has some errors. Consult the erratum, as well as the more recent (Grassl, 2002) and (Grassl, 2011), and be aware that this implementation also uses H instead of Z gates.

source
QuantumClifford.ECC.naive_syndrome_circuitFunction

Generate the non-fault-tolerant stabilizer measurement cicuit for a given code instance or parity check tableau.

Use the ancillary_index and bit_index arguments to offset where the corresponding part the circuit starts.

Returns the circuit, the number of ancillary qubits that were added, and a list of bit indices that will store the measurement results.

See also: shor_syndrome_circuit

source
QuantumClifford.ECC.shor_syndrome_circuitFunction

Generate the Shor fault-tolerant stabilizer measurement cicuit for a given code instance or parity check tableau.

Use the ancillary_index and bit_index arguments to offset where the corresponding part the circuit starts. Ancillary qubits

Returns:

  • The ancillary cat state preparation circuit.
  • The Shor syndrome measurement circuit.
  • The number of ancillary qubits that were added.
  • The list of bit indices that store the final measurement results.

See also: naive_syndrome_circuit

source
+truesource
QuantumClifford.ECC.naive_encoding_circuitMethod

Encoding physical qubits into a larger logical code.

The initial physical qubits to be encoded have to be at indices n-k+1:n.

Encoding circuits are not fault-tolerant

Encoding circuits are not fault-tolerant, and thus should not be used in practice. Instead, you should measure the stabilizers of the code and the logical observables, thus projecting into the code space (which can be fault-tolerant).

The canonicalization operation performed on the code may permute the qubits (see canonicalize_gott!). That permutation is corrected for with SWAP gates by default (controlled by the undoperm keyword argument).

Based on (Cleve and Gottesman, 1997) and (Gottesman, 1997), however it seems the published algorithm has some errors. Consult the erratum, as well as the more recent (Grassl, 2002) and (Grassl, 2011), and be aware that this implementation also uses H instead of Z gates.

source
QuantumClifford.ECC.naive_syndrome_circuitFunction

Generate the non-fault-tolerant stabilizer measurement cicuit for a given code instance or parity check tableau.

Use the ancillary_index and bit_index arguments to offset where the corresponding part the circuit starts.

Returns the circuit, the number of ancillary qubits that were added, and a list of bit indices that will store the measurement results.

See also: shor_syndrome_circuit

source
QuantumClifford.ECC.parity_checksFunction

Parity check tableau of a code.

See also: parity_checks_x and parity_checks_z

source
QuantumClifford.ECC.parity_checks_xMethod

Parity check boolean matrix of a code (only the X entries in the tableau, i.e. the checks for Z errors).

Only CSS codes have this method.

See also: parity_checks

source
QuantumClifford.ECC.parity_checks_zMethod

Parity check boolean matrix of a code (only the Z entries in the tableau, i.e. the checks for X errors).

Only CSS codes have this method.

See also: parity_checks

source
QuantumClifford.ECC.rateMethod

The rate of a code.

source
QuantumClifford.ECC.shor_syndrome_circuitFunction

Generate the Shor fault-tolerant stabilizer measurement cicuit for a given code instance or parity check tableau.

Use the ancillary_index and bit_index arguments to offset where the corresponding part the circuit starts. Ancillary qubits

Returns:

  • The ancillary cat state preparation circuit.
  • The Shor syndrome measurement circuit.
  • The number of ancillary qubits that were added.
  • The list of bit indices that store the final measurement results.

See also: naive_syndrome_circuit

source
diff --git a/dev/ECC_evaluating/987beb83.png b/dev/ECC_evaluating/987beb83.png new file mode 100644 index 0000000000000000000000000000000000000000..33157e9f53f1554030fb37e9237011d2507c3d3e GIT binary patch literal 42702 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 G*Z%|1PVFxM literal 0 HcmV?d00001 diff --git a/dev/ECC_evaluating/a228365f.png b/dev/ECC_evaluating/a228365f.png deleted file mode 100644 index a052b7be0697ddcd37fbe013384f85519f758b32..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 56139 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 z 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 DG9XqR diff --git a/dev/ECC_evaluating/ae335f84.png b/dev/ECC_evaluating/ae335f84.png deleted file mode 100644 index 2d2ef72c126027f0b1fb7ff1219eb1e7414b8fc9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 42907 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�eGcJ(SZo+OSEI4@Ai%@-tEh!YIksRiF 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 block output

Testing out the toric code with a decoder provided by the python package pymatching (provided in julia by the meta package PyQDecoders.jl).

import PyQDecoders
+make_decoder_figure(mem_errors, results, "Shor's code with a lookup table decoder")
Example block output

Testing out the toric code with a decoder provided by the python package pymatching (provided in julia by the meta package PyQDecoders.jl).

import PyQDecoders
 
 mem_errors = 0.001:0.005:0.1
 codes = [Toric(4,4), Toric(6,6)]
@@ -53,4 +53,4 @@
     end
 end
 
-make_decoder_figure(mem_errors, results, "Toric code with a MWPM decoder")
Example block output +make_decoder_figure(mem_errors, results, "Toric code with a MWPM decoder")Example block output diff --git a/dev/allops/index.html b/dev/allops/index.html index ded36889c..f18b16fd2 100644 --- a/dev/allops/index.html +++ b/dev/allops/index.html @@ -10,4 +10,4 @@ noise = UnbiasedUncorrelatedNoise(ε) noisy_gate = NoisyGate(SparseGate(tCNOT, [2,4]), noise)Example block output

In circuit diagrams the noise is not depicted, but after each application of the gate defined in noisy_gate, a noise operator will also be applied. The example above is of Pauli Depolarization implemented by UnbiasedUncorrelatedNoise.

One can also apply only the noise operator by using NoiseOp which acts only on specified qubits. Or alternatively, one can use NoiseOpAll in order to apply noise to all qubits.

[NoiseOp(noise, [4,5]), NoiseOpAll(noise)]
Example block output

The machinery behind noise processes and different types of noise is detailed in the section on noise

Coincidence Measurements

Global parity measurements involving single-qubit projections and classical communication are implemented with BellMeasurement. One needs to specify the axes of measurement and the qubits being measured. If the parity is trivial, the circuit continues, if the parity is non-trivial, the circuit ends and reports a detected failure. This operator is frequently used in the simulation of entanglement purification.

BellMeasurement([sMX(1), sMY(3), sMZ(4)])
Example block output

There is also NoisyBellMeasurement that takes the bit-flip probability of a single-qubit measurement as a third argument.

Stabilizer Measurements

A measurement over one or more qubits can also be performed, e.g., a direct stabilizer measurement on multiple qubits without the use of ancillary qubits. When applied to multiple qubits, this differs from BellMeasurement as it performs a single projection, unlike BellMeasurement which performs a separate projection for every single qubit involved. This measurement is implemented in PauliMeasurement which requires a Pauli operator on which to project and the index of the classical bit in which to store the result. Alternatively, there are sMX, sMZ, sMY if you are measuring a single qubit.

[PauliMeasurement(P"XYZ", 1), sMZ(2, 2)]
Example block output

Reset Operations

The Reset operations lets you trace out the specified qubits and set their state to a specific tableau.

new_state = random_stabilizer(3)
 qubit_indices = [1,2,3]
-Reset(new_state, qubit_indices)
Example block output

It can be done anywhere in a circuit, not just at the beginning.

+Reset(new_state, qubit_indices)Example block output

It can be done anywhere in a circuit, not just at the beginning.

diff --git a/dev/canonicalization/03a5321d.png b/dev/canonicalization/03a5321d.png deleted file mode 100644 index 64aebb3d059bce179ad11f753999e366f90f15bf..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 9447 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 Qa)Eqn<9M8O^vt#Y1t$e=`Tzg` diff --git a/dev/canonicalization/18569f0a.png b/dev/canonicalization/18569f0a.png new file mode 100644 index 0000000000000000000000000000000000000000..0601754bb238b726f80b49bde7242b58e62f01b2 GIT binary patch literal 9497 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 Dk981O literal 0 HcmV?d00001 diff --git a/dev/canonicalization/b9c71a95.png b/dev/canonicalization/b9c71a95.png deleted file mode 100644 index 0b267a2d63a662f88268fd6a1cd06eddd647bf84..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 9703 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 jX=t=S)xjO&va&J?=`M2+ltG<> zlclgYn1PuaOpTB!dgrlneHKb16XLWA7>2FrW4kgkGEfp1)EjtLM2lM0p;d8JZQRE! zq-Hvc6espnPLvy4*kb+t39m8&X0x=itdetW_^n&6_P^BbFo|gCKl^fHaBwh%CgFFW zK%<>h@(e4d0itq}Ld#~%-Y+jN&+9if8qPU+yIP|Vve|45-cB`W+_mOYq~cXpz`2p@ z6q=MFrE?XmN=-kGuGJ`geSI-_7Z;Z^KZV+x%!_TfWPyyIq_HblB8F0APfj{3L=QxD zc;e(QeEskR`|=bb+;@vFi0UNm&9;fZ&|=?w@1}khuG-D!*PF{*$f_2uJdb;t;yL}) zk_ls&uvT+FJZadPsjOx8`X!Dy;N-nbb#H%PGj*Nr-@U|XX0nprt)`i zx3#v0s53W#5IKsD&oBcZPfz&mK?=>-Ha<}3zq32w;*!AB)ri=M2f1NqDB$V@zz}xk zOc7TM3ayo?rCct2x|0slnxNK-&QR#@zHpM4qm<>v)S4n;MxZyGKsUBano4(|#%tt! zF(?6rJ34HNlQq~Qr_)%RvfyUW`^xB~&uYQv1iIZVJf`X$0;bbh9GOg(%~Xx#=jWr7 zc$PYZ_rd<1WdIez5~c>U>oldUz1`fxB95>$dhiM;7>l$*u4HPJStQl4ny3}kibYbT z)47&C8W3xhP}ogd@M(R!{HmS&Dg|dERY^K3Swx8&Nx~K48)CCJw7fyW`Kly!l!W$jgVhHBZW~t zl?!XT+jJ40Q;Zjc(_xks5t^`5J+Vs~GR2L}eLlvzmxWYlq)aBQf~3_%@!LUhsZJL& z=`banFci0{mY43ZFK;XDWXJ6=^szY0WC+0rMIa9G_hO^kHiMHifP+NGvsi*v^l&jw zEdWK9bfCCGxIGWSPc5mxVL=$%fgwc^%U?#qrGuKRi%j?NlnN&#zWJ*X!-=4WK(?PnzlJAXW@31DolRJaM1mi#Kc> zNzKwaEUa(Wr8G3x8?GI7WA^wdpL-)8g>xAc)H%iMxDIh#F*mk1R45eY*_t2(XzeT- zowv2iFA*-r(c#9id1J|I7F4~nO&H5($}*UzDE-lIz7J3A?G2DgN|Ht#z=tGAOxhe= zLQ-cUL}L@k?xVVSO;ocu{ba&WTTmziiA25A1_mj zNm?as7q`qJ39*2tOvauqiEnbK zAdXGxN*hNoohanrP7I5RiW0(Fy4LCmn2toIHcy;XK%2V`Dr!&U*~gCU?2b&eup3Z3 zkRj=^CMQO8bA2Fq&kw_0r&GCFMoNG9`_nd<^EOktFTl$m`>~SWIJp1&k2Ovk^L+i1 TmBM8p7wFqv`*u>j0x$j(* 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 U)A6%3a0znC$oxd^ap&v*1$T7(y8r+H diff --git a/dev/canonicalization/ec31a5b5.png b/dev/canonicalization/ec31a5b5.png new file mode 100644 index 0000000000000000000000000000000000000000..3e617cf28023f5a5978fcf5b9f4cd7fcb046384f GIT binary patch literal 9379 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 VdsVS`0Z0$XsS{?##N$pk{|5q|`Go)g literal 0 HcmV?d00001 diff --git a/dev/canonicalization/ed23db7b.png b/dev/canonicalization/ed23db7b.png new file mode 100644 index 0000000000000000000000000000000000000000..94871f38f42be046a0a9a948e12d9cb1c73624d3 GIT binary patch literal 9302 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 literal 0 HcmV?d00001 diff --git a/dev/canonicalization/index.html b/dev/canonicalization/index.html index 4e8ed123e..0544be4e5 100644 --- a/dev/canonicalization/index.html +++ b/dev/canonicalization/index.html @@ -2,13 +2,13 @@ Canonicalization · QuantumClifford.jl

Canonicalization operations

Different types of canonicalization operations are implemented. All of them are types of Gaussian elimination.

canonicalize!

First do elimination on all X components and only then perform elimination on the Z components. Based on (Garcia et al., 2012). It is used in logdot for inner products of stabilizer states.

The final tableaux, if square should look like the following

If the tableaux is shorter than a square, the diagonals might not reach all the way to the right.

using QuantumClifford, CairoMakie
 f=Figure()
 stabilizerplot_axis(f[1,1], canonicalize!(random_stabilizer(20,30)))
-f
Example block output

canonicalize_rref!

Cycle between elimination on X and Z for each qubit. Particularly useful for tracing out qubits. Based on (Audenaert and Plenio, 2005). For convenience reasons, the canonicalization starts from the bottom row, and you can specify as a second argument which columns to be canonicalized (useful for tracing out arbitrary qubits, e.g., in traceout!).

The tableau canonicalization is done in recursive steps, each one of which results in something akin to one of these three options

using QuantumClifford, CairoMakie
+f
Example block output

canonicalize_rref!

Cycle between elimination on X and Z for each qubit. Particularly useful for tracing out qubits. Based on (Audenaert and Plenio, 2005). For convenience reasons, the canonicalization starts from the bottom row, and you can specify as a second argument which columns to be canonicalized (useful for tracing out arbitrary qubits, e.g., in traceout!).

The tableau canonicalization is done in recursive steps, each one of which results in something akin to one of these three options

using QuantumClifford, CairoMakie
 f=Figure()
 stabilizerplot_axis(f[1,1], canonicalize_rref!(random_stabilizer(20,30),1:30)[1])
-f
Example block output

canonicalize_gott!

First do elimination on all X components and only then perform elimination on the Z components, but without touching the qubits that were eliminated during the X pass. Unlike other canonicalization operations, qubit columns are reordered, providing for a straight diagonal in each block. Particularly useful as certain blocks of the new created matrix are related to logical operations of the corresponding code, e.g. computing the logical X and Z operators of a MixedDestabilizer. Based on (Gottesman, 1997).

A canonicalized tableau would look like the following (the right-most block does not exist for square tableaux).

using QuantumClifford, CairoMakie
+f
Example block output

canonicalize_gott!

First do elimination on all X components and only then perform elimination on the Z components, but without touching the qubits that were eliminated during the X pass. Unlike other canonicalization operations, qubit columns are reordered, providing for a straight diagonal in each block. Particularly useful as certain blocks of the new created matrix are related to logical operations of the corresponding code, e.g. computing the logical X and Z operators of a MixedDestabilizer. Based on (Gottesman, 1997).

A canonicalized tableau would look like the following (the right-most block does not exist for square tableaux).

using QuantumClifford, CairoMakie
 f=Figure()
 stabilizerplot_axis(f[1,1], canonicalize_gott!(random_stabilizer(30))[1])
-f
Example block output

canonicalize_clip!

Convert to the "clipped" gauge of a stabilizer state resulting in a "river" of non-identity operators around the diagonal.

using QuantumClifford, CairoMakie
+f
Example block output

canonicalize_clip!

Convert to the "clipped" gauge of a stabilizer state resulting in a "river" of non-identity operators around the diagonal.

using QuantumClifford, CairoMakie
 f=Figure()
 stabilizerplot_axis(f[1,1], canonicalize_clip!(random_stabilizer(30)))
-f
Example block output

The properties of the clipped gauge are:

  1. Each qubit is the left/right "endpoint" of exactly two stabilizer rows.
  2. For the same qubit the two endpoints are always different Pauli operators.

This canonicalization is used to derive the bigram a stabilizer state, which is also related to entanglement entropy in the state.

Introduced in (Nahum et al., 2017), with a more detailed explanation of the algorithm in Appendix A of (Li et al., 2019).

+fExample block output

The properties of the clipped gauge are:

  1. Each qubit is the left/right "endpoint" of exactly two stabilizer rows.
  2. For the same qubit the two endpoints are always different Pauli operators.

This canonicalization is used to derive the bigram a stabilizer state, which is also related to entanglement entropy in the state.

Introduced in (Nahum et al., 2017), with a more detailed explanation of the algorithm in Appendix A of (Li et al., 2019).

diff --git a/dev/commonstates/index.html b/dev/commonstates/index.html index 466bc968a..01fd15cac 100644 --- a/dev/commonstates/index.html +++ b/dev/commonstates/index.html @@ -72,4 +72,4 @@ + XXXX + ZZ__ + _ZZ_ -+ __ZZ ++ __ZZ diff --git a/dev/datastructures/index.html b/dev/datastructures/index.html index f22b43928..2c603bf5a 100644 --- a/dev/datastructures/index.html +++ b/dev/datastructures/index.html @@ -1,2 +1,2 @@ -Datastructure Choice · QuantumClifford.jl

Data Structures Options

Choosing Appropriate Tableau Data Structure

There are four different data structures used to represent stabilizer states. If you will never need projective measurements you probably would want to use Stabilizer. If you require projective measurements, but only on pure states, Destabilizer should be the appropriate data structure. If mixed stabilizer states are involved, MixedStabilizer would be necessary.

Stabilizer is simply a list of Pauli operators in a tableau form. As a data structure it does not enforce the requirements for a pure stabilizer state (the rows of the tableau do not necessarily commute, nor are they forced to be Hermitian; the tableau might be underdetermined, redundant, or contradictory). It is up to the user to ensure that the initial values in the tableau are meaningful and consistent.

canonicalize!, project!, and generate! can accept an under determined (mixed state) Stabilizer instance and operate correctly. canonicalize! can also accept a redundant Stabilizer (i.e. not all rows are independent), leaving as many identity rows at the bottom of the canonicalized tableau as the number of redundant stabilizers in the initial tableau.

canonicalize! takes $\mathcal{O}(n^3)$ steps. generate! expects a canonicalized input and then takes $\mathcal{O}(n^2)$ steps. project! takes $\mathcal{O}(n^3)$ for projecting on commuting operators due to the need to call canonicalize! and generate!. If the projections is on an anticommuting operator (or if keep_result=false) then it takes $\mathcal{O}(n^2)$ steps.

MixedStabilizer provides explicit tracking of the rank of the mixed state and works properly when the projection is on a commuting operator not in the stabilizer (see table below for details). Otherwise it has the same performance as Stabilizer.

The canonicalization can be made unnecessary if we track the destabilizer generators. There are two data structures capable of that.

Destabilizer stores both the destabilizer and stabilizer states. project! called on it never requires a stabilizer canonicalization, hence it runs in $\mathcal{O}(n^2)$. However, project! will raise an exception if you try to project on a commuting state that is not in the stabilizer as that would be an expensive $\mathcal{O}(n^3)$ operation.

MixedDestabilizer tracks both the destabilizer operators and the logical operators in addition to the stabilizer generators. It does not require canonicalization for measurements and its project! operations always takes $\mathcal{O}(n^2)$.

For the operation _, anticom_index, result = project!(...) we have the following behavior:

projectionStabilizerMixedStabilizerDestabilizerMixedDestabilizer
on anticommuting operator anticom_index>0 result===nothingcorrect result in $\mathcal{O}(n^2)$ stepssame as Stabilizersame as Stabilizersame as Stabilizer
on commuting operator in the stabilizer anticom_index==0 result!==nothing$\mathcal{O}(n^3)$; or $\mathcal{O}(n^2)$ if keep_result=false$\mathcal{O}(n^3)$$\mathcal{O}(n^2)$ if the state is pure, throws exception otherwise$\mathcal{O}(n^2)$
on commuting operator out of the stabilizer[1] anticom_index==rank result===nothing$\mathcal{O}(n^3)$, but the user needs to manually include the new operator to the stabilizer; or $\mathcal{O}(n^2)$ if keep_result=false but then result indistinguishable from cell above and anticom_index==0$\mathcal{O}(n^3)$ and rank goes up by onenot applicable if the state is pure, throws exception otherwise$\mathcal{O}(n^2)$ and rank goes up by one

Notice the results when the projection operator commutes with the state but is not generated by the stabilizers of the state (the last row of the table). In that case we have _, anticom_index, result = project!(...) where both anticom_index==rank and result===nothing, with rank being the new rank after projection, one more than the number of rows in the tableau before the measurement.

Bit Packing in Integers and Array Order

We do not use boolean arrays to store information about the qubits as this would be wasteful (7 out of 8 bits in the boolean would be unused). Instead, we use all 8 qubits in a byte and perform bitwise logical operations as necessary. Implementation details of the object in RAM can matter for performance. The library permits any of the standard UInt types to be used for packing the bits, and larger UInt types (like UInt64) are usually faster as they permit working on 64 qubits at a time (instead of 1 if we used a boolean, or 8 if we used a byte).

Moreover, how a tableau is stored in memory can affect performance, as a row-major storage usually permits more efficient use of the CPU cache (for the particular algorithms we use).

Both of these parameters are benchmarked (testing the application of a Pauli operator, which is an $\mathcal{O}(n^2)$ operation; and testing the canonicalization of a Stabilizer, which is an $\mathcal{O}(n^3)$ operation). Row-major UInt64 is the best performing and it is used by default in this library.

  • 1This can occur only if the state being projected is mixed. Both Stabilizer and Destabilizer can be used for mixed states by simply providing fewer stabilizer generators than qubits at initialization. This can be useful for low-level code that tries to avoid the extra memory cost of using MixedStabilizer and MixedDestabilizer but should be avoided otherwise. project! works correctly or raises an explicit warning on all 4 data structures.
+Datastructure Choice · QuantumClifford.jl

Data Structures Options

Choosing Appropriate Tableau Data Structure

There are four different data structures used to represent stabilizer states. If you will never need projective measurements you probably would want to use Stabilizer. If you require projective measurements, but only on pure states, Destabilizer should be the appropriate data structure. If mixed stabilizer states are involved, MixedStabilizer would be necessary.

Stabilizer is simply a list of Pauli operators in a tableau form. As a data structure it does not enforce the requirements for a pure stabilizer state (the rows of the tableau do not necessarily commute, nor are they forced to be Hermitian; the tableau might be underdetermined, redundant, or contradictory). It is up to the user to ensure that the initial values in the tableau are meaningful and consistent.

canonicalize!, project!, and generate! can accept an under determined (mixed state) Stabilizer instance and operate correctly. canonicalize! can also accept a redundant Stabilizer (i.e. not all rows are independent), leaving as many identity rows at the bottom of the canonicalized tableau as the number of redundant stabilizers in the initial tableau.

canonicalize! takes $\mathcal{O}(n^3)$ steps. generate! expects a canonicalized input and then takes $\mathcal{O}(n^2)$ steps. project! takes $\mathcal{O}(n^3)$ for projecting on commuting operators due to the need to call canonicalize! and generate!. If the projections is on an anticommuting operator (or if keep_result=false) then it takes $\mathcal{O}(n^2)$ steps.

MixedStabilizer provides explicit tracking of the rank of the mixed state and works properly when the projection is on a commuting operator not in the stabilizer (see table below for details). Otherwise it has the same performance as Stabilizer.

The canonicalization can be made unnecessary if we track the destabilizer generators. There are two data structures capable of that.

Destabilizer stores both the destabilizer and stabilizer states. project! called on it never requires a stabilizer canonicalization, hence it runs in $\mathcal{O}(n^2)$. However, project! will raise an exception if you try to project on a commuting state that is not in the stabilizer as that would be an expensive $\mathcal{O}(n^3)$ operation.

MixedDestabilizer tracks both the destabilizer operators and the logical operators in addition to the stabilizer generators. It does not require canonicalization for measurements and its project! operations always takes $\mathcal{O}(n^2)$.

For the operation _, anticom_index, result = project!(...) we have the following behavior:

projectionStabilizerMixedStabilizerDestabilizerMixedDestabilizer
on anticommuting operator anticom_index>0 result===nothingcorrect result in $\mathcal{O}(n^2)$ stepssame as Stabilizersame as Stabilizersame as Stabilizer
on commuting operator in the stabilizer anticom_index==0 result!==nothing$\mathcal{O}(n^3)$; or $\mathcal{O}(n^2)$ if keep_result=false$\mathcal{O}(n^3)$$\mathcal{O}(n^2)$ if the state is pure, throws exception otherwise$\mathcal{O}(n^2)$
on commuting operator out of the stabilizer[1] anticom_index==rank result===nothing$\mathcal{O}(n^3)$, but the user needs to manually include the new operator to the stabilizer; or $\mathcal{O}(n^2)$ if keep_result=false but then result indistinguishable from cell above and anticom_index==0$\mathcal{O}(n^3)$ and rank goes up by onenot applicable if the state is pure, throws exception otherwise$\mathcal{O}(n^2)$ and rank goes up by one

Notice the results when the projection operator commutes with the state but is not generated by the stabilizers of the state (the last row of the table). In that case we have _, anticom_index, result = project!(...) where both anticom_index==rank and result===nothing, with rank being the new rank after projection, one more than the number of rows in the tableau before the measurement.

Bit Packing in Integers and Array Order

We do not use boolean arrays to store information about the qubits as this would be wasteful (7 out of 8 bits in the boolean would be unused). Instead, we use all 8 qubits in a byte and perform bitwise logical operations as necessary. Implementation details of the object in RAM can matter for performance. The library permits any of the standard UInt types to be used for packing the bits, and larger UInt types (like UInt64) are usually faster as they permit working on 64 qubits at a time (instead of 1 if we used a boolean, or 8 if we used a byte).

Moreover, how a tableau is stored in memory can affect performance, as a row-major storage usually permits more efficient use of the CPU cache (for the particular algorithms we use).

Both of these parameters are benchmarked (testing the application of a Pauli operator, which is an $\mathcal{O}(n^2)$ operation; and testing the canonicalization of a Stabilizer, which is an $\mathcal{O}(n^3)$ operation). Row-major UInt64 is the best performing and it is used by default in this library.

  • 1This can occur only if the state being projected is mixed. Both Stabilizer and Destabilizer can be used for mixed states by simply providing fewer stabilizer generators than qubits at initialization. This can be useful for low-level code that tries to avoid the extra memory cost of using MixedStabilizer and MixedDestabilizer but should be avoided otherwise. project! works correctly or raises an explicit warning on all 4 data structures.
diff --git a/dev/ecc_example_sim/index.html b/dev/ecc_example_sim/index.html index dbb35b530..ee906dff5 100644 --- a/dev/ecc_example_sim/index.html +++ b/dev/ecc_example_sim/index.html @@ -20,7 +20,7 @@ errors = [PauliError(i,errprob) for i in 1:code_n(code)] fullcircuit = [ecirc..., errors..., scirc...]Example block output

And running this noisy simulation:

frames = pftrajectories(fullcircuit; trajectories=nframes)
 pfmeasurements(frames)
4×6 Matrix{Bool}:
- 1  1  0  0  0  0
- 0  1  1  0  1  1
- 0  0  0  0  0  0
- 0  0  0  0  0  0
+ 0 1 0 0 1 1 + 0 0 0 1 1 0 + 0 1 1 0 0 0 + 0 0 0 0 0 0 diff --git a/dev/graphs/index.html b/dev/graphs/index.html index 0dbd7f620..51c0084eb 100644 --- a/dev/graphs/index.html +++ b/dev/graphs/index.html @@ -43,4 +43,4 @@ + XZZ_ + ZX_Z + Z_XZ -+ _ZZX

Graphs are represented with the Graphs.jl package and plotting can be done both in Plots.jl and Makie.jl (with GraphMakie).

++ _ZZX

Graphs are represented with the Graphs.jl package and plotting can be done both in Plots.jl and Makie.jl (with GraphMakie).

diff --git a/dev/index.html b/dev/index.html index 10e038c9e..f112ad05f 100644 --- a/dev/index.html +++ b/dev/index.html @@ -15,4 +15,4 @@ julia> tCNOT * S"-XX +ZZ" - X_ -+ _Z

Circuit Simulation

The circuit simulation component of QuantumClifford.jl enables Monte Carlo (or symbolic) simulations of noisy Clifford circuits. It provides three main simulation methods: mctrajectories, pftrajectories, and petrajectories. These methods offer varying levels of efficiency, accuracy, and insight.

Monte Carlo Simulations with Stabilizer Tableaux (mctrajectories)

The mctrajectories method runs Monte Carlo simulations using a Stabilizer tableau representation for the quantum states.

Monte Carlo Simulations with Pauli Frames (pftrajectories)

The pftrajectories method runs Monte Carlo simulations of Pauli frames over a single reference Stabilizer tableau simulation. This approach is much more efficient but supports a smaller class of circuits.

Symbolic Depth-First Traversal of Quantum Trajectories (petrajectories)

The petrajectories method performs a depth-first traversal of the most probable quantum trajectories, providing a fixed-order approximation of the circuit's behavior. This approach gives symbolic expressions for various figures of merit instead of just a numeric value.

++ _Z

Circuit Simulation

The circuit simulation component of QuantumClifford.jl enables Monte Carlo (or symbolic) simulations of noisy Clifford circuits. It provides three main simulation methods: mctrajectories, pftrajectories, and petrajectories. These methods offer varying levels of efficiency, accuracy, and insight.

Monte Carlo Simulations with Stabilizer Tableaux (mctrajectories)

The mctrajectories method runs Monte Carlo simulations using a Stabilizer tableau representation for the quantum states.

Monte Carlo Simulations with Pauli Frames (pftrajectories)

The pftrajectories method runs Monte Carlo simulations of Pauli frames over a single reference Stabilizer tableau simulation. This approach is much more efficient but supports a smaller class of circuits.

Symbolic Depth-First Traversal of Quantum Trajectories (petrajectories)

The petrajectories method performs a depth-first traversal of the most probable quantum trajectories, providing a fixed-order approximation of the circuit's behavior. This approach gives symbolic expressions for various figures of merit instead of just a numeric value.

diff --git a/dev/mixed/index.html b/dev/mixed/index.html index fe1d59ea7..087c6c772 100644 --- a/dev/mixed/index.html +++ b/dev/mixed/index.html @@ -57,4 +57,4 @@ + XXX + ZZ_ 𝒵ₗ━━━ -+ Z_Z

Destabilizer and MixedStabilizer do not use any column swaps on instantiation as they do not track the logical operators.

++ Z_Z

Destabilizer and MixedStabilizer do not use any column swaps on instantiation as they do not track the logical operators.

diff --git a/dev/noise/index.html b/dev/noise/index.html index bb3ab7fdc..2e619fabc 100644 --- a/dev/noise/index.html +++ b/dev/noise/index.html @@ -1,2 +1,2 @@ -Noise Processes · QuantumClifford.jl
+Noise Processes · QuantumClifford.jl
diff --git a/dev/noisycircuits/index.html b/dev/noisycircuits/index.html index 3772f9564..ca653a2c3 100644 --- a/dev/noisycircuits/index.html +++ b/dev/noisycircuits/index.html @@ -1,2 +1,2 @@ -Simulation of Noisy Circuits · QuantumClifford.jl

Simulation of Noisy Clifford Circuits

Unstable

This is unfinished experimental functionality that will change significantly.

We have experimental support for simulation of noisy Clifford circuits which can be imported with using QuantumClifford.Experimental.NoisyCircuits.

Both Monte Carlo and Perturbative Expansion approaches are supported. When performing a perturbative expansion in the noise parameter, the expansion can optionally be performed symbolically, to arbitrary high orders.

Multiple notebooks with examples are also available. For instance, see this tutorial on entanglement purification for many examples.

+Simulation of Noisy Circuits · QuantumClifford.jl

Simulation of Noisy Clifford Circuits

Unstable

This is unfinished experimental functionality that will change significantly.

We have experimental support for simulation of noisy Clifford circuits which can be imported with using QuantumClifford.Experimental.NoisyCircuits.

Both Monte Carlo and Perturbative Expansion approaches are supported. When performing a perturbative expansion in the noise parameter, the expansion can optionally be performed symbolically, to arbitrary high orders.

Multiple notebooks with examples are also available. For instance, see this tutorial on entanglement purification for many examples.

diff --git a/dev/noisycircuits_API/index.html b/dev/noisycircuits_API/index.html index 39167d069..c540aae28 100644 --- a/dev/noisycircuits_API/index.html +++ b/dev/noisycircuits_API/index.html @@ -1,2 +1,2 @@ -API · QuantumClifford.jl

Full API (autogenerated)

Unstable

This is experimental functionality with an unstable API.

+API · QuantumClifford.jl

Full API (autogenerated)

Unstable

This is experimental functionality with an unstable API.

diff --git a/dev/noisycircuits_mc/index.html b/dev/noisycircuits_mc/index.html index 533c01b47..d4611a152 100644 --- a/dev/noisycircuits_mc/index.html +++ b/dev/noisycircuits_mc/index.html @@ -15,7 +15,7 @@ # then a Bell measurement # followed by checking whether the final result indeed corresponds to the correct Bell pair. circuit = [n,g1,g2,m,v]Example block output

And we can run a Monte Carlo simulation of that circuit with mctrajectories.

mctrajectories(initial_state, circuit, trajectories=500)
Dict{CircuitStatus, Float64} with 4 entries:
+  true_success:CircuitStatus(1)  => 487.0
   continue:CircuitStatus(0)      => 0.0
   failure:CircuitStatus(3)       => 4.0
-  true_success:CircuitStatus(1)  => 487.0
-  false_success:CircuitStatus(2) => 9.0

For more examples, see the notebook comparing the Monte Carlo and Perturbative method or this tutorial on entanglement purification for many examples.

Interface for custom operations

If you want to create a custom gate type (e.g. calling it Operation), you need to definite the following methods.

applywstatus!(s::T, g::Operation)::Tuple{T,Symbol} where T is a tableaux type like Stabilizer or a Register. The Symbol is the status of the operation. Predefined statuses are kept in the registered_statuses list, but you can add more. Be sure to expand this list if you want the trajectory simulators using your custom statuses to output all trajectories.

There is also applynoise! which is convenient wait to create a noise model that can then be plugged into the NoisyGate struct, letting you reuse the predefined perfect gates and measurements. However, you can also just make up your own noise operator simply by implementing applywstatus! for it.

You can also consult the list of implemented operators.

+ false_success:CircuitStatus(2) => 9.0

For more examples, see the notebook comparing the Monte Carlo and Perturbative method or this tutorial on entanglement purification for many examples.

Interface for custom operations

If you want to create a custom gate type (e.g. calling it Operation), you need to definite the following methods.

applywstatus!(s::T, g::Operation)::Tuple{T,Symbol} where T is a tableaux type like Stabilizer or a Register. The Symbol is the status of the operation. Predefined statuses are kept in the registered_statuses list, but you can add more. Be sure to expand this list if you want the trajectory simulators using your custom statuses to output all trajectories.

There is also applynoise! which is convenient wait to create a noise model that can then be plugged into the NoisyGate struct, letting you reuse the predefined perfect gates and measurements. However, you can also just make up your own noise operator simply by implementing applywstatus! for it.

You can also consult the list of implemented operators.

diff --git a/dev/noisycircuits_ops/index.html b/dev/noisycircuits_ops/index.html index 3d23e62a2..c39adf5e4 100644 --- a/dev/noisycircuits_ops/index.html +++ b/dev/noisycircuits_ops/index.html @@ -10,4 +10,4 @@ gate3 = SparseGate(tSWAP, [1,3]) cg = ConditionalGate(gate1, gate2, 2) dg = DecisionGate([gate1,gate2,gate3], bit_register->1) # it will always perform gate1 -[sMX(4,1), sMZ(5,2), cg, dg]Example block output

TODO: Split ConditionalGate into quantum conditional and classical conditional

+[sMX(4,1), sMZ(5,2), cg, dg]Example block output

TODO: Split ConditionalGate into quantum conditional and classical conditional

diff --git a/dev/noisycircuits_perturb/index.html b/dev/noisycircuits_perturb/index.html index ab84fd527..4bcbdf1e9 100644 --- a/dev/noisycircuits_perturb/index.html +++ b/dev/noisycircuits_perturb/index.html @@ -18,6 +18,6 @@ circuit = [n,g1,g2,m,v] petrajectories(initial_state, circuit)
Dict{CircuitStatus, Float64} with 3 entries:
-  failure:CircuitStatus(3)       => 0.0129373
   true_success:CircuitStatus(1)  => 0.967065
-  false_success:CircuitStatus(2) => 0.019406

For more examples, see the notebook comparing the Monte Carlo and Perturbative method or this tutorial on entanglement purification.

Symbolic expansions

The perturbative expansion method works with symbolic variables as well. One can use any of the symbolic libraries available in Julia and simply plug symbolic parameters in lieu of numeric parameters. A detailed example is available as a Jupyter notebook.

Interface for custom operations

If you want to create a custom gate type (e.g. calling it Operation), you need to definite the following methods.

applyop_branches!(s::T, g::Operation; max_order=1)::Vector{Tuple{T,Symbol,Real,Int}} where T is a tableaux type like Stabilizer or a Register. The Symbol is the status of the operation, the Real is the probability for that branch, and the Int is the order of that branch.

There is also applynoise_branches! which is convenient for use in NoisyGate, but you can also just make up your own noise operator simply by implementing applyop_branches! for it.

You can also consult the list of implemented operators.

+ failure:CircuitStatus(3) => 0.0129373 + false_success:CircuitStatus(2) => 0.019406

For more examples, see the notebook comparing the Monte Carlo and Perturbative method or this tutorial on entanglement purification.

Symbolic expansions

The perturbative expansion method works with symbolic variables as well. One can use any of the symbolic libraries available in Julia and simply plug symbolic parameters in lieu of numeric parameters. A detailed example is available as a Jupyter notebook.

Interface for custom operations

If you want to create a custom gate type (e.g. calling it Operation), you need to definite the following methods.

applyop_branches!(s::T, g::Operation; max_order=1)::Vector{Tuple{T,Symbol,Real,Int}} where T is a tableaux type like Stabilizer or a Register. The Symbol is the status of the operation, the Real is the probability for that branch, and the Int is the order of that branch.

There is also applynoise_branches! which is convenient for use in NoisyGate, but you can also just make up your own noise operator simply by implementing applyop_branches! for it.

You can also consult the list of implemented operators.

diff --git a/dev/plotting/8c14f637.svg b/dev/plotting/02c64905.svg similarity index 99% rename from dev/plotting/8c14f637.svg rename to dev/plotting/02c64905.svg index 763187b78..1cbab765f 100644 --- a/dev/plotting/8c14f637.svg +++ b/dev/plotting/02c64905.svg @@ -1,23 +1,23 @@ - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + Visualizations · QuantumClifford.jl

Visualizations

Stabilizers have a plot recipe that can be used with Plots.jl or Makie.jl. It simply displays the corresponding parity check matrix (extracted with stab_to_gf2) as a bitmap image. Circuits can be visualized with Quantikz.jl.

Importing the aforementioned packages together with QuantumClifford is necessary to enable the plotting functionality (implemented as package extensions).

Plots.jl

In Plots.jl we have a simple recipe plot(s::Stabilizer; xzcomponents=...) where xzcomponents=:split plots the tableau heatmap in a wide form, X bits on the left, Z bits on the right; or xzcomponents=:together plots them overlapping, with different colors for I, X, Z, and Y.

using QuantumClifford, Plots
-plot(random_stabilizer(20,30), xzcomponents=:split)
Example block output
using QuantumClifford, Plots
-plot(canonicalize!(random_stabilizer(20,30)))
Example block output
using QuantumClifford, Plots
-plot(canonicalize_gott!(random_stabilizer(30))[1], xzcomponents=:split)
Example block output
using QuantumClifford, Plots
-plot(canonicalize_gott!(random_stabilizer(30))[1]; xzcomponents=:together)
Example block output
using QuantumClifford, Plots
-plot(canonicalize_rref!(random_stabilizer(20,30),1:30)[1]; xzcomponents=:together)
Example block output

Makie.jl

Makie's heatmap can be directly called on Stabilizer.

using QuantumClifford, CairoMakie
+plot(random_stabilizer(20,30), xzcomponents=:split)
Example block output
using QuantumClifford, Plots
+plot(canonicalize!(random_stabilizer(20,30)))
Example block output
using QuantumClifford, Plots
+plot(canonicalize_gott!(random_stabilizer(30))[1], xzcomponents=:split)
Example block output
using QuantumClifford, Plots
+plot(canonicalize_gott!(random_stabilizer(30))[1]; xzcomponents=:together)
Example block output
using QuantumClifford, Plots
+plot(canonicalize_rref!(random_stabilizer(20,30),1:30)[1]; xzcomponents=:together)
Example block output

Makie.jl

Makie's heatmap can be directly called on Stabilizer.

using QuantumClifford, CairoMakie
 s = S"IIXZ
       ZZIZ
       YYIZ
@@ -47,4 +47,4 @@
 f=Figure()
 stabilizerplot_axis(f[1,1],random_stabilizer(100))
 f
Example block output

Quantikz.jl

With the Quantikz library you can visualize gates or sequences of gates.

using QuantumClifford, Quantikz
-circuit = [sCNOT(1,2), SparseGate(random_clifford(4), [1,4,5,6]), sMZ(4)]
Example block output
+circuit = [sCNOT(1,2), SparseGate(random_clifford(4), [1,4,5,6]), sMZ(4)]Example block output diff --git a/dev/references.bib b/dev/references.bib index 4bdfce7f9..8bfae0f5e 100644 --- a/dev/references.bib +++ b/dev/references.bib @@ -262,3 +262,50 @@ @article{gottesman1996class year={1996}, publisher={APS} } + +@article{muller1954application, + title={Application of Boolean algebra to switching circuit design and to error detection}, + author={Muller, David E}, + journal={Transactions of the IRE professional group on electronic computers}, + number={3}, + pages={6--12}, + year={1954}, + publisher={IEEE} +} + +@article{reed1954class, + title={A class of multiple-error-correcting codes and the decoding scheme}, + author={Reed, Irving S}, + journal={IEEE Transactions on Information Theory}, + volume={4}, + number={4}, + pages={38--49}, + year={1954} +} + +@article{raaphorst2003reed, + title={Reed-muller codes}, + author={Raaphorst, Sebastian}, + journal={Carleton University, May}, + volume={9}, + year={2003}, + publisher={Citeseer} +} + +@article{abbe2020reed, + title={Reed--Muller codes: Theory and algorithms}, + author={Abbe, Emmanuel and Shpilka, Amir and Ye, Min}, + journal={IEEE Transactions on Information Theory}, + volume={67}, + number={6}, + pages={3251--3277}, + year={2020}, + publisher={IEEE} +} + +@book{djordjevic2021quantum, + title={Quantum information processing, quantum computing, and quantum error correction: an engineering approach}, + author={Djordjevic, Ivan B}, + year={2021}, + publisher={Academic Press} +} \ No newline at end of file diff --git a/dev/references/index.html b/dev/references/index.html index a7021406f..642517048 100644 --- a/dev/references/index.html +++ b/dev/references/index.html @@ -1,2 +1,2 @@ -Suggested Readings & References · QuantumClifford.jl

Suggested reading

For the basis of the tableaux methods first read (Gottesman, 1998) followed by the more efficient approach described in (Aaronson and Gottesman, 2004).

The tableaux can be canonicalized (i.e. Gaussian elimination can be performed on them) in a number of different ways, and considering the different approaches provides useful insight. The following methods are implemented in this library:

For the use of these methods in error correction and the subtle overlap between the two fields consider these resources. They are also useful in defining some of the specific constraints in commutation between rows in the tableaux:

These publications describe the uniform sampling of random stabilizer states:

For circuit construction routines (for stabilizer measurements for a given code):

For code construction routines:

References

+Suggested Readings & References · QuantumClifford.jl

Suggested reading

For the basis of the tableaux methods first read (Gottesman, 1998) followed by the more efficient approach described in (Aaronson and Gottesman, 2004).

The tableaux can be canonicalized (i.e. Gaussian elimination can be performed on them) in a number of different ways, and considering the different approaches provides useful insight. The following methods are implemented in this library:

For the use of these methods in error correction and the subtle overlap between the two fields consider these resources. They are also useful in defining some of the specific constraints in commutation between rows in the tableaux:

These publications describe the uniform sampling of random stabilizer states:

For circuit construction routines (for stabilizer measurements for a given code):

For quantum code construction routines:

For classical code construction routines:

References

  • Aaronson, S. and Gottesman, D. (2004). Improved simulation of stabilizer circuits. Physical Review A 70, 052328.
  • Abbe, E.; Shpilka, A. and Ye, M. (2020). Reed–Muller codes: Theory and algorithms. IEEE Transactions on Information Theory 67, 3251–3277.
  • Audenaert, K. M. and Plenio, M. B. (2005). Entanglement on mixed stabilizer states: normal forms and reduction procedures. New Journal of Physics 7, 170.
  • Bravyi, S. and Maslov, D. (2021). Hadamard-free circuits expose the structure of the Clifford group. IEEE Transactions on Information Theory 67, 4546–4563.
  • Calderbank, A. R.; Rains, E. M.; Shor, P. and Sloane, N. J. (1998). Quantum error correction via codes over GF (4). IEEE Transactions on Information Theory 44, 1369–1387.
  • Chao, R. and Reichardt, B. W. (2017). Quantum Error Correction with Only Two Extra Qubits. Physical review letters 121 5, 050502.
  • Cleve, R. and Gottesman, D. (1997). Efficient computations of encodings for quantum error correction. Physical Review A 56, 76.
  • Djordjevic, I. B. (2021). Quantum information processing, quantum computing, and quantum error correction: an engineering approach (Academic Press).
  • Garcia, H. J.; Markov, I. L. and Cross, A. W. (2012). Efficient inner-product algorithm for stabilizer states, arXiv preprint arXiv:1210.6646.
  • Gottesman, D. (1996). Class of quantum error-correcting codes saturating the quantum Hamming bound. Physical Review A 54, 1862.
  • Gottesman, D. (1997). Stabilizer codes and quantum error correction. Ph.D. Thesis, California Institute of Technology.
  • Gottesman, D. (1998). The Heisenberg representation of quantum computers. In: International Conference on Group Theoretic Methods in Physics (Citeseer).
  • Grassl, M. (2002). Algorithmic aspects of quantum error-correcting codes. Mathematics of Quantum Computation, 223–252.
  • Grassl, M. (2011). Variations on encoding circuits for stabilizer quantum codes. In: International Conference on Coding and Cryptology (Springer); pp. 142–158.
  • Gullans, M. J.; Krastanov, S.; Huse, D. A.; Jiang, L. and Flammia, S. T. (2020). Quantum coding with low-depth random circuits, arXiv preprint arXiv:2010.09775.
  • Koenig, R. and Smolin, J. A. (2014). How to efficiently select an arbitrary Clifford group element. Journal of Mathematical Physics 55, 122202.
  • Krastanov, S.; de la Cerda, A. S. and Narang, P. (2020). Heterogeneous Multipartite Entanglement Purification for Size-Constrained Quantum Devices, arXiv preprint arXiv:2011.11640.
  • Li, Y.; Chen, X. and Fisher, M. P. (2019). Measurement-driven entanglement transition in hybrid quantum circuits. Physical Review B 100, 134306.
  • MacKay, D. J.; Mitchison, G. and McFadden, P. L. (2004). Sparse-graph codes for quantum error correction. IEEE Transactions on Information Theory 50, 2315–2330.
  • Muller, D. E. (1954). Application of Boolean algebra to switching circuit design and to error detection. Transactions of the IRE professional group on electronic computers, 6–12.
  • Nahum, A.; Ruhman, J.; Vijay, S. and Haah, J. (2017). Quantum Entanglement Growth under Random Unitary Dynamics. Physical Review X 7, 031016.
  • Raaphorst, S. (2003). Reed-muller codes. Carleton University, May 9.
  • Reed, I. S. (1954). A class of multiple-error-correcting codes and the decoding scheme. IEEE Transactions on Information Theory 4, 38–49.
  • Steane, A. M. (2007). A tutorial on quantum error correction. In: PROCEEDINGS-INTERNATIONAL SCHOOL OF PHYSICS ENRICO FERMI, Vol. 162 (IOS Press; Ohmsha; 1999); p. 1.
  • Van Den Berg, E. (2021). A simple method for sampling random Clifford operators. In: 2021 IEEE International Conference on Quantum Computing and Engineering (QCE) (IEEE); pp. 54–59.
  • Wilde, M. M. (2009). Logical operators of quantum codes. Physical Review A 79, 062322.
  • Yu, S.; Bierbrauer, J.; Dong, Y.; Chen, Q. and Oh, C. H. (2013). All the Stabilizer Codes of Distance 3. IEEE Transactions on Information Theory 59, 5179–5185.
diff --git a/dev/search_index.js b/dev/search_index.js index 87f1ca473..cd0614066 100644 --- a/dev/search_index.js +++ b/dev/search_index.js @@ -1,3 +1,3 @@ var documenterSearchIndex = {"docs": -[{"location":"references/#Suggested-reading","page":"Suggested Readings & References","title":"Suggested reading","text":"","category":"section"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"For the basis of the tableaux methods first read (Gottesman, 1998) followed by the more efficient approach described in (Aaronson and Gottesman, 2004).","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"The tableaux can be canonicalized (i.e. Gaussian elimination can be performed on them) in a number of different ways, and considering the different approaches provides useful insight. The following methods are implemented in this library:","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"The default one: (Garcia et al., 2012)\nUseful when in need of tracing out a set of qubits: (Audenaert and Plenio, 2005)\nUseful when defining logical operators of codes: (Gottesman, 1997)","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"For the use of these methods in error correction and the subtle overlap between the two fields consider these resources. They are also useful in defining some of the specific constraints in commutation between rows in the tableaux:","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"(Steane, 2007)\n(Calderbank et al., 1998)\n(MacKay et al., 2004)\n(Wilde, 2009)","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"These publications describe the uniform sampling of random stabilizer states:","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"(Koenig and Smolin, 2014)\n(Bravyi and Maslov, 2021)\n(Van Den Berg, 2021)\n(Li et al., 2019)","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"For circuit construction routines (for stabilizer measurements for a given code):","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"(Cleve and Gottesman, 1997)\n(Gottesman, 1997) (and its erratum)\n(Grassl, 2002)\n(Grassl, 2011)","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"For code construction routines:","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"(Cleve and Gottesman, 1997)\n(Gottesman, 1996)\n(Gottesman, 1997)\n(Yu et al., 2013)\n(Chao and Reichardt, 2017)","category":"page"},{"location":"references/#References","page":"Suggested Readings & References","title":"References","text":"","category":"section"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"Aaronson, S. and Gottesman, D. (2004). Improved simulation of stabilizer circuits. Physical Review A 70, 052328.\n\n\n\nAudenaert, K. M. and Plenio, M. B. (2005). Entanglement on mixed stabilizer states: normal forms and reduction procedures. New Journal of Physics 7, 170.\n\n\n\nBravyi, S. and Maslov, D. (2021). Hadamard-free circuits expose the structure of the Clifford group. IEEE Transactions on Information Theory 67, 4546–4563.\n\n\n\nCalderbank, A. R.; Rains, E. M.; Shor, P. and Sloane, N. J. (1998). Quantum error correction via codes over GF (4). IEEE Transactions on Information Theory 44, 1369–1387.\n\n\n\nChao, R. and Reichardt, B. W. (2017). Quantum Error Correction with Only Two Extra Qubits. Physical review letters 121 5, 050502.\n\n\n\nCleve, R. and Gottesman, D. (1997). Efficient computations of encodings for quantum error correction. Physical Review A 56, 76.\n\n\n\nGarcia, H. J.; Markov, I. L. and Cross, A. W. (2012). Efficient inner-product algorithm for stabilizer states, arXiv preprint arXiv:1210.6646.\n\n\n\nGottesman, D. (1996). Class of quantum error-correcting codes saturating the quantum Hamming bound. Physical Review A 54, 1862.\n\n\n\nGottesman, D. (1997). Stabilizer codes and quantum error correction. Ph.D. Thesis, California Institute of Technology.\n\n\n\nGottesman, D. (1998). The Heisenberg representation of quantum computers. In: International Conference on Group Theoretic Methods in Physics (Citeseer).\n\n\n\nGrassl, M. (2002). Algorithmic aspects of quantum error-correcting codes. Mathematics of Quantum Computation, 223–252.\n\n\n\nGrassl, M. (2011). Variations on encoding circuits for stabilizer quantum codes. In: International Conference on Coding and Cryptology (Springer); pp. 142–158.\n\n\n\nGullans, M. J.; Krastanov, S.; Huse, D. A.; Jiang, L. and Flammia, S. T. (2020). Quantum coding with low-depth random circuits, arXiv preprint arXiv:2010.09775.\n\n\n\nKoenig, R. and Smolin, J. A. (2014). How to efficiently select an arbitrary Clifford group element. Journal of Mathematical Physics 55, 122202.\n\n\n\nKrastanov, S.; de la Cerda, A. S. and Narang, P. (2020). Heterogeneous Multipartite Entanglement Purification for Size-Constrained Quantum Devices, arXiv preprint arXiv:2011.11640.\n\n\n\nLi, Y.; Chen, X. and Fisher, M. P. (2019). Measurement-driven entanglement transition in hybrid quantum circuits. Physical Review B 100, 134306.\n\n\n\nMacKay, D. J.; Mitchison, G. and McFadden, P. L. (2004). Sparse-graph codes for quantum error correction. IEEE Transactions on Information Theory 50, 2315–2330.\n\n\n\nNahum, A.; Ruhman, J.; Vijay, S. and Haah, J. (2017). Quantum Entanglement Growth under Random Unitary Dynamics. Physical Review X 7, 031016.\n\n\n\nSteane, A. M. (2007). A tutorial on quantum error correction. In: PROCEEDINGS-INTERNATIONAL SCHOOL OF PHYSICS ENRICO FERMI, Vol. 162 (IOS Press; Ohmsha; 1999); p. 1.\n\n\n\nVan Den Berg, E. (2021). A simple method for sampling random Clifford operators. In: 2021 IEEE International Conference on Quantum Computing and Engineering (QCE) (IEEE); pp. 54–59.\n\n\n\nWilde, M. M. (2009). Logical operators of quantum codes. Physical Review A 79, 062322.\n\n\n\nYu, S.; Bierbrauer, J.; Dong, Y.; Chen, Q. and Oh, C. H. (2013). All the Stabilizer Codes of Distance 3. IEEE Transactions on Information Theory 59, 5179–5185.\n\n\n\n","category":"page"},{"location":"noise/#noise","page":"Noise Processes","title":"Noise Processes","text":"","category":"section"},{"location":"noise/","page":"Noise Processes","title":"Noise Processes","text":"DocTestSetup = quote\n using QuantumClifford\n using StableRNGs\n rng = StableRNG(42)\nend","category":"page"},{"location":"noise/","page":"Noise Processes","title":"Noise Processes","text":"As seen in the list of possible gates, the simulator is capable of modeling different types of noise. If that is your goal, please consider using the available Monte Carlo simulator or the Symbolic Perturbative Expansion system.","category":"page"},{"location":"noise/","page":"Noise Processes","title":"Noise Processes","text":"The implemented types of noise include:","category":"page"},{"location":"noise/","page":"Noise Processes","title":"Noise Processes","text":"UnbiasedUncorrelatedNoise\nPauliNoise","category":"page"},{"location":"noise/","page":"Noise Processes","title":"Noise Processes","text":"The low-level functionality to work with noise is applynoise!, but most of the time you would probably just want to use PauliError, NoisyGate, NoiseOp and NoiseOpAll.","category":"page"},{"location":"ecc_example_sim/#noisycircuits_pf_ecc_example","page":"ECC example","title":"ECC example with Pauli Frames","text":"","category":"section"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"DocTestSetup = quote\n using QuantumClifford\n using Quantikz\nend\nCurrentModule = QuantumClifford.Experimental.NoisyCircuits","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"warning: The documentation is incomplete\nWaiting for a better documentation than the small example below. Check out also the page on ECC performance evaluators","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"Consider Steane 7-qubit code:","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"using QuantumClifford\nusing QuantumClifford.ECC: Steane7, naive_syndrome_circuit, naive_encoding_circuit, parity_checks, code_s, code_n\nusing Quantikz\n\ncode = Steane7()\nH = parity_checks(code)","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"... and the corresponding encoding circuit","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"ecirc = naive_encoding_circuit(code)","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"... and the corresponding syndrome measurement circuit (the non-fault tolerant one)","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"scirc, _ = naive_syndrome_circuit(code)","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"The most straightforward way to start sampling syndromes is to set up a table of Pauli frames.","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"circuit = [ecirc..., scirc...]\nnframes = 4\nframes = pftrajectories(circuit; trajectories=nframes) # run the sims\npfmeasurements(frames) # extract the measurements","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"The pftrajectories function is multithreaded. If you want more low-level control over these Pauli frame simulations, check out the PauliFrame structure, the other methods of pftrajectories, and the circuit compactifaction function compactify_circuit.","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"If you want to model Pauli errors, use:","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"The helper PauliError for unbiased Pauli noise operation acting on a given qubit\nThe lower level NoiseOp (for a single qubit) or NoiseOpAll (for all qubits) parameterized with a particular noise type, e.g. UnbiasedUncorrelatedNoise","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"errprob = 0.1\nerrors = [PauliError(i,errprob) for i in 1:code_n(code)]\nfullcircuit = [ecirc..., errors..., scirc...]","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"And running this noisy simulation:","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"frames = pftrajectories(fullcircuit; trajectories=nframes)\npfmeasurements(frames)","category":"page"},{"location":"stab-algebra-manual/#Stabilizer-Tableau-Algebra-Manual","page":"Manual","title":"Stabilizer Tableau Algebra Manual","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"DocTestSetup = quote\n using QuantumClifford\nend","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The library consists of two main parts: Tools for working with the algebra of Stabilizer tableaux and tools specifically for efficient Circuit Simulation. This chapter discusses the former \"lower level\" Stabilizer tableau algebra tools.","category":"page"},{"location":"stab-algebra-manual/#Pauli-Operators","page":"Manual","title":"Pauli Operators","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The PauliOperator object represents multi-qubit Pauli operator (1iIZXY^otimes n). It is stored in memory as a phase (a single byte where 0x0,0x1,0x2,0x3 corresponds to 1i-1-i) and two bit-arrays, for X and for Z components.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"You can create them with a P string.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> P\"-iXZ\"\n-iXZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Or by specifying phase and X/Z components:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> PauliOperator(0x0,Bool[0,1,0],Bool[0,0,1])\n+ _XZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Both underscore and I can be used for identity.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> P\"I_XYZ\"\n+ __XYZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Multiplication with scalars or other Pauli operators works as expected, as well as tensor products of Pauli operators.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> -1im*P\"X\"\n-iX\n\njulia> P\"X\" * P\"Z\"\n-iY\n\njulia> P\"X\" ⊗ P\"Z\"\n+ XZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"One can check for commutativity with comm.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> comm(P\"X\",P\"Z\")\n0x01\n\njulia> comm(P\"XX\",P\"ZZ\")\n0x00","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"And check the phase of a product with prodphase.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> prodphase(P\"X\", P\"Z\")\n0x03\n\njulia> prodphase(P\"X\", P\"iZ\")\n0x00\n\njulia> prodphase(P\"X\",P\"Y\")\n0x01","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Indexing operations are available.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> p = P\"IXYZ\";\n\njulia> p[1], p[2], p[3], p[4]\n((false, false), (true, false), (true, true), (false, true))\n\njulia> p = P\"III\";\n\njulia> p[2] = (true, true);\n\njulia> p\n+ _Y_","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Including fancy indexing:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> P\"IXYZ\"[[2,3]]\n+ XY\n\njulia> P\"IXYZ\"[[false,true,true,false]]\n+ XY","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The operator is represented in memory by bit arrays (much denser than using byte arrays).","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> p = P\"-IXYZ\";\n\njulia> p.nqubits, p.xz\n(4, UInt64[0x0000000000000006, 0x000000000000000c])","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Views that give just the X or Z components of the xz bitarray are available through xview and zview.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> xview(P\"XYZI\")\n1-element view(::Vector{UInt64}, 1:1) with eltype UInt64:\n 0x0000000000000003","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The convenience methods xbit and zbit give you Bool (GF2) vectors.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> xbit(P\"XYZI\")\n4-element Vector{Bool}:\n 1\n 1\n 0\n 0","category":"page"},{"location":"stab-algebra-manual/#Stabilizers","page":"Manual","title":"Stabilizers","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"A Stabilizer object is a tableau of Pauli operators. When the tableau is meant to represent a (pure or mixed) stabilizer state, all of these operators should commute (but that is not enforced, rather Stabilizer is a generic tableau data structure). It is stored in memory as a phase list and a bit-matrix for X and Z components. It can be instantiated by an S string, or with a number of different constructors.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"tip: Stabilizers and Destabilizers\nIn many cases you probably would prefer to use the MixedDestabilizer data structure, as it caries a lot of useful additional information, like tracking rank and destabilizer operators. Stabilizer has mostly a pedagogical value, and it is also used for slightly faster simulation of a particular subset of Clifford operations. See also the data structures discussion page.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> S\"-XX\n +ZZ\"\n- XX\n+ ZZ\n\njulia> Stabilizer([P\"-XX\",P\"+ZZ\"])\n- XX\n+ ZZ\n\njulia> Stabilizer([0x2, 0x0],\n Bool[1 1;\n 0 0],\n Bool[0 0;\n 1 1])\n- XX\n+ ZZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Direct sums can be performed,","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> S\"-XX\" ⊗ S\"ZZ\"\n- XX__\n+ __ZZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Indexing operations are available, including fancy indexing. Be careful about how phase information gets transferred during sub-indexing.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XYZ\n -ZIX\n +XIZ\";\n\njulia> s[1]\n- XYZ\n\njulia> s[1,2]\n(true, true)\n\njulia> s[[3,1]]\n+ X_Z\n- XYZ\n\njulia> s[[3,1],[2]]\n+ _\n- Y","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Consistency at creation is not verified so nonsensical stabilizers can be created, both in terms of content and shape.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> S\"iX\n +Z\"\n+iX\n+ Z","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Similarly to the Pauli operators, a bit array representation is used.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZI\n -IZZ\"\n- XXX\n+ ZZ_\n- _ZZ\n\njulia> phases(s), tab(s).xzs\n(UInt8[0x02, 0x00, 0x02], UInt64[0x0000000000000007 0x0000000000000000 0x0000000000000000; 0x0000000000000000 0x0000000000000003 0x0000000000000006])","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"And there are convenience functions that can extract the corresponding binary check matrix.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> stab_to_gf2(s)\n3×6 Matrix{Bool}:\n 1 1 1 0 0 0\n 0 0 0 1 1 0\n 0 0 0 0 1 1","category":"page"},{"location":"stab-algebra-manual/#Canonicalization-of-Stabilizers","page":"Manual","title":"Canonicalization of Stabilizers","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Canonicalization (akin to Gaussian elimination over F(2,2)) is implemented in the canonicalize! function. Besides the default canonicalization prescription, alternative ones are available as described in the canonicalization page.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZX\n +III\";\n\njulia> canonicalize!(s)\n+ YY_\n+ ZZX\n+ ___","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"If phases are inconsequential, the operations can be faster by not tracking and updating them.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZX\n +III\";\n\njulia> canonicalize!(s; phases=false)\n- YY_\n+ ZZX\n+ ___","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"These operations are in place (as customarily signified by \"!\").","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZX\n +III\";\n\njulia> canonicalize!(s; phases=false);\n\njulia> s\n- YY_\n+ ZZX\n+ ___","category":"page"},{"location":"stab-algebra-manual/#Projective-Measurements","page":"Manual","title":"Projective Measurements","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The project! function is used to perform generic projective measurements.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"tip: Single qubit projections\nIf you know your Pauli measurement operator acts on a single qubit, there are much faster projection functions available, discussed in the next section. Namely projectX!, projectY!, and projectZ!.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"To observe the effect of different projections, we will start with a GHZ state.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZI\n -IZZ\";","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The project! function returns the new stabilizer, the index where the anticommutation was detected, and the result of the projection (nothing being an undetermined result). For instance here we project on an operator that does not commute with all stabilizer generators.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(copy(s), P\"ZII\")[1]\n+ Z__\n+ ZZ_\n- _ZZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Importantly, when there is an undetermined result, we return nothing and leave the phase of the new stabilizer the same as the phase of the projection operator. If you want to perform a Monte Carlo simulation, you need to randomize the phase of the stabilizer at the anticommuting index yourself. For instance, one can do:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> newstate, anticomindex, result = project!(copy(s), P\"XII\")\n if isnothing(result)\n phases(newstate)[anticomindex] = rand([0x0,0x2])\n end\n result, anticomindex\n(nothing, 2)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Of course, this is a rather cumbersome way to run a simulation, so we also provide projectrand! which does the necessary randomization automatically, for cases where you do not need the fine grained control of project!.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"We can project on a commuting operator, hence no anticommuting terms (the index is zero), and the result is perfectly determined (-1, or in our convention to represent the phase, 0x2).","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(copy(s), P\"-ZZI\")\n(Stabilizer 3×3, 0, 0x02)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"When the projection is consistent with the stabilizer (i.e. the measurement result is not nothing), this would trigger an expensive canonicalization procedure in order to calculate the measurement result (unless we are using more advanced data structures to represent the state, which are discussed later). If all you want to know is whether the projection is consistent with the stabilizer, but you do not care about the measurement result, you can skip the canonicalization and calculation of the result.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(copy(s), P\"-ZZI\", keep_result=false)\n(Stabilizer 3×3, 0, nothing)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Lastly, in either case, you can skip the calculation of the phases as well, if they are unimportant.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(copy(s), P\"ZZI\", phases=false)\n(Stabilizer 3×3, 0, 0x00)","category":"page"},{"location":"stab-algebra-manual/#Sparse-single-qubit-measurements","page":"Manual","title":"Sparse single-qubit measurements","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"In many circumstances only a single-qubit operator is being measured. In that case one should use the projectX!, projectY!, and projectZ! functions as they are much faster thanks to tracking only a single qubit. They have versions that randomize the phase as necessary as well: projectXrand!, projectYrand!, and projectZrand!.","category":"page"},{"location":"stab-algebra-manual/#Gate-like-interface","page":"Manual","title":"Gate-like interface","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"If you do not need all this boilerplate, and especially if you want to perform the randomization automatically, you can use the gate-like \"symbolic\" objects sMX, sMY, and sMZ, that perform the measurement and the necessary randomization of phase. If the measurement result is to be stored, you can use the Register structure that stores both stabilizer tableaux and bit values.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> state = Register(ghz(3), [false,false])\nRegister{Vector{UInt8}, Matrix{UInt64}}(Rank 3 stabilizer\n+ Z__\n+ _X_\n+ __X\n═════\n+ XXX\n+ ZZ_\n+ Z_Z\n═════\n, Bool[0, 0])\n\njulia> apply!(state, sMX(3,2)) # which qubit is measured and in which bit it is stored\nRegister{Vector{UInt8}, Matrix{UInt64}}(Rank 3 stabilizer\n+ Z__\n+ _X_\n+ Z_Z\n═════\n+ XXX\n+ ZZ_\n- __X\n═════\n, Bool[0, 1])\n\njulia> bitview(state)\n2-element Vector{Bool}:\n 0\n 1","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Or you can use the projectXrand!, projectYrand!, and projectZrand! if you prefer a function-call interface.","category":"page"},{"location":"stab-algebra-manual/#Partial-Traces","page":"Manual","title":"Partial Traces","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Partial trace (using traceout!) over even a single qubit might cause many of them to decohere due to entanglement.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> ghz = S\"XXX\n ZZ_\n _ZZ\";\n\njulia> traceout!(ghz, [1])\n+ _ZZ\n+ ___\n+ ___","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"This is somewhat more elegant when the datastructure being used explicitly supports mixed states.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> ghz = MixedStabilizer(S\"XXX\n ZZ_\n _ZZ\");\n\njulia> traceout!(ghz, [1])\n+ _ZZ","category":"page"},{"location":"stab-algebra-manual/#Generating-a-Pauli-Operator-with-Stabilizer-Generators","page":"Manual","title":"Generating a Pauli Operator with Stabilizer Generators","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The generate! function attempts to generate a Pauli operator by multiplying together the operators belonging to a given stabilizer (or reports their independence). This particular function requires the stabilizer to be already canonicalized.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZI\n -IZZ\";\n\njulia> s = canonicalize!(s)\n- XXX\n- Z_Z\n- _ZZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"It modifies the Pauli operator in place, reducing it to identity if possible. The leftover phase is present to indicate if the phase itself could not have been canceled. The list of indices specifies which rows of the stabilizer were used to generated the desired Pauli operator.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> generate!(P\"XYY\", s)\n(- ___, [1, 3])","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Phases can be neglected, for higher performance.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> generate!(P\"XYY\", s, phases=false)\n(+ ___, [1, 3])","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"If the Pauli operator can not be generated by the stabilizer, nothing value is returned.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> generate!(P\"ZZZ\", s)\n\njulia> generate!(P\"XZX\", s)\n\njulia> generate!(P\"YYY\", s)","category":"page"},{"location":"stab-algebra-manual/#Clifford-Operators","page":"Manual","title":"Clifford Operators","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The CliffordOperator structure represents a linear mapping between stabilizers (which should also preserve commutation relationships, but that is not checked at instantiation). These are n-qubit dense tableaux, representing an operation on n-qubit states. For single- or two-qubit gates, it is much more efficient to use small sparse symbolic clifford operators. A number of predefined Clifford operators are available, their name prefixed with t to mark them as dense tableaux.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> tHadamard\nX₁ ⟼ + Z\nZ₁ ⟼ + X\n\njulia> tPhase\nX₁ ⟼ + Y\nZ₁ ⟼ + Z\n\njulia> tCNOT\nX₁ ⟼ + XX\nX₂ ⟼ + _X\nZ₁ ⟼ + Z_\nZ₂ ⟼ + ZZ\n\njulia> tId1\nX₁ ⟼ + X\nZ₁ ⟼ + Z","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Chaining and tensor products are possible. Same for qubit permutations.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> tHadamard ⊗ tPhase\nX₁ ⟼ + Z_\nX₂ ⟼ + _Y\nZ₁ ⟼ + X_\nZ₂ ⟼ + _Z\n\njulia> tHadamard * tPhase\nX₁ ⟼ - Y\nZ₁ ⟼ + X\n\njulia> permute(tCNOT, [2,1])\nX₁ ⟼ + X_\nX₂ ⟼ + XX\nZ₁ ⟼ + ZZ\nZ₂ ⟼ + _Z","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"You can create custom Clifford operators with C-strings or with a list of Pauli operators.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> C\"-ZZ\n +_Z\n -X_\n +XX\"\nX₁ ⟼ - ZZ\nX₂ ⟼ + _Z\nZ₁ ⟼ - X_\nZ₂ ⟼ + XX\n\njulia> CliffordOperator([P\"-ZZ\", P\"_Z\", P\"-X_\", P\"XX\"])\nX₁ ⟼ - ZZ\nX₂ ⟼ + _Z\nZ₁ ⟼ - X_\nZ₂ ⟼ + XX","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Naturally, the operators can be applied to stabilizer states. This includes high performance in-place operations (and the phase can be neglected with phases=false for faster computation).","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> tCNOT * S\"X_\"\n+ XX\n\njulia> s = S\"X_\";\n\njulia> apply!(s,tCNOT)\n+ XX","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Sparse applications where a small Clifford operator is applied only on a particular subset of a larger stabilizer is also possible, but in such circumstances it is useful to consider using symbolic operators too.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"Z_YX\";\n\njulia> apply!(s, tCNOT, [4,2]) # Apply the CNOT on qubits 4 and 2\n+ ZXYX","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Pauli operators act as Clifford operators too (but they are rather boring, as they only change signs).","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> P\"XII\" * S\"ZXX\"\n- ZXX","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Internally, the CliffordOperator structure simply stores the tableau representation of the operation.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The apply! function is efficiently multithreaded for CliffordOperators. To start multithreaded Julia, use julia -t where specifies the number of threads.","category":"page"},{"location":"stab-algebra-manual/#Symbolic-Clifford-Operators","page":"Manual","title":"Symbolic Clifford Operators","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Much faster implementations for a number of common Clifford operators are available. They are stored as special named structs, instead of as a full tableau. These are the subtypes of AbstractSingleQubitOperator and AbstractTwoQubitOperator. Currently these are:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"using QuantumClifford # hide\nusing InteractiveUtils # hide\nsubtypes(QuantumClifford.AbstractSingleQubitOperator)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"subtypes(QuantumClifford.AbstractTwoQubitOperator)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Generally, they have the prefix s for symbolic/small/sparse. They are used slightly differently, as one needs to specify the qubits on which they act while instantiating them:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> sHadamard(2)\nsHadamard on qubit 2\nX₁ ⟼ + Z\nZ₁ ⟼ + X\n\njulia> sHadamard(2)*S\"XXX\"\n+ XZX\n\njulia> sCNOT(2,3)*S\"XYY\"\n- XXZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The apply! function is efficiently multithreaded for these symbolic operators as well. To start multithreaded Julia, use julia -t where specifies the number of threads.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Symbolic projectors on single qubits also exist: sMX, sMY, sMZ. When used with the Register state representation, they can store the measurement results in the corresponding classical register.","category":"page"},{"location":"stab-algebra-manual/#Destabilizers","page":"Manual","title":"Destabilizers","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Slightly abusing the name: What we call \"destabilizers\" here is a stabilizer and its destabilizing operators saved together. They are implemented with the Destabilizer object and are initialized from a stabilizer.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s=S\"-XXX\n -ZZI\n +IZZ\";\n\njulia> d = Destabilizer(s)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ _X_\n+ __X\n𝒮𝓉𝒶𝒷━\n- XXX\n- ZZ_\n- Z_Z","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"They have convenience methods to extract only the stabilizer and destabilizer pieces:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> stabilizerview(d)\n- XXX\n- ZZ_\n- Z_Z\n\njulia> destabilizerview(d)\n+ Z__\n+ _X_\n+ __X","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Importantly commuting projections are much faster when tracking the destabilizer as canonicalization is not necessary (an mathcalO(n^2) complexity because it avoids the expensive mathcalO(n^3) canonicalization operation).","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(d,P\"ZZI\")\n(Destablizer 3×3, 0, 0x02)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Non-commuting projections are just as fast as when using only stabilizers.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(d,P\"ZZZ\")\n(Destablizer 3×3, 1, nothing)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Clifford operations can be applied the same way they are applied to stabilizers.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> apply!(d,tCNOT⊗tHadamard)\n𝒟ℯ𝓈𝓉𝒶𝒷\n- X_Z\n+ XXZ\n+ X__\n𝒮𝓉𝒶𝒷━\n+ _ZX\n- _Z_\n- Z_X","category":"page"},{"location":"stab-algebra-manual/#Mixed-States","page":"Manual","title":"Mixed States","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Both the Stabilizer and Destabilizer structures have more general forms that enable work with mixed stabilizer states. They are the MixedStabilizer and MixedDestabilizer structures, described in Mixed States. More information that can be seen in the data structures page, which expands upon the algorithms available for each structure.","category":"page"},{"location":"stab-algebra-manual/#Random-States-and-Circuits","page":"Manual","title":"Random States and Circuits","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"random_clifford, random_stabilizer, and enumerate_cliffords can be used for the generation of random states.","category":"page"},{"location":"tutandpub/#tutandpub","page":"Tutorials and Publications","title":"Tutorials and Publications","text":"","category":"section"},{"location":"tutandpub/","page":"Tutorials and Publications","title":"Tutorials and Publications","text":"This list has a number of notebooks with tutorials, examples, and reproduction of published results (some of these results originally obtained with this very library).","category":"page"},{"location":"tutandpub/#On-the-topic-of-explicit-use-of-the-Tableaux-formalism-for-Stabilizer-states","page":"Tutorials and Publications","title":"On the topic of explicit use of the Tableaux formalism for Stabilizer states","text":"","category":"section"},{"location":"tutandpub/","page":"Tutorials and Publications","title":"Tutorials and Publications","text":"Quantum coding with low-depth random circuits reproducing results from (Gullans et al., 2020). view on nbviewer.jupyter.org","category":"page"},{"location":"tutandpub/#On-the-Monte-Carlo-and-Perturbative-Expansions-for-**Noisy**-Clifford-circuits","page":"Tutorials and Publications","title":"On the Monte Carlo and Perturbative Expansions for Noisy Clifford circuits","text":"","category":"section"},{"location":"tutandpub/","page":"Tutorials and Publications","title":"Tutorials and Publications","text":"In-depth study of multi-partite entanglement purification circuits reproducing results from (Krastanov et al., 2020). view on nbviewer.jupyter.org\nComparing the Monte Carlo and Perturbative method for noisy circuit simulations. view on nbviewer.jupyter.org\nShowcasing symbolic perturbative expansions of noisy circuits. view on nbviewer.jupyter.org","category":"page"},{"location":"noisycircuits_API/#Full-API-(autogenerated)","page":"API","title":"Full API (autogenerated)","text":"","category":"section"},{"location":"noisycircuits_API/","page":"API","title":"API","text":"warning: Unstable\nThis is experimental functionality with an unstable API.","category":"page"},{"location":"noisycircuits_API/","page":"API","title":"API","text":"Modules = [QuantumClifford.Experimental.NoisyCircuits]\nPrivate = false","category":"page"},{"location":"noisycircuits_API/#QuantumClifford.Experimental.NoisyCircuits.ConditionalGate","page":"API","title":"QuantumClifford.Experimental.NoisyCircuits.ConditionalGate","text":"A conditional gate that either performs truegate or falsegate, depending on the value of controlbit.\n\n\n\n\n\n","category":"type"},{"location":"noisycircuits_API/#QuantumClifford.Experimental.NoisyCircuits.DecisionGate","page":"API","title":"QuantumClifford.Experimental.NoisyCircuits.DecisionGate","text":"A conditional gate that performs one of the gates, depending on the output of decisionfunction applied to the entire classical bit register.\n\n\n\n\n\n","category":"type"},{"location":"noisycircuits_API/#QuantumClifford.Experimental.NoisyCircuits.NoisyBellMeasurement","page":"API","title":"QuantumClifford.Experimental.NoisyCircuits.NoisyBellMeasurement","text":"A Bell measurement in which each of the measured qubits has a chance to have flipped.\n\n\n\n\n\n","category":"type"},{"location":"noisycircuits_perturb/#noisycircuits_perturb","page":"Perturbative Expansions","title":"Perturbative expansions for simulating noisy Clifford circuits","text":"","category":"section"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"DocTestSetup = quote\n using QuantumClifford\n using QuantumClifford.Experimental.NoisyCircuits\n using Quantikz\nend\nCurrentModule = QuantumClifford.Experimental.NoisyCircuits","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"warning: Unstable\nThis is experimental functionality with an unstable API.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"Import with using QuantumClifford.Experimental.NoisyCircuits.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"This module enables the simulation of noisy Clifford circuits through a perturbative expansion in the noise parameter (assuming the noise is small). Instead of simulating many Monte Carlo trajectories, only the leading order trajectories are exhaustively enumerated and simulated.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"Here is an example of a purification circuit (the same circuit seen in the Monte Carlo example)","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\ngood_bell_state = S\"XX\n ZZ\"\ncanonicalize_rref!(good_bell_state)\ninitial_state = MixedDestabilizer(good_bell_state⊗good_bell_state)\n\ng1 = sCNOT(1,3) # CNOT between qubit 1 and qubit 3 (both with Alice)\ng2 = sCNOT(2,4) # CNOT between qubit 2 and qubit 4 (both with Bob)\nm = BellMeasurement([sMX(3),sMX(4)]) # Bell measurement on qubit 3 and 4\nv = VerifyOp(good_bell_state,[1,2]) # Verify that qubit 1 and 2 indeed form a good Bell pair\nepsilon = 0.01 # The error rate\nn = NoiseOpAll(UnbiasedUncorrelatedNoise(epsilon))\n\n# This circuit performs a depolarization at rate `epsilon` to all qubits,\n# then bilater CNOT operations\n# then a Bell measurement\n# followed by checking whether the final result indeed corresponds to the correct Bell pair.\ncircuit = [n,g1,g2,m,v]\n\npetrajectories(initial_state, circuit)","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"For more examples, see the notebook comparing the Monte Carlo and Perturbative method or this tutorial on entanglement purification.","category":"page"},{"location":"noisycircuits_perturb/#Symbolic-expansions","page":"Perturbative Expansions","title":"Symbolic expansions","text":"","category":"section"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"The perturbative expansion method works with symbolic variables as well. One can use any of the symbolic libraries available in Julia and simply plug symbolic parameters in lieu of numeric parameters. A detailed example is available as a Jupyter notebook.","category":"page"},{"location":"noisycircuits_perturb/#Interface-for-custom-operations","page":"Perturbative Expansions","title":"Interface for custom operations","text":"","category":"section"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"If you want to create a custom gate type (e.g. calling it Operation), you need to definite the following methods.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"applyop_branches!(s::T, g::Operation; max_order=1)::Vector{Tuple{T,Symbol,Real,Int}} where T is a tableaux type like Stabilizer or a Register. The Symbol is the status of the operation, the Real is the probability for that branch, and the Int is the order of that branch.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"There is also applynoise_branches! which is convenient for use in NoisyGate, but you can also just make up your own noise operator simply by implementing applyop_branches! for it.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"You can also consult the list of implemented operators.","category":"page"},{"location":"graphs/#Graph-States","page":"Graph States","title":"Graph States","text":"","category":"section"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"warning: The `graphstate` API is not considered stable\ngraphstate returns a lot of information about encoding a given stabilizer state in a graph. A different API is being designed that streamlines the work with graph states.","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"Conversion to and from graph states is possible.","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"Consider a GHZ state:","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"using QuantumClifford # hide\nghz(4)","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"It can be converted to a graph state with graphstate","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"graphstate(ghz(4))[1]","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"using Random; Random.seed!(1); using QuantumClifford, GraphMakie, CairoMakie;\nf = Figure(resolution=(200,200))\na = Axis(f[1,1])\ngraphplot!(a,graphstate(ghz(4))[1])\nhidedecorations!(a); hidespines!(a)\na.aspect = DataAspect()\nsave(\"ghz4graph.png\", f); nothing","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"(Image: )","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"Notice that the initial GHZ state was not in the typical graph state form. We can see that explicitly by converting back and forth between the two forms:","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"julia> using Graphs, QuantumClifford\n\njulia> ghz(4)\n+ XXXX\n+ ZZ__\n+ _ZZ_\n+ __ZZ\n\njulia> Stabilizer(Graph(ghz(4)))\n+ XZZZ\n+ ZX__\n+ Z_X_\n+ Z__X","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"There is a set of single-qubit operations that can convert any stabilizer tableau into a state representable as a graph. These transformations are performed implicitly by the Graph constructor when converting from a Stabilizer. If you need the explicit transformation you can use the graphstate function that specifies which qubits require a Hadamard, Inverse Phase, or Phase Flip gate. The graph_gatesequence or graph_gate helper functions can be used to generate the exact operations:","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"julia> s = ghz(4)\n+ XXXX\n+ ZZ__\n+ _ZZ_\n+ __ZZ\n\njulia> g, h_idx, ip_idx, z_idx = graphstate(s);\n\njulia> gate = graph_gate(h_idx, ip_idx, z_idx, nqubits(s))\nX₁ ⟼ + X___\nX₂ ⟼ + _Z__\nX₃ ⟼ + __Z_\nX₄ ⟼ + ___Z\nZ₁ ⟼ + Z___\nZ₂ ⟼ + _X__\nZ₃ ⟼ + __X_\nZ₄ ⟼ + ___X\n\njulia> canonicalize!(apply!(s,gate)) == canonicalize!(Stabilizer(g))\ntrue","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"These converters also provides for a convenient way to create graph and cluster states, by using the helper constructors provided in Graphs.jl.","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"julia> Stabilizer(grid([4,1])) # Linear cluster state\n+ XZ__\n+ ZXZ_\n+ _ZXZ\n+ __ZX\n\njulia> Stabilizer(grid([2,2])) # Small 2D cluster state\n+ XZZ_\n+ ZX_Z\n+ Z_XZ\n+ _ZZX","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"Graphs are represented with the Graphs.jl package and plotting can be done both in Plots.jl and Makie.jl (with GraphMakie).","category":"page"},{"location":"ECC_API/#Full-ECC-API-(autogenerated)","page":"API","title":"Full ECC API (autogenerated)","text":"","category":"section"},{"location":"ECC_API/","page":"API","title":"API","text":"Modules = [QuantumClifford.ECC]\nPrivate = false","category":"page"},{"location":"ECC_API/#QuantumClifford.ECC.CSS","page":"API","title":"QuantumClifford.ECC.CSS","text":"An arbitrary CSS error correcting code defined by its X and Z checks.\n\njulia> CSS([0 1 1 0; 1 1 0 0], [1 1 1 1]) |> parity_checks\n+ _XX_\n+ XX__\n+ ZZZZ\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.Cleve8","page":"API","title":"QuantumClifford.ECC.Cleve8","text":"A pedagogical example of a quantum error correcting [8,3] code used in (Cleve and Gottesman, 1997).\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.CommutationCheckECCSetup","page":"API","title":"QuantumClifford.ECC.CommutationCheckECCSetup","text":"Configuration for ECC evaluator that does not simulate any ECC circuits, rather it simply checks the commutation of the parity check and the Pauli error.\n\nThis is much faster than any other simulation method, but it is incapable of noisy-circuit simulations and thus useless for fault-tolerance studies.\n\nSee also: NaiveSyndromeECCSetup, ShorSyndromeECCSetup\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.Gottesman","page":"API","title":"QuantumClifford.ECC.Gottesman","text":"The family of [[2ʲ, 2ʲ - j - 2, 3]] Gottesman codes, also known as quantum Hamming codes, as described in Gottesman's 1997 PhD thesis and in (Gottesman, 1996).\n\nYou might be interested in consulting (Yu et al., 2013) and (Chao and Reichardt, 2017) as well.\n\nThe ECC Zoo has an entry for this family\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.NaiveSyndromeECCSetup","page":"API","title":"QuantumClifford.ECC.NaiveSyndromeECCSetup","text":"Configuration for ECC evaluator that runs the simplest syndrome measurement circuit.\n\nThe circuit is being simulated (as opposed to doing only a quick commutation check). This circuit would give poor performance if there is non-zero gate noise.\n\nSee also: CommutationCheckECCSetup, ShorSyndromeECCSetup\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.ShorSyndromeECCSetup","page":"API","title":"QuantumClifford.ECC.ShorSyndromeECCSetup","text":"Configuration for ECC evaluators that simulate the Shor-style syndrome measurement (without a flag qubit).\n\nThe simulated circuit includes:\n\nperfect noiseless encoding (encoding and its fault tolerance are not being studied here)\none round of \"memory noise\" after the encoding but before the syndrome measurement\nperfect preparation of entangled ancillary qubits\nnoisy Shor-style syndrome measurement (only two-qubit gate noise)\nnoiseless \"logical state measurement\" (providing the comparison data when evaluating the decoder)\n\nSee also: CommutationCheckECCSetup, NaiveSyndromeECCSetup\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.TableDecoder","page":"API","title":"QuantumClifford.ECC.TableDecoder","text":"A simple look-up table decoder for error correcting codes.\n\nThe lookup table contains only weight=1 errors, thus it is small, but at best it provides only for distance=3 decoding.\n\nThe size of the lookup table would grow exponentially quickly for higher distances.\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.Toric","page":"API","title":"QuantumClifford.ECC.Toric","text":"The Toric code.\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.BeliefPropDecoder-Tuple","page":"API","title":"QuantumClifford.ECC.BeliefPropDecoder","text":"A simple Belief Propagation decoder built around tools from LDPCDecoders.jl.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.PyBeliefPropDecoder-Tuple","page":"API","title":"QuantumClifford.ECC.PyBeliefPropDecoder","text":"A Belief Propagation decoder built around tools from the python package ldpc available from the julia package PyQDecoders.jl.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.PyBeliefPropOSDecoder-Tuple","page":"API","title":"QuantumClifford.ECC.PyBeliefPropOSDecoder","text":"A Belief Propagation decoder with ordered statistics decoding, built around tools from the python package ldpc available from the julia package PyQDecoders.jl.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.PyMatchingDecoder-Tuple","page":"API","title":"QuantumClifford.ECC.PyMatchingDecoder","text":"A perfect matching decoder built around tools from the python package pymatching available from the julia package PyQDecoders.jl.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.code_k-Tuple{Any}","page":"API","title":"QuantumClifford.ECC.code_k","text":"The number of logical qubits in a code.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.code_n","page":"API","title":"QuantumClifford.ECC.code_n","text":"The number of physical qubits in a code.\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.code_s","page":"API","title":"QuantumClifford.ECC.code_s","text":"The number of stabilizer checks in a code.\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.distance","page":"API","title":"QuantumClifford.ECC.distance","text":"The distance of a code.\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.evaluate_decoder-Tuple{QuantumClifford.ECC.AbstractSyndromeDecoder, QuantumClifford.ECC.AbstractECCSetup, Int64}","page":"API","title":"QuantumClifford.ECC.evaluate_decoder","text":"Evaluate the performance of a given decoder (e.g. TableDecoder) and a given style of running an ECC code (e.g. ShorSyndromeECCSetup)\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.evaluate_decoder-Tuple{QuantumClifford.ECC.AbstractSyndromeDecoder, Vararg{Any, 5}}","page":"API","title":"QuantumClifford.ECC.evaluate_decoder","text":"Evaluate the performance of an error-correcting circuit.\n\nThis method requires you give the circuit that performs both syndrome measurements and (probably noiseless) logical state measurements. The faults matrix that translates an error vector into corresponding logical errors is necessary as well.\n\nThis is a relatively barebones method that assumes the user prepares necessary circuits, etc. It is a method that is used internally by more user-frienly methods providing automatic conversion of codes and noise models to the necessary noisy circuits.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.faults_matrix-Tuple{Stabilizer}","page":"API","title":"QuantumClifford.ECC.faults_matrix","text":"Error-to-logical-observable map (a.k.a. fault matrix) of a code.\n\nFor a code with n physical qubits and k logical qubits this function returns a 2k × 2n binary matrix O such that O[i,j] is true if the logical observable of index i is flipped by the single physical qubit error of index j. Indexing is such that:\n\nO[1:k,:] is the error-to-logical-X-observable map (logical X observable, i.e. triggered by logical Z errors)\nO[k+1:2k,:] is the error-to-logical-Z-observable map\nO[:,1:n] is the X-physical-error-to-logical-observable map\nO[n+1:2n,:] is the Z-physical-error-to-logical-observable map\n\nE.g. for k=1, n=10, then if O[2,5] is true, then the logical Z observable is flipped by a X₅ error; and if O[1,12] is true, then the logical X observable is flipped by a Z₂ error.\n\nOf note is that there is a lot of freedom in choosing the logical operations! A logical operator multiplied by a stabilizer operator is still a logical operator. Similarly there is a different fault matrix for each choice of logical operators. But once the logical operators are picked, the fault matrix is fixed.\n\nBelow we show an example that uses the Shor code. While it is not the smallest code, it is a convenient choice to showcase the importance of the fault matrix when dealing with degenerate codes where a correction operation and an error do not need to be the same.\n\nFirst, consider a single-qubit error, potential correction operations, and their effect on the Shor code:\n\njulia> using QuantumClifford.ECC: faults_matrix, Shor9\n\njulia> state = MixedDestabilizer(Shor9())\n𝒟ℯ𝓈𝓉𝒶𝒷━━━━━\n+ Z________\n+ ___Z_____\n+ _X_______\n+ __X______\n+ ____X____\n+ _____X___\n+ ______X__\n+ _______X_\n𝒳ₗ━━━━━━━━━\n+ ______XXX\n𝒮𝓉𝒶𝒷━━━━━━━\n+ XXX___XXX\n+ ___XXXXXX\n+ ZZ_______\n+ Z_Z______\n+ ___ZZ____\n+ ___Z_Z___\n+ ______Z_Z\n+ _______ZZ\n𝒵ₗ━━━━━━━━━\n+ Z__Z____Z\n\njulia> err_Z₁ = single_z(9,1) # the error we will simulate\n+ Z________\n\njulia> cor_Z₂ = single_z(9,2) # the correction operation we will perform\n+ _Z_______\n\njulia> err_Z₁ * state # observe that one of the syndrome bits is now flipped\n𝒟ℯ𝓈𝓉𝒶𝒷━━━━━\n+ Z________\n+ ___Z_____\n+ _X_______\n+ __X______\n+ ____X____\n+ _____X___\n+ ______X__\n+ _______X_\n𝒳ₗ━━━━━━━━━\n+ ______XXX\n𝒮𝓉𝒶𝒷━━━━━━━\n- XXX___XXX\n+ ___XXXXXX\n+ ZZ_______\n+ Z_Z______\n+ ___ZZ____\n+ ___Z_Z___\n+ ______Z_Z\n+ _______ZZ\n𝒵ₗ━━━━━━━━━\n+ Z__Z____Z\n\njulia> cor_Z₂ * err_Z₁ * state # we are back to a good code state\n𝒟ℯ𝓈𝓉𝒶𝒷━━━━━\n+ Z________\n+ ___Z_____\n- _X_______\n+ __X______\n+ ____X____\n+ _____X___\n+ ______X__\n+ _______X_\n𝒳ₗ━━━━━━━━━\n+ ______XXX\n𝒮𝓉𝒶𝒷━━━━━━━\n+ XXX___XXX\n+ ___XXXXXX\n+ ZZ_______\n+ Z_Z______\n+ ___ZZ____\n+ ___Z_Z___\n+ ______Z_Z\n+ _______ZZ\n𝒵ₗ━━━━━━━━━\n+ Z__Z____Z\n\njulia> bad_Z₆Z₉ = single_z(9,6) * single_z(9,9) # a different \"correction\" operation\n+ _____Z__Z\n\njulia> bad_Z₆Z₉ * err_Z₁ * state # the syndrome is trivial, but now we have a logical error\n𝒟ℯ𝓈𝓉𝒶𝒷━━━━━\n+ Z________\n+ ___Z_____\n+ _X_______\n+ __X______\n+ ____X____\n- _____X___\n+ ______X__\n+ _______X_\n𝒳ₗ━━━━━━━━━\n- ______XXX\n𝒮𝓉𝒶𝒷━━━━━━━\n+ XXX___XXX\n+ ___XXXXXX\n+ ZZ_______\n+ Z_Z______\n+ ___ZZ____\n+ ___Z_Z___\n+ ______Z_Z\n+ _______ZZ\n𝒵ₗ━━━━━━━━━\n+ Z__Z____Z\n\nThe success of cor_Z₂ and the failure of bad_Z₆Z₉ can be immediately seen through the fault matrix, as the wrong \"correction\" does not result in the same logical flips ad the error:\n\njulia> O = faults_matrix(Shor9())\n2×18 BitMatrix:\n 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 1\n 1 0 0 1 0 0 0 0 1 0 0 0 0 0 0 0 0 0\n\njulia> O * stab_to_gf2(err_Z₁)\n2-element Vector{Int64}:\n 0\n 0\n\njulia> O * stab_to_gf2(cor_Z₂)\n2-element Vector{Int64}:\n 0\n 0\n\njulia> O * stab_to_gf2(bad_Z₆Z₉)\n2-element Vector{Int64}:\n 1\n 0\n\nWhile its use in this situation is rather contrived, the fault matrix is incredibly useful when running large scale simulations in which we want a separate fast error sampling process, (e.g. with Pauli frames) and a syndrome decoding process, without coupling between them. We just gather all our syndrome measurement and logical observables from the Pauli frame simulations, and then use them with the fault matrix in the syndrome decoding simulation.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.isdegenerate","page":"API","title":"QuantumClifford.ECC.isdegenerate","text":"Check if the code is degenerate with respect to a given set of error or with respect to all \"up to d physical-qubit\" errors (defaulting to d=1).\n\njulia> using QuantumClifford.ECC\n\njulia> isdegenerate(Shor9(), [single_z(9,1), single_z(9,2)])\ntrue\n\njulia> isdegenerate(Shor9(), [single_z(9,1), single_x(9,1)])\nfalse\n\njulia> isdegenerate(Steane7(), 1)\nfalse\n\njulia> isdegenerate(Steane7(), 2)\ntrue\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.naive_encoding_circuit-Tuple{Any}","page":"API","title":"QuantumClifford.ECC.naive_encoding_circuit","text":"Encoding physical qubits into a larger logical code.\n\nThe initial physical qubits to be encoded have to be at indices n-k+1:n.\n\ninfo: Encoding circuits are not fault-tolerant\nEncoding circuits are not fault-tolerant, and thus should not be used in practice. Instead, you should measure the stabilizers of the code and the logical observables, thus projecting into the code space (which can be fault-tolerant).\n\nThe canonicalization operation performed on the code may permute the qubits (see canonicalize_gott!). That permutation is corrected for with SWAP gates by default (controlled by the undoperm keyword argument).\n\nBased on (Cleve and Gottesman, 1997) and (Gottesman, 1997), however it seems the published algorithm has some errors. Consult the erratum, as well as the more recent (Grassl, 2002) and (Grassl, 2011), and be aware that this implementation also uses H instead of Z gates.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.naive_syndrome_circuit","page":"API","title":"QuantumClifford.ECC.naive_syndrome_circuit","text":"Generate the non-fault-tolerant stabilizer measurement cicuit for a given code instance or parity check tableau.\n\nUse the ancillary_index and bit_index arguments to offset where the corresponding part the circuit starts.\n\nReturns the circuit, the number of ancillary qubits that were added, and a list of bit indices that will store the measurement results.\n\nSee also: shor_syndrome_circuit\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.parity_checks","page":"API","title":"QuantumClifford.ECC.parity_checks","text":"Parity check tableau of a code.\n\nSee also: parity_checks_x and parity_checks_z\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.parity_checks_x-Tuple{QuantumClifford.ECC.AbstractECC}","page":"API","title":"QuantumClifford.ECC.parity_checks_x","text":"Parity check boolean matrix of a code (only the X entries in the tableau, i.e. the checks for Z errors).\n\nOnly CSS codes have this method.\n\nSee also: parity_checks\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.parity_checks_z-Tuple{QuantumClifford.ECC.AbstractECC}","page":"API","title":"QuantumClifford.ECC.parity_checks_z","text":"Parity check boolean matrix of a code (only the Z entries in the tableau, i.e. the checks for X errors).\n\nOnly CSS codes have this method.\n\nSee also: parity_checks\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.rate-Tuple{Any}","page":"API","title":"QuantumClifford.ECC.rate","text":"The rate of a code.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.shor_syndrome_circuit","page":"API","title":"QuantumClifford.ECC.shor_syndrome_circuit","text":"Generate the Shor fault-tolerant stabilizer measurement cicuit for a given code instance or parity check tableau.\n\nUse the ancillary_index and bit_index arguments to offset where the corresponding part the circuit starts. Ancillary qubits\n\nReturns:\n\nThe ancillary cat state preparation circuit.\nThe Shor syndrome measurement circuit.\nThe number of ancillary qubits that were added.\nThe list of bit indices that store the final measurement results.\n\nSee also: naive_syndrome_circuit\n\n\n\n\n\n","category":"function"},{"location":"allops/#all-operations","page":"All Gates","title":"Operations - Gates, Measurements, and More","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"DocTestSetup = quote\n using QuantumClifford\n using StableRNGs\n rng = StableRNG(42)\nend","category":"page"},{"location":"allops/#Operations","page":"All Gates","title":"Operations","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Acting on quantum states can be performed either:","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"In a \"linear algebra\" language where unitaries, measurements, and other operations have separate interfaces. This is an explicitly deterministic lower-level interface, which provides a great deal of control over how tableaux are manipulated. See the Stabilizer Tableau Algebra Manual as a primer on these approaches.\nOr in a \"circuit\" language, where the operators (and measurements and noise) are represented as circuit gates. This is a higher-level interface in which the outcome of an operation can be stochastic. The API for it is centered around the apply! function. Particularly useful for Monte Carlo simulations and Perturbative Expansion Symbolic Results.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"In the circuit language, all operations can be applied on a state with the apply! function. Whether they are deterministic and their computational complexity is listed in the table below. A list of lower-level \"linear algebra style\" functions for more control over how an operation is performed is also given.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Type Deterministic 𝒪(nˣ) Low-level functions\nAbstractOperation \n├─ AbstractCliffordOperator \n│ ├─ AbstractSymbolicOperator \n│ │ ├─ AbstractSingleQubitOperator \n│ │ │ ├─ SingleQubitOperator ✔️ n \n│ │ │ ├─ sHadamard ✔️ n \n│ │ │ ├─ sId1 ✔️ n \n│ │ │ ├─ sInvPhase ✔️ n \n│ │ │ ├─ sPhase ✔️ n \n│ │ │ ├─ sX ✔️ n \n│ │ │ ├─ sY ✔️ n \n│ │ │ └─ sZ ✔️ n \n│ │ └─ AbstractTwoQubitOperator \n│ │ ├─ sCNOT ✔️ n \n│ │ ├─ sCPHASE ✔️ n \n│ │ └─ sSWAP ✔️ n \n│ │ \n│ ├─ CliffordOperator ✔️ n³ \n│ ├─ PauliOperator ✔️ n² \n│ └─ SparseGate ✔️ kn² \n├─ AbstractMeasurement \n│ ├─ PauliMeasurement ❌ n² project!, projectrand!\n│ ├─ sMX ❌ n² projectX!\n│ ├─ sMY ❌ n² projectY!\n│ └─ sMZ ❌ n² projectZ!\n│ \n├─ BellMeasurement ❌ n² \n├─ NoiseOp ❌ ? applynoise!\n├─ NoiseOpAll ❌ ? applynoise!\n├─ NoisyGate ❌ ? applynoise!\n└─ Reset ✔️ kn² reset_qubits!","category":"page"},{"location":"allops/#Details-of-Operations-Supported-by-[apply!](@ref)","page":"All Gates","title":"Details of Operations Supported by apply!","text":"","category":"section"},{"location":"allops/#Unitary-Gates","page":"All Gates","title":"Unitary Gates","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"We distinguish between symbolic gates like sCNOT that have specialized (fast) apply! methods (usually just for single and two qubit gates) and general tableau representation of gates like CliffordOperator that can represent any multi-qubit gate.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Predefined unitary gates are available, like sCNOT, sHadamard, etc.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\n[sCNOT(2,4),sHadamard(2),sCPHASE(1,3),sSWAP(2,4)]","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Any arbitrary tableaux can be used as a gate too. ","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"They can be specified by giving a Clifford operator tableaux and the indices on which it acts (particularly useful for gates acting on a small part of a circuit):","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\nSparseGate(tCNOT, [2,4])","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"The Clifford operator tableaux can be completely arbitrary.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"SparseGate(random_clifford(3), [2,4,5])","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"If the Clifford operator acts on all qubits, we do not need to specify indices, just use the operator.","category":"page"},{"location":"allops/#Noisy-Gates","page":"All Gates","title":"Noisy Gates","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Each gate can be followed by noise applied to the qubits on which it has acted. This is done by wrapping the given gate into a NoisyGate","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"ε = 0.03 # X/Y/Z error probability\nnoise = UnbiasedUncorrelatedNoise(ε)\nnoisy_gate = NoisyGate(SparseGate(tCNOT, [2,4]), noise)","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"In circuit diagrams the noise is not depicted, but after each application of the gate defined in noisy_gate, a noise operator will also be applied. The example above is of Pauli Depolarization implemented by UnbiasedUncorrelatedNoise.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"One can also apply only the noise operator by using NoiseOp which acts only on specified qubits. Or alternatively, one can use NoiseOpAll in order to apply noise to all qubits.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"[NoiseOp(noise, [4,5]), NoiseOpAll(noise)]","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"The machinery behind noise processes and different types of noise is detailed in the section on noise","category":"page"},{"location":"allops/#Coincidence-Measurements","page":"All Gates","title":"Coincidence Measurements","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Global parity measurements involving single-qubit projections and classical communication are implemented with BellMeasurement. One needs to specify the axes of measurement and the qubits being measured. If the parity is trivial, the circuit continues, if the parity is non-trivial, the circuit ends and reports a detected failure. This operator is frequently used in the simulation of entanglement purification.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"BellMeasurement([sMX(1), sMY(3), sMZ(4)])","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"There is also NoisyBellMeasurement that takes the bit-flip probability of a single-qubit measurement as a third argument.","category":"page"},{"location":"allops/#Stabilizer-Measurements","page":"All Gates","title":"Stabilizer Measurements","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"A measurement over one or more qubits can also be performed, e.g., a direct stabilizer measurement on multiple qubits without the use of ancillary qubits. When applied to multiple qubits, this differs from BellMeasurement as it performs a single projection, unlike BellMeasurement which performs a separate projection for every single qubit involved. This measurement is implemented in PauliMeasurement which requires a Pauli operator on which to project and the index of the classical bit in which to store the result. Alternatively, there are sMX, sMZ, sMY if you are measuring a single qubit.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"[PauliMeasurement(P\"XYZ\", 1), sMZ(2, 2)]","category":"page"},{"location":"allops/#Reset-Operations","page":"All Gates","title":"Reset Operations","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"The Reset operations lets you trace out the specified qubits and set their state to a specific tableau.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"new_state = random_stabilizer(3)\nqubit_indices = [1,2,3]\nReset(new_state, qubit_indices)","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"It can be done anywhere in a circuit, not just at the beginning.","category":"page"},{"location":"noisycircuits/#Simulation-of-Noisy-Clifford-Circuits","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Clifford Circuits","text":"","category":"section"},{"location":"noisycircuits/","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Circuits","text":"DocTestSetup = quote\n using QuantumClifford\n using QuantumClifford.Experimental.NoisyCircuits\nend\nCurrentModule = QuantumClifford.Experimental.NoisyCircuits","category":"page"},{"location":"noisycircuits/","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Circuits","text":"warning: Unstable\nThis is unfinished experimental functionality that will change significantly.","category":"page"},{"location":"noisycircuits/","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Circuits","text":"We have experimental support for simulation of noisy Clifford circuits which can be imported with using QuantumClifford.Experimental.NoisyCircuits.","category":"page"},{"location":"noisycircuits/","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Circuits","text":"Both Monte Carlo and Perturbative Expansion approaches are supported. When performing a perturbative expansion in the noise parameter, the expansion can optionally be performed symbolically, to arbitrary high orders.","category":"page"},{"location":"noisycircuits/","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Circuits","text":"Multiple notebooks with examples are also available. For instance, see this tutorial on entanglement purification for many examples.","category":"page"},{"location":"ECC_evaluating/#ecc_evaluating","page":"Evaluating codes and decoders","title":"Evaluating an ECC code and decoders","text":"","category":"section"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"DocTestSetup = quote\n using QuantumClifford\n using QuantumClifford.ECC\nend\nCurrentModule = QuantumClifford.ECC","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"warning: The documentation is incomplete\nWhile waiting for a better documentation than the small example below, consider looking into evaluate_decoder, TableDecoder, BeliefPropDecoder, PyBeliefPropDecoder, PyMatchingDecoder, CommutationCheckECCSetup, NaiveSyndromeECCSetup, ShorSyndromeECCSetup","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"This is a quick and durty example on how to use some of the decoders.","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"A function to plot the results of ","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"using CairoMakie\n\nfunction make_decoder_figure(phys_errors, results, title=\"\")\n minlim = min(minimum(phys_errors),minimum(results[results.!=0]))\n maxlim = min(1, max(maximum(phys_errors),maximum(results[results.!=0])))\n\n fresults = copy(results)\n fresults[results.==0] .= NaN\n\n f = Figure()\n a = Axis(f[1,1],\n xscale=log10, yscale=log10,\n limits=(minlim,maxlim,minlim,maxlim),\n aspect=DataAspect(),\n xlabel=\"physical error rate\",\n ylabel=\"logical error rate\",\n title=title)\n lines!(a, [minlim,maxlim],[minlim,maxlim], color=:black)\n for (i,sresults) in enumerate(eachslice(fresults, dims=1))\n scatter!(a, phys_errors, sresults[:,1], marker=:+, color=Cycled(i))\n scatter!(a, phys_errors, sresults[:,2], marker=:x, color=Cycled(i))\n end\n f\nend","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"Testing out a lookup table decoder on a small code.","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"using QuantumClifford\nusing QuantumClifford.ECC\n\nmem_errors = 0.001:0.0005:0.01\ncodes = [Shor9()]\nresults = zeros(length(codes), length(mem_errors), 2)\n\nfor (ic, c) in pairs(codes)\n for (i,m) in pairs(mem_errors)\n setup = CommutationCheckECCSetup(m)\n decoder = TableDecoder(c)\n r = evaluate_decoder(decoder, setup, 10000)\n results[ic,i,:] .= r\n end\nend\n\nmake_decoder_figure(mem_errors, results, \"Shor's code with a lookup table decoder\")","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"Testing out the toric code with a decoder provided by the python package pymatching (provided in julia by the meta package PyQDecoders.jl).","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"import PyQDecoders\n\nmem_errors = 0.001:0.005:0.1\ncodes = [Toric(4,4), Toric(6,6)]\nresults = zeros(length(codes), length(mem_errors), 2)\n\nfor (ic, c) in pairs(codes)\n for (i,m) in pairs(mem_errors)\n setup = ShorSyndromeECCSetup(m, 0)\n decoder = PyMatchingDecoder(c)\n r = evaluate_decoder(decoder, setup, 1000)\n results[ic,i,:] .= r\n end\nend\n\nmake_decoder_figure(mem_errors, results, \"Toric code with a MWPM decoder\")","category":"page"},{"location":"canonicalization/#Canonicalization-operations","page":"Canonicalization","title":"Canonicalization operations","text":"","category":"section"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"Different types of canonicalization operations are implemented. All of them are types of Gaussian elimination.","category":"page"},{"location":"canonicalization/#[canonicalize!](@ref)","page":"Canonicalization","title":"canonicalize!","text":"","category":"section"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"First do elimination on all X components and only then perform elimination on the Z components. Based on (Garcia et al., 2012). It is used in logdot for inner products of stabilizer states.","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"The final tableaux, if square should look like the following (Image: )","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"If the tableaux is shorter than a square, the diagonals might not reach all the way to the right.","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"using QuantumClifford, CairoMakie\nf=Figure()\nstabilizerplot_axis(f[1,1], canonicalize!(random_stabilizer(20,30)))\nf","category":"page"},{"location":"canonicalization/#[canonicalize_rref!](@ref)","page":"Canonicalization","title":"canonicalize_rref!","text":"","category":"section"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"Cycle between elimination on X and Z for each qubit. Particularly useful for tracing out qubits. Based on (Audenaert and Plenio, 2005). For convenience reasons, the canonicalization starts from the bottom row, and you can specify as a second argument which columns to be canonicalized (useful for tracing out arbitrary qubits, e.g., in traceout!).","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"The tableau canonicalization is done in recursive steps, each one of which results in something akin to one of these three options (Image: )","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"using QuantumClifford, CairoMakie\nf=Figure()\nstabilizerplot_axis(f[1,1], canonicalize_rref!(random_stabilizer(20,30),1:30)[1])\nf","category":"page"},{"location":"canonicalization/#[canonicalize_gott!](@ref)","page":"Canonicalization","title":"canonicalize_gott!","text":"","category":"section"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"First do elimination on all X components and only then perform elimination on the Z components, but without touching the qubits that were eliminated during the X pass. Unlike other canonicalization operations, qubit columns are reordered, providing for a straight diagonal in each block. Particularly useful as certain blocks of the new created matrix are related to logical operations of the corresponding code, e.g. computing the logical X and Z operators of a MixedDestabilizer. Based on (Gottesman, 1997).","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"A canonicalized tableau would look like the following (the right-most block does not exist for square tableaux). (Image: )","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"using QuantumClifford, CairoMakie\nf=Figure()\nstabilizerplot_axis(f[1,1], canonicalize_gott!(random_stabilizer(30))[1])\nf","category":"page"},{"location":"canonicalization/#[canonicalize_clip!](@ref)","page":"Canonicalization","title":"canonicalize_clip!","text":"","category":"section"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"Convert to the \"clipped\" gauge of a stabilizer state resulting in a \"river\" of non-identity operators around the diagonal.","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"using QuantumClifford, CairoMakie\nf=Figure()\nstabilizerplot_axis(f[1,1], canonicalize_clip!(random_stabilizer(30)))\nf","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"The properties of the clipped gauge are:","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"Each qubit is the left/right \"endpoint\" of exactly two stabilizer rows.\nFor the same qubit the two endpoints are always different Pauli operators.","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"This canonicalization is used to derive the bigram a stabilizer state, which is also related to entanglement entropy in the state.","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"Introduced in (Nahum et al., 2017), with a more detailed explanation of the algorithm in Appendix A of (Li et al., 2019).","category":"page"},{"location":"noisycircuits_mc/#noisycircuits_mc","page":"Monte Carlo","title":"Monte Carlo simulations of noisy Clifford circuits","text":"","category":"section"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"DocTestSetup = quote\n using QuantumClifford\n using QuantumClifford.Experimental.NoisyCircuits\n using Quantikz\nend\nCurrentModule = QuantumClifford.Experimental.NoisyCircuits","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"warning: Unstable\nThis is experimental functionality with an unstable API.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"Import with using QuantumClifford.Experimental.NoisyCircuits.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"This module enables the simulation of noisy Clifford circuits through a Monte Carlo method where the same circuit is evaluated multiple times with random errors interspersed through it as prescribed by a given error model.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"Below is an example of a purification circuit. We first prepare the circuit we desire to use, including a noise model. Quantikz.jl was is used to visualize the circuit.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\ngood_bell_state = S\"XX\n ZZ\"\ninitial_state = MixedDestabilizer(good_bell_state⊗good_bell_state)\n\ng1 = sCNOT(1,3) # CNOT between qubit 1 and qubit 3 (both with Alice)\ng2 = sCNOT(2,4) # CNOT between qubit 2 and qubit 4 (both with Bob)\nm = BellMeasurement([sMX(3),sMX(4)]) # Bell measurement on qubit 3 and 4\nv = VerifyOp(good_bell_state,[1,2]) # Verify that qubit 1 and 2 indeed form a good Bell pair\nepsilon = 0.01 # The error rate\nn = NoiseOpAll(UnbiasedUncorrelatedNoise(epsilon))\n\n# This circuit performs a depolarization at rate `epsilon` to all qubits,\n# then bilater CNOT operations\n# then a Bell measurement\n# followed by checking whether the final result indeed corresponds to the correct Bell pair.\ncircuit = [n,g1,g2,m,v]","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"And we can run a Monte Carlo simulation of that circuit with mctrajectories.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"mctrajectories(initial_state, circuit, trajectories=500)","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"For more examples, see the notebook comparing the Monte Carlo and Perturbative method or this tutorial on entanglement purification for many examples.","category":"page"},{"location":"noisycircuits_mc/#Interface-for-custom-operations","page":"Monte Carlo","title":"Interface for custom operations","text":"","category":"section"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"If you want to create a custom gate type (e.g. calling it Operation), you need to definite the following methods.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"applywstatus!(s::T, g::Operation)::Tuple{T,Symbol} where T is a tableaux type like Stabilizer or a Register. The Symbol is the status of the operation. Predefined statuses are kept in the registered_statuses list, but you can add more. Be sure to expand this list if you want the trajectory simulators using your custom statuses to output all trajectories.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"There is also applynoise! which is convenient wait to create a noise model that can then be plugged into the NoisyGate struct, letting you reuse the predefined perfect gates and measurements. However, you can also just make up your own noise operator simply by implementing applywstatus! for it.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"You can also consult the list of implemented operators.","category":"page"},{"location":"commonstates/#Useful-States-and-Operators","page":"Useful States","title":"Useful States and Operators","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"DocTestSetup = quote\n using QuantumClifford\n using StableRNGs\n rng = StableRNG(42)\nend","category":"page"},{"location":"commonstates/#States","page":"Useful States","title":"States","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Stabilizer states can be represented with the Stabilizer, Destabilizer, MixedStabilizer, and MixedDestabilizer tableau data structures. You probably want to use MixedDestabilizer which supports the widest set of operations.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Moreover, a MixedDestabilizer can be stored inside a Register together with a set of classical bits in which measurement results can be written.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Below are convenience constructors for common types of states and operators, already implemented in this library.","category":"page"},{"location":"commonstates/#Pauli-Operators","page":"Useful States","title":"Pauli Operators","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Single qubit PauliOperator is implemented in [single_z] and [single_x].","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> single_z(4,2)\n+ _Z__\n\njulia> single_x(4,3)\n+ __X_","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"All identity operators use zero.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> zero(PauliOperator, 3)\n+ ___\n\njulia> zero(P\"XYZXYZ\")\n+ ______","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Random Pauli operators are implemented as well (with or without a random phase).","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> using StableRNGs; rng = StableRNG(42);\n\njulia> random_pauli(rng, 4)\n+ ZYY_\n\njulia> random_pauli(rng, 4; nophase=false)\n- YZ_X","category":"page"},{"location":"commonstates/#Stabilizer-States","page":"Useful States","title":"Stabilizer States","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"An all-identity stabilizer can be created with zero.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> zero(Stabilizer, 3)\n+ ___\n+ ___\n+ ___\n\njulia> zero(Stabilizer, 2, 3)\n+ ___\n+ ___\n\njulia> zero(S\"XIZ\n YZX\")\n+ ___\n+ ___","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Diagonal stabilizers in different bases are available as well, through one.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> one(Stabilizer, 3)\n+ Z__\n+ _Z_\n+ __Z\n\njulia> one(Stabilizer, 3; basis=:Y)\n+ Y__\n+ _Y_\n+ __Y\n\njulia> one(S\"XX\n ZZ\")\n+ Z_\n+ _Z","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"A random stabilizer (or destabilizers or Clifford operators) can be created as well. We use the algorithm described in (Bravyi and Maslov, 2021).","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> random_stabilizer(rng, 2,5)\n+ YZXZZ\n- XZYYY","category":"page"},{"location":"commonstates/#Mixed-States","page":"Useful States","title":"Mixed States","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Similarly, one can create a diagonal mixed state.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> one(MixedDestabilizer, 2, 3)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ X__\n+ _X_\n𝒳ₗ━━━\n+ __X\n𝒮𝓉𝒶𝒷━\n+ Z__\n+ _Z_\n𝒵ₗ━━━\n+ __Z","category":"page"},{"location":"commonstates/#Enumerating-all-Clifford-Operations","page":"Useful States","title":"Enumerating all Clifford Operations","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"The algorithm from (Koenig and Smolin, 2014) can be used to enumerate all Clifford operations on a given number of qubits through enumerate_cliffords. Or one can use random_clifford, random_stabilizer to directly sample from that set.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> length(enumerate_cliffords(1))\n6\n\njulia> length(enumerate_cliffords(2))\n720","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"To also enumerate possible phases, you can use enumerate_phases.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> length(collect(enumerate_phases(tCNOT)))\n16\n\njulia> length(collect(enumerate_phases(enumerate_cliffords(2))))\n11520","category":"page"},{"location":"commonstates/#Common-entangled-states","page":"Useful States","title":"Common entangled states","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Bell states and GHZ states have convenience constructors:","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> bell()\n+ XX\n+ ZZ\n\njulia> bell(2)\n+ XX__\n+ ZZ__\n+ __XX\n+ __ZZ\n\njulia> ghz(4)\n+ XXXX\n+ ZZ__\n+ _ZZ_\n+ __ZZ","category":"page"},{"location":"API/#Full-API","page":"API","title":"Full API","text":"","category":"section"},{"location":"API/","page":"API","title":"API","text":"","category":"page"},{"location":"API/#States","page":"API","title":"States","text":"","category":"section"},{"location":"API/","page":"API","title":"API","text":"Stabilizer states can be represented with the Stabilizer, Destabilizer, MixedStabilizer, and MixedDestabilizer tableau data structures. You probably want to use MixedDestabilizer which supports the widest set of operations.","category":"page"},{"location":"API/","page":"API","title":"API","text":"Moreover, a MixedDestabilizer can be stored inside a Register together with a set of classical bits in which measurement results can be written.","category":"page"},{"location":"API/","page":"API","title":"API","text":"Lastly, for Pauli frame simulations there is the PauliFrame type, a tableau in which each row represents a different Pauli frame.","category":"page"},{"location":"API/","page":"API","title":"API","text":"There are convenience constructors for common types of states and operators.","category":"page"},{"location":"API/#Operations","page":"API","title":"Operations","text":"","category":"section"},{"location":"API/","page":"API","title":"API","text":"Acting on quantum states can be performed either:","category":"page"},{"location":"API/","page":"API","title":"API","text":"In a \"linear algebra\" language where unitaries, measurements, and other operations have separate interfaces. This is an explicitly deterministic lower-level interface, which provides a great deal of control over how tableaux are manipulated. See the Stabilizer Tableau Algebra Manual as a primer on these approaches.\nOr in a \"circuit\" language, where the operators (and measurements and noise) are represented as circuit gates. This is a higher-level interface in which the outcome of an operation can be stochastic. The API for it is centered around the apply! function. Particularly useful for Monte Carlo simulations and Perturbative Expansion Symbolic Results.","category":"page"},{"location":"API/","page":"API","title":"API","text":"See the full list of operations for a list of implemented operations.","category":"page"},{"location":"API/#Autogenerated-API-list","page":"API","title":"Autogenerated API list","text":"","category":"section"},{"location":"API/","page":"API","title":"API","text":"Modules = [QuantumClifford]\nPrivate = false","category":"page"},{"location":"API/#QuantumClifford.QuantumClifford","page":"API","title":"QuantumClifford.QuantumClifford","text":"A module for using the Stabilizer formalism and simulating Clifford circuits.\n\n\n\n\n\n","category":"module"},{"location":"API/#QuantumClifford.continue_stat","page":"API","title":"QuantumClifford.continue_stat","text":"Returned by applywstatus! if the circuit simulation should continue.\n\n\n\n\n\n","category":"constant"},{"location":"API/#QuantumClifford.failure_stat","page":"API","title":"QuantumClifford.failure_stat","text":"Returned by applywstatus! if the circuit reports a failure.\n\nSee also: VerifyOp, BellMeasurement.\n\n\n\n\n\n","category":"constant"},{"location":"API/#QuantumClifford.false_success_stat","page":"API","title":"QuantumClifford.false_success_stat","text":"Returned by applywstatus! if the circuit reports a success, but it is a false positive (i.e., there was an undetected error).\n\nSee also: VerifyOp, BellMeasurement.\n\n\n\n\n\n","category":"constant"},{"location":"API/#QuantumClifford.true_success_stat","page":"API","title":"QuantumClifford.true_success_stat","text":"Returned by applywstatus! if the circuit reports a success and there is no undetected error.\n\nSee also: VerifyOp, BellMeasurement.\n\n\n\n\n\n","category":"constant"},{"location":"API/#QuantumClifford.AbstractSingleQubitOperator","page":"API","title":"QuantumClifford.AbstractSingleQubitOperator","text":"Supertype of all single-qubit symbolic operators.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.AbstractSymbolicOperator","page":"API","title":"QuantumClifford.AbstractSymbolicOperator","text":"Supertype of all symbolic operators. Subtype of AbstractCliffordOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.AbstractTwoQubitOperator","page":"API","title":"QuantumClifford.AbstractTwoQubitOperator","text":"Supertype of all two-qubit symbolic operators.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.BellMeasurement","page":"API","title":"QuantumClifford.BellMeasurement","text":"A Bell measurement performing the correlation measurement corresponding to the given pauli projections on the qubits at the selected indices.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.CircuitStatus","page":"API","title":"QuantumClifford.CircuitStatus","text":"A convenience struct to represent the status of a circuit simulated by mctrajectories\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.ClassicalXOR","page":"API","title":"QuantumClifford.ClassicalXOR","text":"Applies an XOR gate to classical bits. Currently only implemented for functionality with pauli frames.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.CliffordOperator","page":"API","title":"QuantumClifford.CliffordOperator","text":"Clifford Operator specified by the mapping of the basis generators.\n\njulia> tCNOT\nX₁ ⟼ + XX\nX₂ ⟼ + _X\nZ₁ ⟼ + Z_\nZ₂ ⟼ + ZZ\n\njulia> phase_gate = C\"Y\n Z\"\nX₁ ⟼ + Y\nZ₁ ⟼ + Z\n\njulia> stab = S\"XI\n IZ\";\n\n\njulia> entangled = tCNOT*stab\n+ XX\n+ ZZ\n\njulia> CliffordOperator(T\"YY\")\nERROR: DimensionMismatch: Input tableau should be of size 2n×n (top half is the X mappings and the bottom half are the Z mappings).\n[...]\n\nDestabilizer can also be converted.\n\njulia> d = Destabilizer(S\"Y\")\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z\n𝒮𝓉𝒶𝒷\n+ Y\n\njulia> CliffordOperator(d)\nX₁ ⟼ + Z\nZ₁ ⟼ + Y\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Destabilizer","page":"API","title":"QuantumClifford.Destabilizer","text":"A tableau representation of a pure stabilizer state. The tableau tracks the destabilizers as well, for efficient projections. On initialization there are no checks that the provided state is indeed pure. This enables the use of this data structure for mixed stabilizer state, but a better choice would be to use MixedDestabilizer.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.MixedDestabilizer","page":"API","title":"QuantumClifford.MixedDestabilizer","text":"A tableau representation for mixed stabilizer states that keeps track of the destabilizers in order to provide efficient projection operations.\n\nThe rank r of the n-qubit tableau is tracked, either so that it can be used to represent a mixed stabilizer state, or so that it can be used to represent an n-r logical-qubit code over n physical qubits. The \"logical\" operators are tracked as well.\n\nWhen the constructor is called on an incomplete Stabilizer it automatically calculates the destabilizers and logical operators, following chapter 4 of (Gottesman, 1997). Under the hood the conversion uses the canonicalize_gott! canonicalization. That canonicalization permutes the columns of the tableau, but we automatically undo the column permutation in the preparation of a MixedDestabilizer so that qubits are not reindexed. The boolean keyword arguments undoperm and reportperm can be used to control this behavior and to report the permutations explicitly.\n\nSee also: stabilizerview, destabilizerview, logicalxview, logicalzview\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.MixedStabilizer","page":"API","title":"QuantumClifford.MixedStabilizer","text":"A slight improvement of the Stabilizer data structure that enables more naturally and completely the treatment of mixed states, in particular when the project! function is used.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.NoiseOp","page":"API","title":"QuantumClifford.NoiseOp","text":"An operator that applies the given noise model to the qubits at the selected indices.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.NoiseOpAll","page":"API","title":"QuantumClifford.NoiseOpAll","text":"An operator that applies the given noise model to all qubits.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.NoisyGate","page":"API","title":"QuantumClifford.NoisyGate","text":"A gate consisting of the given noise applied after the given perfect Clifford gate.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.PauliChannel","page":"API","title":"QuantumClifford.PauliChannel","text":"A Pauli channel datastructure, mainly for use with StabMixture\n\nSee also: UnitaryPauliChannel\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.PauliFrame","page":"API","title":"QuantumClifford.PauliFrame","text":"struct PauliFrame{T, S} <: QuantumClifford.AbstractQCState\n\nThis is a wrapper around a tableau. This \"frame\" tableau is not to be viewed as a normal stabilizer tableau, although it does conjugate the same under Clifford operations. Each row in the tableau refers to a single frame. The row represents the Pauli operation by which the frame and the reference differ.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.PauliFrame-Tuple{Any, Any, Any}","page":"API","title":"QuantumClifford.PauliFrame","text":"PauliFrame(\n frames,\n qubits,\n measurements\n) -> PauliFrame{Stabilizer{QuantumClifford.Tableau{Vector{UInt8}, LinearAlgebra.Adjoint{UInt64, Matrix{UInt64}}}}}\n\n\nPrepare an empty set of Pauli frames with the given number of frames and qubits. Preallocates spaces for measurement number of measurements.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.PauliMeasurement","page":"API","title":"QuantumClifford.PauliMeasurement","text":"A Stabilizer measurement on the entirety of the quantum register.\n\nprojectrand!(state, pauli) and apply!(state, PauliMeasurement(pauli)) give the same (possibly non-deterministic) result. Particularly useful when acting on Register.\n\nSee also: apply!, projectrand!.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.PauliOperator","page":"API","title":"QuantumClifford.PauliOperator","text":"A multi-qubit Pauli operator (1iIZXY^otimes n).\n\nA Pauli can be constructed with the P custom string macro or by building up one through products and tensor products of smaller operators.\n\njulia> pauli3 = P\"-iXYZ\"\n-iXYZ\n\njulia> pauli4 = 1im * pauli3 ⊗ X\n+ XYZX\n\njulia> Z*X\n+iY\n\nWe use a typical F(2,2) encoding internally. The X and Z bits are stored in a single concatenated padded array of UInt chunks of a bit array.\n\njulia> p = P\"-IZXY\";\n\n\njulia> p.xz\n2-element Vector{UInt64}:\n 0x000000000000000c\n 0x000000000000000a\n\nYou can access the X and Z bits through getters and setters or through the xview, zview, xbit, and zbit functions.\n\njulia> p = P\"XYZ\"; p[1]\n(true, false)\n\njulia> p[1] = (true, true); p\n+ YYZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Register","page":"API","title":"QuantumClifford.Register","text":"A register, representing the state of a computer including both a tableaux and an array of classical bits (e.g. for storing measurement results)\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Reset","page":"API","title":"QuantumClifford.Reset","text":"Reset the specified qubits to the given state.\n\nBe careful, this operation implies first tracing out the qubits, which can lead to mixed states if these qubits were entangled with the rest of the system.\n\nSee also: sMRZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.SingleQubitOperator","page":"API","title":"QuantumClifford.SingleQubitOperator","text":"A \"symbolic\" general single-qubit operator which permits faster multiplication than an operator expressed as an explicit tableau.\n\njulia> op = SingleQubitOperator(2, true, true, true, false, true, true) # Tableau components and phases\nSingleQubitOperator on qubit 2\nX₁ ⟼ - Y\nZ₁ ⟼ - X\n\njulia> typeof(op)\nSingleQubitOperator\n\njulia> t_op = CliffordOperator(op, 3) # Transforming it back into an explicit tableau representation (specifying the size)\nX₁ ⟼ + X__\nX₂ ⟼ - _Y_\nX₃ ⟼ + __X\nZ₁ ⟼ + Z__\nZ₂ ⟼ - _X_\nZ₃ ⟼ + __Z\n\njulia> typeof(t_op)\nCliffordOperator{QuantumClifford.Tableau{Vector{UInt8}, Matrix{UInt64}}}\n\njulia> CliffordOperator(op, 1, compact=true) # You can also extract just the non-trivial part of the tableau\nX₁ ⟼ - Y\nZ₁ ⟼ - X\n\nSee also: sHadamard, sPhase, sId1, sX, sY, sZ, CliffordOperator\n\nOr simply consult subtypes(QuantumClifford.AbstractSingleQubitOperator) and subtypes(QuantumClifford.AbstractTwoQubitOperator) for a full list. You can think of the s prefix as \"symbolic\" or \"sparse\".\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.SparseGate","page":"API","title":"QuantumClifford.SparseGate","text":"A Clifford gate, applying the given cliff operator to the qubits at the selected indices.\n\napply!(state, cliff, indices) and apply!(state, SparseGate(cliff, indices)) give the same result.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.StabMixture","page":"API","title":"QuantumClifford.StabMixture","text":"mutable struct StabMixture{T, F}\n\nRepresents mixture ∑ ϕᵢⱼ Pᵢ ρ Pⱼ† where ρ is a pure stabilizer state.\n\njulia> StabMixture(S\"-X\")\nA mixture ∑ ϕᵢⱼ Pᵢ ρ Pⱼ† where ρ is\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z\n𝒮𝓉𝒶𝒷\n- X\nwith ϕᵢⱼ | Pᵢ | Pⱼ:\n 1.0+0.0im | + _ | + _\n\njulia> pcT\nA unitary Pauli channel P = ∑ ϕᵢ Pᵢ with the following branches:\nwith ϕᵢ | Pᵢ\n 0.853553+0.353553im | + _\n 0.146447-0.353553im | + Z\n\njulia> apply!(StabMixture(S\"-X\"), pcT)\nA mixture ∑ ϕᵢⱼ Pᵢ ρ Pⱼ† where ρ is\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z\n𝒮𝓉𝒶𝒷\n- X\nwith ϕᵢⱼ | Pᵢ | Pⱼ:\n 0.0+0.353553im | + _ | + Z\n 0.0-0.353553im | + Z | + _\n 0.853553+0.0im | + _ | + _\n 0.146447+0.0im | + Z | + Z\n\nSee also: PauliChannel\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Stabilizer","page":"API","title":"QuantumClifford.Stabilizer","text":"Stabilizer, i.e. a list of commuting multi-qubit Hermitian Pauli operators.\n\nInstances can be created with the S custom string macro or as direct sum of other stabilizers.\n\ntip: Stabilizers and Destabilizers\nIn many cases you probably would prefer to use the MixedDestabilizer data structure, as it caries a lot of useful additional information, like tracking rank and destabilizer operators. Stabilizer has mostly a pedagogical value, and it is also used for slightly faster simulation of a particular subset of Clifford operations.\n\njulia> s = S\"XXX\n ZZI\n IZZ\"\n+ XXX\n+ ZZ_\n+ _ZZ\n\njulia> s⊗s\n+ XXX___\n+ ZZ____\n+ _ZZ___\n+ ___XXX\n+ ___ZZ_\n+ ____ZZ\n\nIt has an indexing API, looking like a list of PauliOperators.\n\njulia> s[2]\n+ ZZ_\n\nPauli operators can act directly on the a stabilizer.\n\njulia> P\"YYY\" * s\n- XXX\n+ ZZ_\n+ _ZZ\n\nThere are a number of ways to create a Stabilizer, including:\n\ngenerate Stabilizers from a list of Pauli operators\n\njulia> Stabilizer([P\"XX\", P\"ZZ\"])\n+ XX\n+ ZZ\n\ngenerate Stabilizers from boolean matrices\n\njulia> a = [true true; false false]; b = [false true; true true];\n\njulia> Stabilizer(a, b)\n+ XY\n+ ZZ\n\njulia> Stabilizer([0x0, 0x2], a, b)\n+ XY\n- ZZ\n\ninitialize an empty Stabilizer and fill it through indexing\n\njulia> s = zero(Stabilizer, 2)\n+ __\n+ __\n\njulia> s[1,1] = (true, false); s\n+ X_\n+ __\n\nThere are no automatic checks for correctness (i.e. independence of all rows, commutativity of all rows, hermiticity of all rows). The rank (number of rows) is permitted to be less than the number of qubits (number of columns): canonilization, projection, etc. continue working in that case. To great extent this library uses the Stabilizer data structure simply as a tableau. This might be properly abstracted away in future versions.\n\nSee also: PauliOperator, canonicalize!\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Stabilizer-Tuple{Graphs.SimpleGraphs.SimpleGraph}","page":"API","title":"QuantumClifford.Stabilizer","text":"Convert a graph representing a stabilizer state to an explicit Stabilizer.\n\nSee also: graphstate\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.UnbiasedUncorrelatedNoise","page":"API","title":"QuantumClifford.UnbiasedUncorrelatedNoise","text":"Depolarization noise model with total probability of error p.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.UnitaryPauliChannel","page":"API","title":"QuantumClifford.UnitaryPauliChannel","text":"A Pauli channel datastructure, mainly for use with StabMixture.\n\nMore convenient to use than PauliChannel when you know your Pauli channel is unitary.\n\njulia> Tgate = UnitaryPauliChannel(\n (I, Z),\n ((1+exp(im*π/4))/2, (1-exp(im*π/4))/2)\n )\nA unitary Pauli channel P = ∑ ϕᵢ Pᵢ with the following branches:\nwith ϕᵢ | Pᵢ\n 0.853553+0.353553im | + _\n 0.146447-0.353553im | + Z\n\njulia> PauliChannel(Tgate)\nPauli channel ρ ↦ ∑ ϕᵢⱼ Pᵢ ρ Pⱼ† with the following branches:\nwith ϕᵢⱼ | Pᵢ | Pⱼ:\n 0.853553+0.0im | + _ | + _\n 0.0+0.353553im | + _ | + Z\n 0.0-0.353553im | + Z | + _\n 0.146447+0.0im | + Z | + Z\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.VerifyOp","page":"API","title":"QuantumClifford.VerifyOp","text":"A \"probe\" to verify that the state of the qubits corresponds to a desired good_state, e.g. at the end of the execution of a circuit.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sCNOT","page":"API","title":"QuantumClifford.sCNOT","text":"A \"symbolic\" CNOT. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sCPHASE","page":"API","title":"QuantumClifford.sCPHASE","text":"A \"symbolic\" CPHASE. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sHadamard","page":"API","title":"QuantumClifford.sHadamard","text":"A \"symbolic\" single-qubit Hadamard. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sId1","page":"API","title":"QuantumClifford.sId1","text":"A \"symbolic\" single-qubit Identity operation.\n\nSee also: SingleQubitOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sInvPhase","page":"API","title":"QuantumClifford.sInvPhase","text":"A \"symbolic\" single-qubit InvPhase. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMRX","page":"API","title":"QuantumClifford.sMRX","text":"Measure a qubit in the X basis and reset to the |+⟩ state.\n\nSee also: sMRZ, Reset, sMZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMRY","page":"API","title":"QuantumClifford.sMRY","text":"Measure a qubit in the Y basis and reset to the |i₊⟩ state.\n\nSee also: sMRZ, Reset, sMZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMRZ","page":"API","title":"QuantumClifford.sMRZ","text":"Measure a qubit in the Z basis and reset to the |0⟩ state.\n\nwarning: It does not trace out the qubit!\nAs described below there is a difference between measuring the qubit (followed by setting it to a given known state) and \"tracing out\" the qubit. By reset here we mean \"measuring and setting to a known state\", not \"tracing out\".\n\njulia> s = MixedDestabilizer(S\"XXX ZZI IZZ\") # |000⟩+|111⟩\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ _X_\n+ __X\n𝒮𝓉𝒶𝒷━\n+ XXX\n+ ZZ_\n+ Z_Z\n\njulia> traceout!(copy(s), 1) # = I⊗(|00⟩⟨00| + |11⟩⟨11|)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ _X_\n𝒳ₗ━━━\n+ _XX\n+ Z__\n𝒮𝓉𝒶𝒷━\n+ _ZZ\n𝒵ₗ━━━\n+ Z_Z\n+ XXX\n\njulia> projectZ!(traceout!(copy(s), 1), 1)[1] # = |000⟩⟨000|+|011⟩⟨011| or |100⟩⟨100|+|111⟩⟨111| (use projectZrand! to actually get a random result)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ _X_\n+ XXX\n𝒳ₗ━━━\n+ _XX\n𝒮𝓉𝒶𝒷━\n+ _ZZ\n+ Z__\n𝒵ₗ━━━\n+ Z_Z\n\njulia> projectZ!(copy(s), 1)[1] # = |000⟩ or |111⟩ (use projectZrand! to actually get a random result)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ XXX\n+ _X_\n+ __X\n𝒮𝓉𝒶𝒷━\n+ Z__\n+ ZZ_\n+ Z_Z\n\njulia> apply!(Register(copy(s)), sMRZ(1)) |> quantumstate # |000⟩ or |011⟩, depending on randomization\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ XXX\n+ _X_\n+ __X\n𝒮𝓉𝒶𝒷━\n+ Z__\n- ZZ_\n- Z_Z\n\nSee also: Reset, sMZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMX","page":"API","title":"QuantumClifford.sMX","text":"Symbolic single qubit X measurement. See also Register, projectXrand!, sMY, sMZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMY","page":"API","title":"QuantumClifford.sMY","text":"Symbolic single qubit Y measurement. See also Register, projectYrand!, sMX, sMZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMZ","page":"API","title":"QuantumClifford.sMZ","text":"Symbolic single qubit Z measurement. See also Register, projectZrand!, sMX, sMY\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sPhase","page":"API","title":"QuantumClifford.sPhase","text":"A \"symbolic\" single-qubit Phase. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sSWAP","page":"API","title":"QuantumClifford.sSWAP","text":"A \"symbolic\" SWAP. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sX","page":"API","title":"QuantumClifford.sX","text":"A \"symbolic\" single-qubit X. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sXCX","page":"API","title":"QuantumClifford.sXCX","text":"A \"symbolic\" XCX. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sXCY","page":"API","title":"QuantumClifford.sXCY","text":"A \"symbolic\" XCY. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sXCZ","page":"API","title":"QuantumClifford.sXCZ","text":"A \"symbolic\" XCZ. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sY","page":"API","title":"QuantumClifford.sY","text":"A \"symbolic\" single-qubit Y. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sYCX","page":"API","title":"QuantumClifford.sYCX","text":"A \"symbolic\" YCX. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sYCY","page":"API","title":"QuantumClifford.sYCY","text":"A \"symbolic\" YCY. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sYCZ","page":"API","title":"QuantumClifford.sYCZ","text":"A \"symbolic\" YCZ. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sZ","page":"API","title":"QuantumClifford.sZ","text":"A \"symbolic\" single-qubit Z. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sZCX","page":"API","title":"QuantumClifford.sZCX","text":"A \"symbolic\" ZCX. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sZCY","page":"API","title":"QuantumClifford.sZCY","text":"A \"symbolic\" ZCY. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sZCZ","page":"API","title":"QuantumClifford.sZCZ","text":"A \"symbolic\" ZCZ. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sZCrY","page":"API","title":"QuantumClifford.sZCrY","text":"A \"symbolic\" ZCrY. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.PauliError","page":"API","title":"QuantumClifford.PauliError","text":"A convenient constructor for various types of Pauli errors, that can be used as circuit gates in simulations. Returns more specific types when necessary.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.PauliError-Tuple{Any, Any}","page":"API","title":"QuantumClifford.PauliError","text":"\"Construct a gate operation that applies an unbiased Pauli error on all qubits, each with independent probability p.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.PauliError-Tuple{Int64, Any}","page":"API","title":"QuantumClifford.PauliError","text":"\"Construct a gate operation that applies an unbiased Pauli error on qubit q with probability p.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.PauliNoise","page":"API","title":"QuantumClifford.PauliNoise","text":"A convenient constructor for various types of Pauli noise models. Returns more specific types when necessary.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.PauliNoise-Tuple{Any}","page":"API","title":"QuantumClifford.PauliNoise","text":"Constructs an unbiased Pauli noise model with total probability of error p.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.affectedqubits","page":"API","title":"QuantumClifford.affectedqubits","text":"A method giving the qubits acted upon by a given operation. Part of the Noise interface.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.applybranches","page":"API","title":"QuantumClifford.applybranches","text":"Compute all possible new states after the application of the given operator. Reports the probability of each one of them. Deterministic (as it reports all branches of potentially random processes), part of the Perturbative Expansion interface.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.applynoise!","page":"API","title":"QuantumClifford.applynoise!","text":"A method modifying a given state by applying the corresponding noise model. It is non-deterministic, part of the Noise interface.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.applywstatus!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.applywstatus!","text":"Used for mctrajectories.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.bell","page":"API","title":"QuantumClifford.bell","text":"Prepare one or more Bell pairs (with optional phases).\n\njulia> bell()\n+ XX\n+ ZZ\n\njulia> bell(2)\n+ XX__\n+ ZZ__\n+ __XX\n+ __ZZ\n\njulia> bell((true, false))\n- XX\n+ ZZ\n\njulia> bell([true, false, true, true])\n- XX__\n+ ZZ__\n- __XX\n- __ZZ\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.bigram-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.bigram","text":"bigram(\n state::QuantumClifford.AbstractStabilizer;\n clip\n) -> Matrix{Int64}\n\n\nGet the bigram of a tableau.\n\nIt is the list of endpoints of a tableau in the clipped gauge.\n\nIf clip=true (the default) the tableau is converted to the clipped gauge in-place before calculating the bigram. Otherwise, the clip gauge conversion is skipped (for cases where the input is already known to be in the correct gauge).\n\nIntroduced in (Nahum et al., 2017), with a more detailed explanation of the algorithm in (Li et al., 2019) and (Gullans et al., 2020).\n\nSee also: canonicalize_clip!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.bitview","page":"API","title":"QuantumClifford.bitview","text":"A view of the classical bits stored with the state\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.canonicalize!-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.canonicalize!","text":"canonicalize!(\n state::QuantumClifford.AbstractStabilizer;\n phases,\n ranks\n) -> Any\n\n\nCanonicalize a stabilizer (in place).\n\nAssumes the input is a valid stabilizer (all operators commute and have real phases). It permits redundant generators and identity generators.\n\njulia> ghz = S\"XXXX\n ZZII\n IZZI\n IIZZ\";\n\n\njulia> canonicalize!(ghz)\n+ XXXX\n+ Z__Z\n+ _Z_Z\n+ __ZZ\n\njulia> canonicalize!(S\"XXXX\n IZZI\n IIZZ\")\n+ XXXX\n+ _Z_Z\n+ __ZZ\n\nNot all rows in the tableau in the next example are independent:\n\njulia> canonicalize!(S\"XXXX\n ZZII\n IZZI\n IZIZ\n IIZZ\")\n+ XXXX\n+ Z__Z\n+ _Z_Z\n+ __ZZ\n+ ____\n\nIn cases of lower rank, more advanced tableau structures might be better. For instance the MixedStabilizer or MixedDestabilizer structures (you can read more about them in the Data Structures section of the documentation).\n\nIf phases=false is set, the canonicalization does not track the phases in the tableau, leading to significant (constant factor) speedup.\n\njulia> s = S\"-ZX\n XZ\"\n- ZX\n+ XZ\n\njulia> canonicalize!(copy(s), phases=false)\n- XZ\n+ ZX\n\njulia> canonicalize!(copy(s))\n+ XZ\n- ZX\n\nIf ranks=true is set, the last pivot indices for the X and Z stage of the canonicalization are returned as well.\n\njulia> s = S\"XXXX\n ZZII\n IZIZ\n ZIIZ\";\n\n\njulia> _, ix, iz = canonicalize!(s, ranks=true); ix, iz\n(1, 3)\n\njulia> s\n+ XXXX\n+ Z__Z\n+ _Z_Z\n+ ____\n\nBased on (Garcia et al., 2012).\n\nSee also: canonicalize_rref!, canonicalize_gott!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.canonicalize_clip!-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.canonicalize_clip!","text":"canonicalize_clip!(\n state::QuantumClifford.AbstractStabilizer;\n phases\n) -> QuantumClifford.AbstractStabilizer\n\n\nFix the clipped gauge of a stabilizer (in place).\n\nAssumes the input is a valid full-rank stabilizer (all operators commute and have real phases).\n\njulia> s = S\"- X_ZX_X\n + XXYZ__\n - YZ_Z_X\n - XZX__Y\n + _Z_Y_Y\n - ____Z_\";\n\n\njulia> canonicalize_clip!(s)\n- X_XY__\n+ YZY___\n+ _XZX__\n- _ZYX_Z\n- __YZ_X\n- ____Z_\n\nIf phases=false is set, the canonicalization does not track the phases in the tableau, leading to a significant speedup.\n\nIntroduced in (Nahum et al., 2017), with a more detailed explanation of the algorithm in Appendix A of (Li et al., 2019)\n\nSee also: canonicalize!, canonicalize_rref!, canonicalize_gott!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.canonicalize_gott!-Tuple{Stabilizer}","page":"API","title":"QuantumClifford.canonicalize_gott!","text":"Inplace Gottesman canonicalization of a tableau.\n\nThis uses different canonical form from canonicalize!. It is used in the computation of the logical X and Z operators of a MixedDestabilizer.\n\nIt returns the (in place) modified state, the indices of the last pivot of both Gaussian elimination steps, and the permutations that have been used to put the X and Z tableaux in standard form.\n\nBased on (Gottesman, 1997).\n\nSee also: canonicalize!, canonicalize_rref!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.canonicalize_rref!-Tuple{QuantumClifford.AbstractStabilizer, Any}","page":"API","title":"QuantumClifford.canonicalize_rref!","text":"canonicalize_rref!(\n state::QuantumClifford.AbstractStabilizer,\n colindices;\n phases\n) -> Tuple{QuantumClifford.AbstractStabilizer, Any}\n\n\nCanonicalize a stabilizer (in place) along only some columns.\n\nThis uses different canonical form from canonicalize!. It also indexes in reverse in order to make its use in traceout! more efficient. Its use in traceout! is its main application.\n\nIt returns the (in place) modified state and the index of the last pivot.\n\nBased on (Audenaert and Plenio, 2005).\n\nSee also: canonicalize!, canonicalize_gott!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.canonicalize_rref!-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.canonicalize_rref!","text":"canonicalize_rref!(\n state::QuantumClifford.AbstractStabilizer;\n phases\n) -> Tuple{QuantumClifford.AbstractStabilizer, Any}\n\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.clifford_cardinality-Tuple{Int64}","page":"API","title":"QuantumClifford.clifford_cardinality","text":"The size of the Clifford group over a given number of qubits, possibly modulo the phases.\n\nFor n qubits, not accounting for phases is 2ⁿⁿΠⱼ₌₁ⁿ(4ʲ-1). There are 4ⁿ different phase configurations.\n\nSee also: enumerate_cliffords.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.comm","page":"API","title":"QuantumClifford.comm","text":"Check whether two operators commute.\n\n0x0 if they commute, 0x1 if they anticommute.\n\njulia> P\"XX\"*P\"ZZ\", P\"ZZ\"*P\"XX\"\n(- YY, - YY)\n\njulia> comm(P\"ZZ\", P\"XX\")\n0x00\n\njulia> comm(P\"IZ\", P\"XX\")\n0x01\n\nSee also: comm!\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.comm!","page":"API","title":"QuantumClifford.comm!","text":"An in-place version of comm, storing its output in the given buffer.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.compactify_circuit-Tuple{Any}","page":"API","title":"QuantumClifford.compactify_circuit","text":"Convert a list of gates to a more optimized \"sum type\" format which permits faster dispatch.\n\nGenerally, this should be called on a circuit before it is used in a simulation.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.destabilizerview-Tuple{Destabilizer}","page":"API","title":"QuantumClifford.destabilizerview","text":"A view of the subtableau corresponding to the destabilizer. See also tab, stabilizerview, logicalxview, logicalzview\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_cliffords-Tuple{Any, Any}","page":"API","title":"QuantumClifford.enumerate_cliffords","text":"Give the i-th n-qubit Clifford operation, where i∈{1..2ⁿⁿΠⱼ₌₁ⁿ(4ʲ-1)}\n\nThe algorithm is detailed in (Koenig and Smolin, 2014).\n\nSee also: symplecticGS, clifford_cardinality.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_cliffords-Tuple{Any}","page":"API","title":"QuantumClifford.enumerate_cliffords","text":"Give all n-qubit Clifford operations.\n\nThe algorithm is detailed in (Koenig and Smolin, 2014).\n\nSee also: symplecticGS, clifford_cardinality.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_phases-Tuple{CliffordOperator}","page":"API","title":"QuantumClifford.enumerate_phases","text":"Given an operator, return all operators that have the same tableau but different phases.\n\njulia> length(collect(enumerate_phases(tCNOT)))\n16\n\nSee also: enumerate_cliffords, clifford_cardinality.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_phases-Tuple{Union{Base.Generator, AbstractVector}}","page":"API","title":"QuantumClifford.enumerate_phases","text":"Given a set of operators, return all operators that have the same tableaux but different phases.\n\njulia> length(collect(enumerate_phases(enumerate_cliffords(2))))\n11520\n\nSee also: enumerate_cliffords, clifford_cardinality.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_single_qubit_gates-Tuple{Any}","page":"API","title":"QuantumClifford.enumerate_single_qubit_gates","text":"Generate a symbolic single-qubit gate given its index. Optionally, set non-trivial phases.\n\nSee also: enumerate_cliffords.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.fastcolumn","page":"API","title":"QuantumClifford.fastcolumn","text":"Convert a tableau to a memory layout that is fast for column operations.\n\nIn this layout a column of the tableau is stored (mostly) contiguously in memory. Due to bitpacking, e.g., packing 64 bits into a single UInt64, the memory layout is not perfectly contiguous, but it is still optimal given that some bitwrangling is required to extract a given bit.\n\nSee also: fastrow\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.fastrow","page":"API","title":"QuantumClifford.fastrow","text":"Convert a tableau to a memory layout that is fast for row operations.\n\nIn this layout a Pauli string (a row of the tableau) is stored contiguously in memory.\n\nSee also: fastrow\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.generate!-Tuple{PauliOperator, Stabilizer}","page":"API","title":"QuantumClifford.generate!","text":"Generate a Pauli operator by using operators from a given the Stabilizer.\n\nIt assumes the stabilizer is already canonicalized. It modifies the Pauli operator in place, generating it in reverse, up to a phase. That phase is left in the modified operator, which should be the identity up to a phase. Returns the new operator and the list of indices denoting the elements of stabilizer that were used for the generation.\n\njulia> ghz = S\"XXXX\n ZZII\n IZZI\n IIZZ\";\n\n\njulia> canonicalize!(ghz)\n+ XXXX\n+ Z__Z\n+ _Z_Z\n+ __ZZ\n\njulia> generate!(P\"-ZIZI\", ghz)\n(- ____, [2, 4])\n\nWhen the Pauli operator can not be generated by the given tableau, nothing is returned.\n\njulia> generate!(P\"XII\",canonicalize!(S\"ZII\")) === nothing\ntrue\n\njulia> generate!(P\"XII\",canonicalize!(S\"XII\")) === nothing\nfalse\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.gf2_H_to_G-Tuple{Any}","page":"API","title":"QuantumClifford.gf2_H_to_G","text":"For a given F(2,2) parity check matrix, return the generator matrix.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.gf2_gausselim!-Tuple{Any}","page":"API","title":"QuantumClifford.gf2_gausselim!","text":"Gaussian elimination over the binary field.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.gf2_invert-Tuple{Any}","page":"API","title":"QuantumClifford.gf2_invert","text":"Invert a binary matrix.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.gf2_isinvertible-Tuple{Any}","page":"API","title":"QuantumClifford.gf2_isinvertible","text":"Check whether a binary matrix is invertible.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.ghz","page":"API","title":"QuantumClifford.ghz","text":"Prepare a GHZ state of n qubits.\n\njulia> ghz()\n+ XXX\n+ ZZ_\n+ _ZZ\n\njulia> ghz(2)\n+ XX\n+ ZZ\n\njulia> ghz(4)\n+ XXXX\n+ ZZ__\n+ _ZZ_\n+ __ZZ\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.graph_gate-NTuple{4, Any}","page":"API","title":"QuantumClifford.graph_gate","text":"A helper function converting the gate indices from graphstate into a Clifford operator.\n\njulia> s = S\" XXX\n YZ_\n -_ZZ\";\n\n\njulia> graph, h_idx, ip_idx, z_idx = graphstate(s);\n\n\njulia> gate = graph_gate(h_idx, ip_idx, z_idx, nqubits(s));\n\n\njulia> apply!(s, gate) # This is now a graph state (notice you need to multiply row 1 by row 2)\n+ YYZ\n+ XZ_\n+ _ZX\n\njulia> canonicalize!(s) == canonicalize!(Stabilizer(graph))\ntrue\n\nSee also: graph_gatesequence\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.graph_gatesequence-Tuple{Vector{Int64}, Vector{Int64}, Vector{Int64}}","page":"API","title":"QuantumClifford.graph_gatesequence","text":"A helper function converting the gate indices from graphstate into a sequence of gates.\n\njulia> s = S\" XXX\n YZ_\n -_ZZ\";\n\n\njulia> graph, h_idx, ip_idx, z_idx = graphstate(s);\n\n\njulia> gates = graph_gatesequence(h_idx, ip_idx, z_idx);\n\n\njulia> for gate in vcat(gates...) apply!(s, gate) end\n\n\njulia> s # This is now a graph state (notice you need to multiply row 1 by row 2)\n+ YYZ\n+ XZ_\n+ _ZX\n\njulia> canonicalize!(s) == canonicalize!(Stabilizer(graph))\ntrue\n\nSee also: graph_gatesequence\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.graphstate!-Tuple{Stabilizer}","page":"API","title":"QuantumClifford.graphstate!","text":"An in-place version of graphstate.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.graphstate-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.graphstate","text":"Convert any stabilizer state to a graph state\n\nGraph states are a special type of entangled stabilizer states that can be represented by a graph. For a graph G=(VE) the corresponding stabilizers are S_v = X_v prod_u N(v) Z_u. Notice that such tableau rows contain only a single X operator. There is a set of single qubit gates that converts any stabilizer state to a graph state.\n\nThis function returns the graph state corresponding to a stabilizer and the gates that might be necessary to convert the stabilizer into a state representable as a graph.\n\nFor a tableau stab you can convert it with:\n\ngraph, hadamard_idx, iphase_idx, flips_idx = graphstate()\n\nwhere graph is the graph representation of stab, and the rest specifies the single-qubit gates converting stab to graph: hadamard_idx are the qubits that require a Hadamard gate (mapping X ↔ Z), iphase_idx are (different) qubits that require an inverse Phase gate (Y → X), and flips_idx are the qubits that require a phase flip (Pauli Z gate), after the previous two sets of gates.\n\njulia> using Graphs\n\njulia> s = S\" XXX\n ZZ_\n -_ZZ\";\n\n\njulia> g, h_idx, ip_idx, z_idx = graphstate(s);\n\n\njulia> collect(edges(g))\n2-element Vector{Graphs.SimpleGraphs.SimpleEdge{Int64}}:\n Edge 1 => 2\n Edge 1 => 3\n\njulia> h_idx\n2-element Vector{Int64}:\n 2\n 3\n\njulia> ip_idx\nInt64[]\n\njulia> z_idx\n1-element Vector{Int64}:\n 3\n\nThe Graphs.jl library provides many graph-theory tools and the MakieGraphs.jl library provides plotting utilities for graphs.\n\nYou can directly call the graph constructor on a stabilizer, if you just want the graph and do not care about the Clifford operation necessary to convert an arbitrary state to a state representable as a graph:\n\njulia> collect(edges( Graph(bell()) ))\n1-element Vector{Graphs.SimpleGraphs.SimpleEdge{Int64}}:\n Edge 1 => 2\n\nFor a version that does not copy the stabilizer, but rather performs transformations in-place, use graphstate!. It would perform canonicalize_gott! on its argument as it finds a way to convert it to a graph state.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.logdot-Tuple{QuantumClifford.AbstractStabilizer, QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.logdot","text":"Logarithm of the inner product between to Stabilizer states.\n\nIf the result is nothing, the dot inner product is zero. Otherwise the inner product is 2^(-logdot/2).\n\nThe actual inner product can be computed with LinearAlgebra.dot.\n\nBased on (Garcia et al., 2012).\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.logicalxview-Tuple{MixedDestabilizer}","page":"API","title":"QuantumClifford.logicalxview","text":"A view of the subtableau corresponding to the logical X operators. See also tab, stabilizerview, destabilizerview, logicalzview\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.logicalzview-Tuple{MixedDestabilizer}","page":"API","title":"QuantumClifford.logicalzview","text":"A view of the subtableau corresponding to the logical Z operators. See also tab, stabilizerview, destabilizerview, logicalxview\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.mctrajectories-Tuple{Any, Any}","page":"API","title":"QuantumClifford.mctrajectories","text":"Run multiple Monte Carlo trajectories and report the aggregate final statuses of each.\n\nSee also: pftrajectories, petrajectories\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.mctrajectory!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.mctrajectory!","text":"Run a single Monte Carlo sample, starting with (and modifying) state by applying the given circuit. Uses apply! under the hood.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.petrajectories-Tuple{Any, Any}","page":"API","title":"QuantumClifford.petrajectories","text":"Run a perturbative expansion to a given order. This is the main public function for the perturbative expansion approach.\n\nSee also: pftrajectories, mctrajectories\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pfmeasurements-Tuple{PauliFrame}","page":"API","title":"QuantumClifford.pfmeasurements","text":"pfmeasurements(frame::PauliFrame) -> Any\n\n\nReturns the measurement results for each frame in the PauliFrame instance.\n\nwarning: Relative measurements\nThe return measurements are relative to the reference measurements, i.e. they only say whether the reference measurements have been flipped in the given frame.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pfmeasurements-Tuple{Register, PauliFrame}","page":"API","title":"QuantumClifford.pfmeasurements","text":"pfmeasurements(register::Register, frame::PauliFrame) -> Any\n\n\nTakes the references measurements from the given Register and applies the flips as prescribed by the PauliFrame relative measurements. The result is the actual (non-relative) measurement results for each frame.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pfmeasurements-Tuple{Register}","page":"API","title":"QuantumClifford.pfmeasurements","text":"pfmeasurements(register::Register) -> Vector{Bool}\n\n\nReturns the measurements stored in the bits of the given Register.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pftrajectories","page":"API","title":"QuantumClifford.pftrajectories","text":"Perform a \"Pauli frame\" style simulation of a quantum circuit.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.pftrajectories-Tuple{Any}","page":"API","title":"QuantumClifford.pftrajectories","text":"pftrajectories(\n circuit;\n trajectories,\n threads\n) -> PauliFrame{Stabilizer{QuantumClifford.Tableau{Vector{UInt8}, LinearAlgebra.Adjoint{UInt64, Matrix{UInt64}}}}, Matrix{Bool}}\n\n\nThe main method for running Pauli frame simulations of circuits. See the other methods for lower level access.\n\nMultithreading is enabled by default, but can be disabled by setting threads=false. Do not forget to launch Julia with multiple threads enabled, e.g. julia -t4, if you want to use multithreading.\n\nSee also: mctrajectories, petrajectories\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pftrajectories-Tuple{PauliFrame, Any}","page":"API","title":"QuantumClifford.pftrajectories","text":"pftrajectories(state::PauliFrame, circuit) -> PauliFrame\n\n\nEvolve each frame stored in PauliFrame by the given circuit.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pftrajectories-Tuple{Register, Any}","page":"API","title":"QuantumClifford.pftrajectories","text":"pftrajectories(\n register::Register,\n circuit;\n trajectories\n) -> Tuple{Register, PauliFrame{Stabilizer{QuantumClifford.Tableau{Vector{UInt8}, LinearAlgebra.Adjoint{UInt64, Matrix{UInt64}}}}, Matrix{Bool}}}\n\n\nFor a given Register and circuit, simulates the reference circuit acting on the register and then also simulate numerous PauliFrame trajectories. Returns the register and the PauliFrame instance.\n\nUse pfmeasurements to get the measurement results.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.phases-Tuple{QuantumClifford.Tableau}","page":"API","title":"QuantumClifford.phases","text":"The phases of a given tableau. It is a view, i.e. if you modify this array, the original tableau caries these changes.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.prodphase-Union{Tuple{T}, Tuple{AbstractVector{T}, AbstractVector{T}}} where T<:Unsigned","page":"API","title":"QuantumClifford.prodphase","text":"Get the phase of the product of two Pauli operators.\n\nPhase is encoded as F(4) in the low qubits of an UInt8.\n\njulia> P\"ZZZ\"*P\"XXX\"\n-iYYY\n\njulia> prodphase(P\"ZZZ\", P\"XXX\")\n0x03\n\njulia> prodphase(P\"XXX\", P\"ZZZ\")\n0x01\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.projectXrand!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.projectXrand!","text":"projectXrand!(state, qubit) -> Tuple{Register, UInt8}\n\n\nProject qubit of state along the X axis and randomize the phase if necessary.\n\nLower boilerplate version of project!.\n\nSee also: project!, projectX!, projectZrand!, projectYrand!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.projectYrand!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.projectYrand!","text":"projectYrand!(state, qubit) -> Tuple{Register, UInt8}\n\n\nProject qubit of state along the Y axis and randomize the phase if necessary.\n\nLower boilerplate version of project!.\n\nSee also: project!, projectY!, projectXrand!, projectZrand!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.projectZrand!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.projectZrand!","text":"projectZrand!(state, qubit) -> Tuple{Register, UInt8}\n\n\nProject qubit of state along the Z axis and randomize the phase if necessary.\n\nLower boilerplate version of project!.\n\nSee also: project!, projectZ!, projectXrand!, projectYrand!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.projectrand!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.projectrand!","text":"projectrand!(state, pauli) -> Tuple{Register, Any}\n\n\nMeasure pauli operator on state and randomize the phase if necessary.\n\nLower boilerplate version of project!.\n\nSee also: project!, projectXrand!, projectZrand!, projectYrand!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.puttableau!-Union{Tuple{M2}, Tuple{M1}, Tuple{T}, Tuple{V2}, Tuple{V1}, Tuple{B}, Tuple{QuantumClifford.Tableau{V1, M1}, QuantumClifford.Tableau{V2, M2}, Int64, Int64}} where {B, V1, V2, T<:Unsigned, M1<:AbstractMatrix{T}, M2<:AbstractMatrix{T}}","page":"API","title":"QuantumClifford.puttableau!","text":"Put source tableau in target tableau at given row and column. Assumes target location is zeroed out.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.quantumstate","page":"API","title":"QuantumClifford.quantumstate","text":"Only the quantum part of the state (excluding classical bits)\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.random_clifford-Tuple{Random.AbstractRNG, Int64}","page":"API","title":"QuantumClifford.random_clifford","text":"A random Clifford operator generated by the Bravyi-Maslov Algorithm 2 from (Bravyi and Maslov, 2021).\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.random_clifford1-Tuple{Random.AbstractRNG, Any}","page":"API","title":"QuantumClifford.random_clifford1","text":"Random symbolic single-qubit Clifford applied to qubit at index qubit.\n\nSee also: SingleQubitOperator, random_clifford\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.random_destabilizer-Tuple{Random.AbstractRNG, Int64}","page":"API","title":"QuantumClifford.random_destabilizer","text":"A random Stabilizer/Destabilizer tableau generated by the Bravyi-Maslov Algorithm 2 from (Bravyi and Maslov, 2021).\n\nrandom_destabilizer(n) gives a n-qubit tableau of rank n. random_destabilizer(r,n) gives a n-qubit tableau of rank r.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.random_pauli","page":"API","title":"QuantumClifford.random_pauli","text":"A random Pauli operator on n qubits.\n\nUse nophase=false to randomize the phase. Use realphase=false to get operators with phases including ±i.\n\nOptionally, a \"flip\" probability p can be provided specified, in which case each bit is set to I with probability 1-p and to X or Y or Z with probability p. Useful for simulating unbiased Pauli noise.\n\nSee also random_pauli!\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.random_pauli!","page":"API","title":"QuantumClifford.random_pauli!","text":"An in-place version of random_pauli\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.random_stabilizer-Tuple{Random.AbstractRNG, Int64}","page":"API","title":"QuantumClifford.random_stabilizer","text":"A random Stabilizer tableau generated by the Bravyi-Maslov Algorithm 2 from (Bravyi and Maslov, 2021).\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.single_x-Tuple{Any, Any}","page":"API","title":"QuantumClifford.single_x","text":"A multiqubit operator corresponding to all identities except for Pauli X at i. See also: sX, sMX\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.single_y-Tuple{Any, Any}","page":"API","title":"QuantumClifford.single_y","text":"A multiqubit operator corresponding to all identities except for Pauli Y at i. See also: sY, sMY\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.single_z-Tuple{Any, Any}","page":"API","title":"QuantumClifford.single_z","text":"A multiqubit operator corresponding to all identities except for Pauli Z at i. See also: sY, sMY\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.stab_to_gf2-Tuple{QuantumClifford.Tableau}","page":"API","title":"QuantumClifford.stab_to_gf2","text":"The F(2,2) matrix of a given tableau, represented as the concatenation of two binary matrices, one for X and one for Z.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.stabilizerplot","page":"API","title":"QuantumClifford.stabilizerplot","text":"A Makie.jl recipe for pictorial representation of a tableau.\n\nRequires a Makie.jl backend to be loaded, e.g. using CairoMakie.\n\nAlternatively, you can use the Plots.jl plotting ecosystem, e.g. using Plots; plot(S\"XXX ZZZ\").\n\nConsult the documentation for more details on visualization options.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.stabilizerplot_axis","page":"API","title":"QuantumClifford.stabilizerplot_axis","text":"A Makie.jl recipe for pictorial representation of a tableau.\n\nRequires a Makie.jl backend to be loaded, e.g. using CairoMakie.\n\nAlternatively, you can use the Plots.jl plotting ecosystem, e.g. using Plots; plot(S\"XXX ZZZ\").\n\nConsult the documentation for more details on visualization options.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.stabilizerview-Tuple{Stabilizer}","page":"API","title":"QuantumClifford.stabilizerview","text":"A view of the subtableau corresponding to the stabilizer. See also tab, destabilizerview, logicalxview, logicalzview\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.symplecticGS-Tuple{PauliOperator}","page":"API","title":"QuantumClifford.symplecticGS","text":"Perform the Symplectic Gram-Schmidt procedure that gives a Clifford operator canonically related to a given Pauli operator.\n\nThe algorithm is detailed in (Koenig and Smolin, 2014).\n\nSee also: enumerate_cliffords, clifford_cardinality.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.tab-Union{Tuple{Stabilizer{T}}, Tuple{T}} where T","page":"API","title":"QuantumClifford.tab","text":"Extract the underlying tableau structure.\n\njulia> s = S\"X\"\n+ X\n\njulia> tab(s)\n+ X\n\njulia> tab(Destabilizer(s))\n+ Z\n+ X\n\njulia> tab(MixedDestabilizer(s))\n+ Z\n+ X\n\njulia> tab(tHadamard)\n+ Z\n+ X\n\njulia> typeof(tab(tHadamard))\nQuantumClifford.Tableau{Vector{UInt8}, Matrix{UInt64}}\n\nSee also: stabilizerview, destabilizerview, logicalxview, logicalzview\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.xbit-Tuple{PauliOperator}","page":"API","title":"QuantumClifford.xbit","text":"Extract as a new bit array the X part of the UInt array of packed qubits of a given Pauli operator.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.xview-Tuple{PauliOperator}","page":"API","title":"QuantumClifford.xview","text":"Get a view of the X part of the UInt array of packed qubits of a given Pauli operator.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.zbit-Tuple{PauliOperator}","page":"API","title":"QuantumClifford.zbit","text":"Extract as a new bit array the Z part of the UInt array of packed qubits of a given Pauli operator.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.zview-Tuple{PauliOperator}","page":"API","title":"QuantumClifford.zview","text":"Get a view of the Y part of the UInt array of packed qubits of a given Pauli operator.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.apply!","page":"API","title":"QuantumInterface.apply!","text":"In QuantumClifford the apply! function is used to apply any quantum operation to a stabilizer state, including unitary Clifford operations, Pauli measurements, and noise. Thus, this function may result in a random/stochastic result (e.g. with measurements or noise).\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumInterface.embed-Tuple{Int64, Int64, PauliOperator}","page":"API","title":"QuantumInterface.embed","text":"Embed a Pauli operator in a larger Pauli operator.\n\njulia> embed(5, 3, P\"-Y\")\n- __Y__\n\njulia> embed(5, (3,5), P\"-YX\")\n- __Y_X\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.entanglement_entropy","page":"API","title":"QuantumInterface.entanglement_entropy","text":"Get bipartite entanglement entropy of a subsystem\n\nDefined as entropy of the reduced density matrix.\n\nIt can be calculated with multiple different algorithms, the most performant one depending on the particular case.\n\nCurrently implemented are the :clip (clipped gauge), :graph (graph state), and :rref (Gaussian elimination) algorithms. Benchmark your particular case to choose the best one.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumInterface.entanglement_entropy-Tuple{QuantumClifford.AbstractStabilizer, AbstractVector, Val{:graph}}","page":"API","title":"QuantumInterface.entanglement_entropy","text":"Get bipartite entanglement entropy by first converting the state to a graph and computing the rank of the adjacency matrix.\n\nBased on \"Entanglement in graph states and its applications\".\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.entanglement_entropy-Tuple{QuantumClifford.AbstractStabilizer, AbstractVector, Val{:rref}}","page":"API","title":"QuantumInterface.entanglement_entropy","text":"Get bipartite entanglement entropy by converting to RREF form (i.e., partial trace form).\n\nThe state will be partially canonicalized in an RREF form.\n\nSee also: canonicalize_rref!, traceout!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.entanglement_entropy-Tuple{QuantumClifford.AbstractStabilizer, UnitRange, Val{:clip}}","page":"API","title":"QuantumInterface.entanglement_entropy","text":"Get bipartite entanglement entropy of a contiguous subsystem by passing through the clipped gauge.\n\nIf clip=false is set the canonicalization step is skipped, useful if the input state is already in the clipped gauge.\n\nSee also: bigram, canonicalize_clip!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.expect-Tuple{PauliOperator, QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumInterface.expect","text":"expect(p::PauliOperator, st::AbstractStabilizer)\n\nCompute the expectation value of a Pauli operator p on a stabilizer state st. This function will allocate a temporary copy of the stabilizer state st.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.nqubits-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumInterface.nqubits","text":"The number of qubits of a given state.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.project!-Tuple{Any, PauliOperator}","page":"API","title":"QuantumInterface.project!","text":"project!(\n state,\n pauli::PauliOperator;\n keep_result,\n phases\n) -> Tuple{MixedStabilizer, Any, Any}\n\n\nProject the state of a Stabilizer on the two eigenspaces of a Pauli operator.\n\nAssumes the input is a valid stabilizer. The projection is done inplace on that stabilizer and it does not modify the projection operator.\n\nIt returns\n\na stabilizer that might not be in canonical form\nthe index of the row where the non-commuting operator was (that row is now equal to pauli; its phase is not updated and for a faithful measurement simulation it needs to be randomized by the user)\nand the result of the projection if there was no non-commuting operator (nothing otherwise)\n\nIf keep_result==false that result of the projection in case of anticommutation is not computed, sparing a canonicalization operation. This canonicalization operation is the only one potentially of cubic complexity. The rest of the calculations are of quadratic complexity.\n\nIf you need to measure a single qubit instead of a multiqubit Pauli operator, the faster projectX!, projectY!, and projectZ! are available.\n\nFor less boilerplate and automatic randomization of the phase use projectrand!.\n\nHere is an example of a projection destroying entanglement:\n\njulia> ghz = S\"XXXX\n ZZII\n IZZI\n IIZZ\";\n\n\njulia> canonicalize!(ghz)\n+ XXXX\n+ Z__Z\n+ _Z_Z\n+ __ZZ\n\njulia> state, anticom_index, result = project!(ghz, P\"ZIII\");\n\n\njulia> state\n+ Z___\n+ Z__Z\n+ _Z_Z\n+ __ZZ\n\njulia> canonicalize!(state)\n+ Z___\n+ _Z__\n+ __Z_\n+ ___Z\n\njulia> anticom_index, result\n(1, nothing)\n\nAnd an example of projection consistent with the stabilizer state.\n\njulia> s = S\"ZII\n IXI\n IIY\";\n\n\njulia> canonicalize!(s)\n+ _X_\n+ __Y\n+ Z__\n\njulia> state, anticom_index, result = project!(s, P\"-ZII\");\n\n\njulia> state\n+ _X_\n+ __Y\n+ Z__\n\njulia> anticom_index, result\n(0, 0x02)\n\nWhile not the best choice, Stabilizer can be used for mixed states, simply by providing an incomplete tableau. In that case it is possible to attempt to project on an operator that can not be generated by the provided stabilizer operators. In that case we have anticom_index==rank and result===nothing, where rank is the the new rank of the tableau, one more than the number of rows in the initial tableau. However, if keep_result was set to false, then anticom_index would stay at zero.\n\njulia> s = S\"XZI\n IZI\";\n\n\njulia> project!(s, P\"IIX\")[1]\n+ X__\n+ _Z_\n\nIf we had used MixedStabilizer we would have added the projector to the list of stabilizers.\n\njulia> s = one(MixedStabilizer, 2, 3)\n+ Z__\n+ _Z_\n\njulia> project!(s, P\"IIX\")[1]\n+ Z__\n+ _Z_\n+ __X\n\nHowever, MixedDestabilizer would be an even better choice as it has mathcalO(n^2) complexity instead of the mathcalO(n^3) complexity of *Stabilizer.\n\njulia> s = one(MixedDestabilizer, 2, 3)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ X__\n+ _X_\n𝒳ₗ━━━\n+ __X\n𝒮𝓉𝒶𝒷━\n+ Z__\n+ _Z_\n𝒵ₗ━━━\n+ __Z\n\njulia> project!(s, P\"IIX\")[1]\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ X__\n+ _X_\n+ __Z\n𝒮𝓉𝒶𝒷━\n+ Z__\n+ _Z_\n+ __X\n\nSee the \"Datastructure Choice\" section in the documentation for more details.\n\nSee also: projectX!, projectY!, projectZ!, projectrand!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.project!-Tuple{MixedStabilizer, PauliOperator}","page":"API","title":"QuantumInterface.project!","text":"project!(\n state::MixedStabilizer,\n pauli::PauliOperator;\n phases\n) -> Tuple{MixedStabilizer, Any, Any}\n\n\nWhen using project! on MixedStabilizer it automates some of the extra steps we encounter when implicitly using the Stabilizer datastructure to represent mixed states. Namely, it helps when the projector is not among the list of stabilizers:\n\njulia> s = S\"XZI\n IZI\";\n\n\njulia> ms = MixedStabilizer(s)\n+ X__\n+ _Z_\n\njulia> project!(ms, P\"IIY\")[1]\n+ X__\n+ _Z_\n+ __Y\n\nSimilarly to project! on Stabilizer, this function has cubic complexity when the Pauli operator commutes with all rows of the tableau. Most of the time it is better to simply use MixedDestabilizer representation.\n\nUnlike other project! methods, this one does not allow for keep_result=false, as the correct rank or anticommutation index can not be calculated without the expensive (cubic) canonicalization operation required by keep_result=true.\n\nSee the \"Datastructure Choice\" section in the documentation for more details.\n\nSee also: projectX!, projectY!, projectZ!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.projectX!-Tuple{MixedDestabilizer, Int64}","page":"API","title":"QuantumInterface.projectX!","text":"Measure a given qubit in the X basis. A faster special-case version of project!.\n\nSee also: project!, projectXrand!, projectY!, projectZ!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.projectY!-Tuple{MixedDestabilizer, Int64}","page":"API","title":"QuantumInterface.projectY!","text":"Measure a given qubit in the Y basis. A faster special-case version of project!.\n\nSee also: project!, projectYrand!, projectX!, projectZ!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.projectZ!-Tuple{MixedDestabilizer, Int64}","page":"API","title":"QuantumInterface.projectZ!","text":"Measure a given qubit in the Z basis. A faster special-case version of project!.\n\nSee also: project!, projectZrand!, projectY!, projectX!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.reset_qubits!-Tuple{MixedDestabilizer, QuantumClifford.AbstractStabilizer, Any}","page":"API","title":"QuantumInterface.reset_qubits!","text":"reset_qubits!(\n s::MixedDestabilizer,\n newstate::QuantumClifford.AbstractStabilizer,\n qubits;\n phases\n) -> MixedDestabilizer\n\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.reset_qubits!-Tuple{MixedStabilizer, Any, Any}","page":"API","title":"QuantumInterface.reset_qubits!","text":"reset_qubits!(\n s::MixedStabilizer,\n newstate,\n qubits;\n phases\n) -> MixedStabilizer\n\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.reset_qubits!-Tuple{Stabilizer, Any, Any}","page":"API","title":"QuantumInterface.reset_qubits!","text":"reset_qubits!(\n s::Stabilizer,\n newstate,\n qubits;\n phases\n) -> Any\n\n\nReset a given set of qubits to be in the state newstate. These qubits are traced out first, which could lead to \"nonlocal\" changes in the tableau.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.tensor","page":"API","title":"QuantumInterface.tensor","text":"Tensor product between operators or tableaux. See also tensor_pow.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumInterface.tensor_pow-Tuple{Union{QuantumClifford.AbstractCliffordOperator, QuantumClifford.AbstractStabilizer}, Any}","page":"API","title":"QuantumInterface.tensor_pow","text":"Repeated tensor product of an operators or a tableau. See also tensor.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.traceout!-Tuple{Stabilizer, Any}","page":"API","title":"QuantumInterface.traceout!","text":"traceout!(s::Stabilizer, qubits; phases, rank) -> Any\n\n\nTrace out a qubit.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.traceout!-Tuple{Union{MixedDestabilizer, MixedStabilizer}, Any}","page":"API","title":"QuantumInterface.traceout!","text":"traceout!(\n s::Union{MixedDestabilizer, MixedStabilizer},\n qubits;\n phases,\n rank\n) -> Any\n\n\n\n\n\n\n","category":"method"},{"location":"API/#Private-API","page":"API","title":"Private API","text":"","category":"section"},{"location":"API/","page":"API","title":"API","text":"danger: Private Implementation Details\nThese functions are used internally by the library and might be drastically modified or deleted without warning or deprecation.","category":"page"},{"location":"API/","page":"API","title":"API","text":"Modules = [QuantumClifford]\nPrivate = true\nPublic = false","category":"page"},{"location":"API/#QuantumClifford.AbstractMeasurement","page":"API","title":"QuantumClifford.AbstractMeasurement","text":"Supertype of all symbolic single-qubit measurements.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Tableau","page":"API","title":"QuantumClifford.Tableau","text":"Internal Tableau type for storing a list of Pauli operators in a compact form. No special semantic meaning is attached to this type, it is just a convenient way to store a list of Pauli operators. E.g. it is not used to represent a stabilizer state, or a stabilizer group, or a Clifford circuit.\n\n\n\n\n\n","category":"type"},{"location":"API/#Base.inv-Tuple{CliffordOperator}","page":"API","title":"Base.inv","text":"inv(\n c::CliffordOperator;\n phases\n) -> CliffordOperator{QuantumClifford.Tableau{Vector{UInt8}, Matrix{UInt64}}}\n\n\nInverse of a CliffordOperator\n\n\n\n\n\n","category":"method"},{"location":"API/#Base.permute!-Tuple{QuantumClifford.AbstractStabilizer, AbstractVector}","page":"API","title":"Base.permute!","text":"Permute the qubits (i.e., columns) of the state in place.\n\n\n\n\n\n","category":"method"},{"location":"API/#Base.permute!-Tuple{QuantumClifford.Tableau, AbstractVector}","page":"API","title":"Base.permute!","text":"Permute the qubits (i.e., columns) of the tableau in place.\n\n\n\n\n\n","category":"method"},{"location":"API/#LinearAlgebra.dot-Tuple{QuantumClifford.AbstractStabilizer, QuantumClifford.AbstractStabilizer}","page":"API","title":"LinearAlgebra.dot","text":"The inner product of two Stabilizers.\n\nBased on (Garcia et al., 2012).\n\nSee also: logdot\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._apply_nonthread!-Tuple{QuantumClifford.AbstractStabilizer, CliffordOperator, AbstractVector{Int64}}","page":"API","title":"QuantumClifford._apply_nonthread!","text":"Nonvectorized version of apply! used for unit tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._apply_nonthread!-Tuple{QuantumClifford.AbstractStabilizer, CliffordOperator}","page":"API","title":"QuantumClifford._apply_nonthread!","text":"Nonvectorized version of apply! used for unit tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._mul_left_nonvec!-Union{Tuple{T}, Tuple{AbstractVector{T}, AbstractVector{T}}} where T<:Unsigned","page":"API","title":"QuantumClifford._mul_left_nonvec!","text":"Nonvectorized version of mul_left! used for unit tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._remove_rowcol!-Tuple{MixedDestabilizer, Any, Any}","page":"API","title":"QuantumClifford._remove_rowcol!","text":"Unexported low-level function that removes a row (by shifting all rows up as necessary)\n\nBecause MixedDestabilizer is not mutable we return a new MixedDestabilizer with the same (modified) xzs array.\n\nUsed on its own, this function will break invariants. Meant to be used with projectremove!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._rowmove!-Union{Tuple{B}, Tuple{QuantumClifford.Tableau, Any, Any}} where B","page":"API","title":"QuantumClifford._rowmove!","text":"Unexported low-level function that moves row i to row j.\n\nUsed on its own, this function will break invariants. Meant to be used in _remove_rowcol!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._stim_prodphase-Union{Tuple{T}, Tuple{AbstractVector{T}, AbstractVector{T}}} where T<:Unsigned","page":"API","title":"QuantumClifford._stim_prodphase","text":"The quantumlib/Stim implementation, which performs the prodphase and mul_left! together. Used for unit tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.apply_single_x!-Tuple{QuantumClifford.AbstractStabilizer, Any}","page":"API","title":"QuantumClifford.apply_single_x!","text":"Apply a Pauli X to the i-th qubit of state s. You should use apply!(stab,sX(i)) instead of this.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.apply_single_y!-Tuple{QuantumClifford.AbstractStabilizer, Any}","page":"API","title":"QuantumClifford.apply_single_y!","text":"Apply a Pauli Y to the i-th qubit of state s. You should use apply!(stab,sY(i)) instead of this.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.apply_single_z!-Tuple{QuantumClifford.AbstractStabilizer, Any}","page":"API","title":"QuantumClifford.apply_single_z!","text":"Apply a Pauli Z to the i-th qubit of state s. You should use apply!(stab,sZ(i)) instead of this.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.applynoise_branches","page":"API","title":"QuantumClifford.applynoise_branches","text":"Compute all possible new states after the application of the given noise model. Reports the probability of each one of them. Deterministic (as it reports all branches of potentially random processes), part of the Noise interface.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.colswap!-Tuple{QuantumClifford.Tableau, Any, Any}","page":"API","title":"QuantumClifford.colswap!","text":"Swap two columns of a stabilizer in place.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.destab_looks_good-Tuple{Any}","page":"API","title":"QuantumClifford.destab_looks_good","text":"Check basic consistency requirements of a destabilizer. Used in tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_cliffords_slow-Tuple{Any, Any}","page":"API","title":"QuantumClifford.enumerate_cliffords_slow","text":"The O(n^4) implementation from (Koenig and Smolin, 2014) – their algorithm seems wrong as ⟨w'₁|wₗ⟩=bₗ which is not always zero.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.fill_tril-Tuple{Any, Any, Any}","page":"API","title":"QuantumClifford.fill_tril","text":"Assign (symmetric) random ints to off diagonals of matrix.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.gf2_H_standard_form_indices-Tuple{Any}","page":"API","title":"QuantumClifford.gf2_H_standard_form_indices","text":"The permutation of columns which turns a binary matrix into standard form. It is assumed the matrix has already undergone Gaussian elimination.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.initZ!-Tuple{PauliFrame}","page":"API","title":"QuantumClifford.initZ!","text":"initZ!(frame::PauliFrame) -> PauliFrame\n\n\nInject random Z errors over all frames and qubits for the supplied PauliFrame with probability 0.5.\n\nCalling this after initialization is essential for simulating any non-deterministic circuit. It is done automatically by most PauliFrame constructors.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.make_sumtype-Tuple{Any}","page":"API","title":"QuantumClifford.make_sumtype","text":"julia> make_sumtype([sCNOT])\nquote\n @sum_type CompactifiedGate :hidden begin\n sCNOT(::Int64, ::Int64)\n end\nend\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.make_sumtype_method","page":"API","title":"QuantumClifford.make_sumtype_method","text":"``` julia> makesumtypemethod([sCNOT], :apply!, (:s,)) quote function QuantumClifford.apply!(s, g::CompactifiedGate) @cases g begin sCNOT(q1, q2) => apply!(s, sCNOT(q1, q2)) end end end\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.make_sumtype_variant_constructor-Tuple{Any}","page":"API","title":"QuantumClifford.make_sumtype_variant_constructor","text":"julia> make_sumtype_variant_constructor(sCNOT)\n:(CompactifiedGate(g::sCNOT) = begin\n (CompactifiedGate').sCNOT(g.q1, g.q2)\nend)\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.make_variant-Tuple{DataType}","page":"API","title":"QuantumClifford.make_variant","text":"julia> make_variant(sCNOT)\n:(sCNOT(::Int64, ::Int64))\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.make_variant_deconstruct","page":"API","title":"QuantumClifford.make_variant_deconstruct","text":"julia> make_variant_deconstruct(sCNOT, :apply!, (:s,))\n:(sCNOT(q1, q2) => apply!(s, sCNOT(q1, q2)))\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.mixed_destab_looks_good-Tuple{Any}","page":"API","title":"QuantumClifford.mixed_destab_looks_good","text":"Check basic consistency requirements of a mixed destabilizer. Used in tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.mixed_stab_looks_good-Tuple{Any}","page":"API","title":"QuantumClifford.mixed_stab_looks_good","text":"Check basic consistency requirements of a mixed stabilizer. Used in tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pfmeasurement","page":"API","title":"QuantumClifford.pfmeasurement","text":"For a given simulated state, e.g. a PauliFrame instance, returns the measurement results.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.precise_inv-Tuple{Any}","page":"API","title":"QuantumClifford.precise_inv","text":"Inverting a binary matrix: uses floating point for small matrices and Nemo for large matrices.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.project_cond!-Union{Tuple{PHASES}, Tuple{RESET}, Tuple{IS}, Tuple{MixedDestabilizer, Int64, Val{IS}, Val{RESET}}} where {IS, RESET, PHASES}","page":"API","title":"QuantumClifford.project_cond!","text":"Internal method used to implement projectX!, projectZ!, and projectY!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.projectremoverand!-Union{Tuple{F}, Tuple{MixedDestabilizer, F, Any}} where F<:Union{typeof(projectX!), typeof(projectY!), typeof(projectZ!)}","page":"API","title":"QuantumClifford.projectremoverand!","text":"Unexported low-level function that projects a qubit and returns the result while making the tableau smaller by a qubit.\n\nBecause MixedDestabilizer is not mutable we return a new MixedDestabilizer with the same (modified) xzs array.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.quantum_mallows-Tuple{Any, Any}","page":"API","title":"QuantumClifford.quantum_mallows","text":"Sample (h, S) from the distribution P_n(h, S) from Bravyi and Maslov Algorithm 1.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.remove_column!-Union{Tuple{M}, Tuple{T}, Tuple{V}, Tuple{QuantumClifford.Tableau{V, M}, Int64}} where {V, T<:Unsigned, M<:AbstractMatrix{T}}","page":"API","title":"QuantumClifford.remove_column!","text":"Unexported low-level function that removes a column (by shifting all columns to the right of the target by one step to the left)\n\nBecause Tableau is not mutable we return a new Tableau with the same (modified) xzs array.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.rowdecompose-Tuple{Any, Union{Destabilizer, MixedDestabilizer}}","page":"API","title":"QuantumClifford.rowdecompose","text":"Decompose a Pauli P in terms of stabilizer and destabilizer rows from a given tableaux.\n\nFor given tableaux of rows destabilizer rows d_i and stabilizer rows s_i, there are boolean vectors b and c such that P = i^p prod_i d_i^b_i prod_i s_i^c_i.\n\nThis function returns p, b, c.\n\njulia> s = MixedDestabilizer(ghz(2))\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z_\n+ _X\n𝒮𝓉𝒶𝒷\n+ XX\n+ ZZ\n\njulia> phase, destab_rows, stab_rows = QuantumClifford.rowdecompose(P\"XY\", s)\n(3, Bool[1, 0], Bool[1, 1])\n\njulia> im^3 * P\"Z_\" * P\"XX\" * P\"ZZ\"\n+ XY\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.sample_geometric_2-Tuple{Any, Integer}","page":"API","title":"QuantumClifford.sample_geometric_2","text":"This function samples a number from 1 to n where n >= 1 probability of outputting i is proportional to 2^i\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.stab_looks_good-Tuple{Any}","page":"API","title":"QuantumClifford.stab_looks_good","text":"Check basic consistency requirements of a stabilizer. Used in tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.to_cpu","page":"API","title":"QuantumClifford.to_cpu","text":"copies the memory content of the object to CPU\n\nYou can only use this function if CUDA.jl is imported\n\nFor more advanced users to_cpu(data, element_type) will reinterpret elements of data and converts them to element_type. For example based on your CPU architecture, if working with matrices of UInt32 is faster than UInt64, you can use to_cpu(data, UInt32)\n\njulia> using QuantumClifford: to_cpu, to_gpu\n\njulia> using CUDA # without this import, to_cpu, to_gpu are just function\n\njulia> stab = S\"- X_Z\\n+ _ZZ\\n+ __Z\"\n- X_Z\n+ _ZZ\n+ __Z\n\njulia> stab_gpu = to_gpu(stab);\n\njulia> apply!(stab_gpu, sHadamard(1));\n\njulia> stab_result_cpu = to_cpu(stab_gpu)\n- Z_Z\n+ _ZZ\n+ __Z\n\njulia> using QuantumClifford: to_cpu, to_gpu\n\njulia> using CUDA # without this import, to_cpu, to_gpu are just function\n\njulia> pf_gpu = to_gpu(PauliFrame(1000, 2, 2));\njulia> circuit = [sMZ(1, 1), sHadamard(2), sMZ(2, 2)];\njulia> pftrajectories(pf_gpu, circuit);\njulia> measurements = to_cpu(pf_gpu.measurements);\n\nSee also: to_gpu\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.to_gpu","page":"API","title":"QuantumClifford.to_gpu","text":"copies the memory content of the object to GPU\n\nYou can only use this function if CUDA.jl is imported\n\nFor more advanced users to_gpu(data, element_type) will reinterpret elements of data and converts them to element_type. For example based on your GPU architecture, if working with matrices of UInt64 is faster than UInt32, you can use to_gpu(data, UInt64)\n\njulia> using QuantumClifford: to_cpu, to_gpu\n\njulia> using CUDA # without this import, to_cpu, to_gpu are just function\n\njulia> stab = S\"- X_Z\\n+ _ZZ\\n+ __Z\"\n- X_Z\n+ _ZZ\n+ __Z\n\njulia> stab_gpu = to_gpu(stab);\n\njulia> apply!(stab_gpu, sHadamard(1));\n\njulia> stab_result_cpu = to_cpu(stab_gpu)\n- Z_Z\n+ _ZZ\n+ __Z\n\njulia> using QuantumClifford: to_cpu, to_gpu\n\njulia> using CUDA # without this import, to_cpu, to_gpu are just function\n\njulia> pf_gpu = to_gpu(PauliFrame(1000, 2, 2));\njulia> circuit = [sMZ(1, 1), sHadamard(2), sMZ(2, 2)];\njulia> pftrajectories(pf_gpu, circuit);\njulia> measurements = to_cpu(pf_gpu.measurements);\n\nSee also: to_cpu\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.trusted_rank","page":"API","title":"QuantumClifford.trusted_rank","text":"A \"trusted\" rank which returns rank(state) for Mixed[De]Stabilizer and length(state) for [De]Stabilizer.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.zero!-Tuple{QuantumClifford.Tableau, Any}","page":"API","title":"QuantumClifford.zero!","text":"Zero-out a given row of a Tableau\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.zero!-Union{Tuple{PauliOperator{Tz, Tv}}, Tuple{Tv}, Tuple{Tve}, Tuple{Tz}} where {Tz, Tve<:Unsigned, Tv<:AbstractVector{Tve}}","page":"API","title":"QuantumClifford.zero!","text":"Zero-out the phases and single-qubit operators in a PauliOperator\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.@qubitop1-Tuple{Any, Any}","page":"API","title":"QuantumClifford.@qubitop1","text":"Macro used to define single qubit symbolic gates and their qubit_kernel methods.\n\n\n\n\n\n","category":"macro"},{"location":"API/#QuantumClifford.@qubitop2-Tuple{Any, Any}","page":"API","title":"QuantumClifford.@qubitop2","text":"Macro used to define 2-qubit symbolic gates and their qubit_kernel methods.\n\n\n\n\n\n","category":"macro"},{"location":"API/#QuantumClifford.@valbooldispatch-Tuple{Any, Vararg{Any}}","page":"API","title":"QuantumClifford.@valbooldispatch","text":"Turns f(Val(x)) into x ? f(Val(true)) : f(Val(false)) in order to avoid dynamic dispatch\n\nSee discourse discussion\n\n\n\n\n\n","category":"macro"},{"location":"mixed/#Mixed-Stabilizer-States","page":"Mixed States","title":"Mixed Stabilizer States","text":"","category":"section"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"DocTestSetup = quote\n using QuantumClifford\nend","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"The Stabilizer and Destabilizer have some support for mixed states (by being initialized with an incomplete list of stabilizer generators), but for most purposes one would use the Mixed* data structures.","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"Mixed stabilizer states are implemented with MixedStabilizer and MixedDestabilizer, the latter of which is the preferred data structure for most tasks as it is much faster by virtue of tracking the destabilizer generators.","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"julia> s = S\"XXX\n IZZ\";\n\njulia> Destabilizer(s)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ _X_\n𝒮𝓉𝒶𝒷━\n+ XXX\n+ _ZZ","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"Unlike Destabilizer, MixedDestabilizer also tracks the logical operation generators.","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"julia> m = MixedDestabilizer(s)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ _X_\n𝒳ₗ━━━\n+ _XX\n𝒮𝓉𝒶𝒷━\n+ XXX\n+ _ZZ\n𝒵ₗ━━━\n+ Z_Z\n\njulia> stabilizerview(m)\n+ XXX\n+ _ZZ\n\njulia> destabilizerview(m)\n+ Z__\n+ _X_\n\njulia> logicalxview(m)\n+ _XX\n\njulia> logicalzview(m)\n+ Z_Z","category":"page"},{"location":"mixed/#Gottesman-Canonicalization","page":"Mixed States","title":"Gottesman Canonicalization","text":"","category":"section"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"To obtain the logical operators we perform a different type of canonicalization, described in Gottesman's thesis and implemented in canonicalize_gott!. Unlike canonicalize! which uses only row operations, canonicalize_gott! performs column swaps as well. MixedDestabilizer undoes those swaps by default when instantiated, but that behavior can be turned off, if you prefer to work with the canonicalized tableau.","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"julia> s = S\"XXX\n ZIZ\";\n\njulia> MixedDestabilizer(s)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ __X\n𝒳ₗ━━━\n+ _X_\n𝒮𝓉𝒶𝒷━\n+ XXX\n+ Z_Z\n𝒵ₗ━━━\n+ ZZ_\n\njulia> MixedDestabilizer(s; undoperm=false)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ _X_\n𝒳ₗ━━━\n+ __X\n𝒮𝓉𝒶𝒷━\n+ XXX\n+ ZZ_\n𝒵ₗ━━━\n+ Z_Z","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"Destabilizer and MixedStabilizer do not use any column swaps on instantiation as they do not track the logical operators.","category":"page"},{"location":"datastructures/#Data-Structures-Options","page":"Datastructure Choice","title":"Data Structures Options","text":"","category":"section"},{"location":"datastructures/#Choosing-Appropriate-Data-Structure","page":"Datastructure Choice","title":"Choosing Appropriate Tableau Data Structure","text":"","category":"section"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"There are four different data structures used to represent stabilizer states. If you will never need projective measurements you probably would want to use Stabilizer. If you require projective measurements, but only on pure states, Destabilizer should be the appropriate data structure. If mixed stabilizer states are involved, MixedStabilizer would be necessary.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"Stabilizer is simply a list of Pauli operators in a tableau form. As a data structure it does not enforce the requirements for a pure stabilizer state (the rows of the tableau do not necessarily commute, nor are they forced to be Hermitian; the tableau might be underdetermined, redundant, or contradictory). It is up to the user to ensure that the initial values in the tableau are meaningful and consistent.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"canonicalize!, project!, and generate! can accept an under determined (mixed state) Stabilizer instance and operate correctly. canonicalize! can also accept a redundant Stabilizer (i.e. not all rows are independent), leaving as many identity rows at the bottom of the canonicalized tableau as the number of redundant stabilizers in the initial tableau.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"canonicalize! takes mathcalO(n^3) steps. generate! expects a canonicalized input and then takes mathcalO(n^2) steps. project! takes mathcalO(n^3) for projecting on commuting operators due to the need to call canonicalize! and generate!. If the projections is on an anticommuting operator (or if keep_result=false) then it takes mathcalO(n^2) steps.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"MixedStabilizer provides explicit tracking of the rank of the mixed state and works properly when the projection is on a commuting operator not in the stabilizer (see table below for details). Otherwise it has the same performance as Stabilizer.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"The canonicalization can be made unnecessary if we track the destabilizer generators. There are two data structures capable of that.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"Destabilizer stores both the destabilizer and stabilizer states. project! called on it never requires a stabilizer canonicalization, hence it runs in mathcalO(n^2). However, project! will raise an exception if you try to project on a commuting state that is not in the stabilizer as that would be an expensive mathcalO(n^3) operation.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"MixedDestabilizer tracks both the destabilizer operators and the logical operators in addition to the stabilizer generators. It does not require canonicalization for measurements and its project! operations always takes mathcalO(n^2).","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"For the operation _, anticom_index, result = project!(...) we have the following behavior:","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"projection Stabilizer MixedStabilizer Destabilizer MixedDestabilizer\non anticommuting operator anticom_index>0 result===nothing correct result in mathcalO(n^2) steps same as Stabilizer same as Stabilizer same as Stabilizer\non commuting operator in the stabilizer anticom_index==0 result!==nothing mathcalO(n^3); or mathcalO(n^2) if keep_result=false mathcalO(n^3) mathcalO(n^2) if the state is pure, throws exception otherwise mathcalO(n^2)\non commuting operator out of the stabilizer[1] anticom_index==rank result===nothing mathcalO(n^3), but the user needs to manually include the new operator to the stabilizer; or mathcalO(n^2) if keep_result=false but then result indistinguishable from cell above and anticom_index==0 mathcalO(n^3) and rank goes up by one not applicable if the state is pure, throws exception otherwise mathcalO(n^2) and rank goes up by one","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"Notice the results when the projection operator commutes with the state but is not generated by the stabilizers of the state (the last row of the table). In that case we have _, anticom_index, result = project!(...) where both anticom_index==rank and result===nothing, with rank being the new rank after projection, one more than the number of rows in the tableau before the measurement. ","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"[1]: This can occur only if the state being projected is mixed. Both Stabilizer and Destabilizer can be used for mixed states by simply providing fewer stabilizer generators than qubits at initialization. This can be useful for low-level code that tries to avoid the extra memory cost of using MixedStabilizer and MixedDestabilizer but should be avoided otherwise. project! works correctly or raises an explicit warning on all 4 data structures.","category":"page"},{"location":"datastructures/#Bit-Packing-in-Integers-and-Array-Order","page":"Datastructure Choice","title":"Bit Packing in Integers and Array Order","text":"","category":"section"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"We do not use boolean arrays to store information about the qubits as this would be wasteful (7 out of 8 bits in the boolean would be unused). Instead, we use all 8 qubits in a byte and perform bitwise logical operations as necessary. Implementation details of the object in RAM can matter for performance. The library permits any of the standard UInt types to be used for packing the bits, and larger UInt types (like UInt64) are usually faster as they permit working on 64 qubits at a time (instead of 1 if we used a boolean, or 8 if we used a byte).","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"Moreover, how a tableau is stored in memory can affect performance, as a row-major storage usually permits more efficient use of the CPU cache (for the particular algorithms we use).","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"Both of these parameters are benchmarked (testing the application of a Pauli operator, which is an mathcalO(n^2) operation; and testing the canonicalization of a Stabilizer, which is an mathcalO(n^3) operation). Row-major UInt64 is the best performing and it is used by default in this library.","category":"page"},{"location":"noisycircuits_ops/#noisycircuits_ops","page":"Circuit Operations","title":"Operators in Circuit Simulations","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"DocTestSetup = quote\n using QuantumClifford\n using QuantumClifford.Experimental.NoisyCircuits\n using Quantikz\nend\nCurrentModule = QuantumClifford.Experimental.NoisyCircuits","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"warning: Unstable\nThis is experimental functionality with an unstable API.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Import with using QuantumClifford.Experimental.NoisyCircuits.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Too see a condensed list of all operations check out the API docs.","category":"page"},{"location":"noisycircuits_ops/#Unitary-Gates","page":"Circuit Operations","title":"Unitary Gates","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"We distinguish between symbolic gates like sCNOT that have specialized (fast) apply! methods (usually just for single and two qubit gates) and general tableau representation of gates like CliffordOperator that can represent any multi-qubit gate.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Predefined unitary gates are available, like sCNOT, sHadamard, etc.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\n[sCNOT(2,4),sHadamard(2),sCPHASE(1,3),sSWAP(2,4)]","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Any arbitrary tableaux can be used as a gate too. ","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"They can be specified by giving a Clifford operator tableaux and the indices on which it acts (particularly useful for gates acting on a small part of a circuit):","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\nSparseGate(tCNOT, [2,4])","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"The Clifford operator tableaux can be completely arbitrary.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"SparseGate(random_clifford(3), [2,4,5])","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"If the Clifford operator acts on all qubits, we do not need to specify indices, just use the operator.","category":"page"},{"location":"noisycircuits_ops/#Noisy-Gates","page":"Circuit Operations","title":"Noisy Gates","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Each gate can be followed by noise applied to the qubits on which it has acted. This is done by wrapping the given gate into a NoisyGate","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"ε = 0.03 # X/Y/Z error probability\nnoise = UnbiasedUncorrelatedNoise(ε)\nnoisy_gate = NoisyGate(SparseGate(tCNOT, [2,4]), noise)","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"In circuit diagrams the noise is not depicted, but after each application of the gate defined in noisy_gate, a noise operator will also be applied. The example above is of Pauli Depolarization implemented by UnbiasedUncorrelatedNoise.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"One can also apply only the noise operator by using NoiseOp which acts only on specified qubits. Or alternatively, one can use NoiseOpAll in order to apply noise to all qubits.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"[NoiseOp(noise, [4,5]), NoiseOpAll(noise)]","category":"page"},{"location":"noisycircuits_ops/#Coincidence-Measurements","page":"Circuit Operations","title":"Coincidence Measurements","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Global parity measurements involving single-qubit projections and classical communication are implemented with BellMeasurement. One needs to specify the axes of measurement and the qubits being measured. If the parity is trivial, the circuit continues, if the parity is non-trivial, the circuit ends and reports a detected failure. This operator is frequently used in the simulation of entanglement purification.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"BellMeasurement([sMX(1), sMY(3), sMZ(4)])","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"There is also NoisyBellMeasurement that takes the bit-flip probability of a single-qubit measurement as a third argument.","category":"page"},{"location":"noisycircuits_ops/#Stabilizer-Measurements","page":"Circuit Operations","title":"Stabilizer Measurements","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"A measurement over one or more qubits can also be performed, e.g., a direct stabilizer measurement on multiple qubits without the use of ancillary qubits. When applied to multiple qubits, this differs from BellMeasurement as it performs a single projection, unlike BellMeasurement which performs a separate projection for every single qubit involved. This measurement is implemented in PauliMeasurement which requires a Pauli operator on which to project and the index of the classical bit in which to store the result. Alternatively, there are sMX, sMZ, sMY if you are measuring a single qubit.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"[PauliMeasurement(P\"XYZ\", 1), sMZ(2, 2)]","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"TODO: SparseMeasurement, NoisyMeasurement","category":"page"},{"location":"noisycircuits_ops/#Verification-Operations","page":"Circuit Operations","title":"Verification Operations","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"At the end of many circuits one might want to check whether they performed correctly. The VerifyOp operation corresponds to an unphysical perfect tomographic operation, checking whether the state of the qubits at the given indices is indeed what is expected. If it is, the operation reports a success, otherwise it reports an undetected error.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"desired_state = random_stabilizer(5)\nqubit_indices = [1,2,3,4,7]\nVerifyOp(desired_state, qubit_indices)","category":"page"},{"location":"noisycircuits_ops/#Reset-Operations","page":"Circuit Operations","title":"Reset Operations","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"The Reset operations lets you trace out the specified qubits and set their state to a specific tableau.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"new_state = random_stabilizer(3)\nqubit_indices = [1,2,3]\nReset(new_state, qubit_indices)","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"It can be done anywhere in a circuit, not just at the beginning.","category":"page"},{"location":"noisycircuits_ops/#Gates-Conditioned-on-Classical-Bits","page":"Circuit Operations","title":"Gates Conditioned on Classical Bits","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"ConditionalGate is a conditional gate that performs one of two provided gates, depending on the value of a given classical bit.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"DecisionGate is a conditional gate that performs one of the supplied gates, depending on the output of decisionfunction applied to the entire classical bit register.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"gate1 = SparseGate(tCNOT, [1,2])\ngate2 = sCPHASE(1,2)\ngate3 = SparseGate(tSWAP, [1,3])\ncg = ConditionalGate(gate1, gate2, 2)\ndg = DecisionGate([gate1,gate2,gate3], bit_register->1) # it will always perform gate1\n[sMX(4,1), sMZ(5,2), cg, dg]","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"TODO: Split ConditionalGate into quantum conditional and classical conditional","category":"page"},{"location":"plotting/#Visualizations","page":"Visualizations","title":"Visualizations","text":"","category":"section"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"Stabilizers have a plot recipe that can be used with Plots.jl or Makie.jl. It simply displays the corresponding parity check matrix (extracted with stab_to_gf2) as a bitmap image. Circuits can be visualized with Quantikz.jl.","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"Importing the aforementioned packages together with QuantumClifford is necessary to enable the plotting functionality (implemented as package extensions).","category":"page"},{"location":"plotting/#Plots.jl","page":"Visualizations","title":"Plots.jl","text":"","category":"section"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"In Plots.jl we have a simple recipe plot(s::Stabilizer; xzcomponents=...) where xzcomponents=:split plots the tableau heatmap in a wide form, X bits on the left, Z bits on the right; or xzcomponents=:together plots them overlapping, with different colors for I, X, Z, and Y.","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Plots\nplot(random_stabilizer(20,30), xzcomponents=:split)","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Plots\nplot(canonicalize!(random_stabilizer(20,30)))","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Plots\nplot(canonicalize_gott!(random_stabilizer(30))[1], xzcomponents=:split)","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Plots\nplot(canonicalize_gott!(random_stabilizer(30))[1]; xzcomponents=:together)","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Plots\nplot(canonicalize_rref!(random_stabilizer(20,30),1:30)[1]; xzcomponents=:together)","category":"page"},{"location":"plotting/#Makie.jl","page":"Visualizations","title":"Makie.jl","text":"","category":"section"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"Makie's heatmap can be directly called on Stabilizer.","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, CairoMakie\ns = S\"IIXZ\n ZZIZ\n YYIZ\n IIIZ\n ZZXZ\"\nf, ax, p = CairoMakie.heatmap(s)\nhidedecorations!(ax); hidespines!(ax); # remove ticks and spines\nax.aspect = DataAspect(); # set a one-to-one aspect ratio\nf","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"A full Makie recipe is available as well (supporting xzcomponents)","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, CairoMakie\ns = S\"IIXZ\n ZZIZ\n YYIZ\n IIIZ\n ZZXZ\"\nf, ax, p = stabilizerplot(s, xzcomponents=:together)\nhidedecorations!(ax); hidespines!(ax)\nax.aspect = DataAspect()\nf","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"You can easily add colorbars (and change the colormap) as well:","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, CairoMakie\nfig = Figure()\nax, p = stabilizerplot(fig[1, 1], s, colormap=cgrad(:heat, 4, categorical = true))\nhidedecorations!(ax)\nhidespines!(ax)\nxlims!(ax, 0.5, size(s,2)+0.5) # otherwise there is padding\nylims!(ax, 0.5, size(s,1)+0.5) # otherwise there is padding\n# set the aspect ratio of the plot\nax.aspect = DataAspect()\n# set the aspect ratio of the layout\ncolsize!(fig.layout, 1, Aspect(1, size(s,2)/size(s,1))) \nColorbar(fig[1, 2], p, ticks = (0:3, [\"I\", \"X\", \"Z\", \"Y\"]))\nfig","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"Or set a completely custom set of colors:","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"fig = Figure()\nax, p = stabilizerplot(fig[1, 1], s, colormap=cgrad([:lightgray,RGBf(1,0.4,0.4),RGBf(0.3,1,0.5),RGBf(0.4,0.4,1)], 4, categorical = true))\nhidedecorations!(ax)\nhidespines!(ax)\nxlims!(ax, 0.5, size(s,2)+0.5)\nylims!(ax, 0.5, size(s,1)+0.5)\nax.aspect = DataAspect()\ncolsize!(fig.layout, 1, Aspect(1, size(s,2)/size(s,1))) \nColorbar(fig[2, 1], p, ticks = (0:3, [\"I\", \"X\", \"Z\", \"Y\"]), vertical = false, flipaxis = false)\nfig","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"You might have noticed, Makie recipes do not let you edit the axes or figure, rather they only permit you to set the plot content. Which is why we use hidedecorations!, hidesplines!, and DataAspect to further modify the plot. However, these defaults are also available in stabilizerplot_axis.","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, CairoMakie\nf=Figure()\nstabilizerplot_axis(f[1,1],random_stabilizer(100))\nf","category":"page"},{"location":"plotting/#Quantikz.jl","page":"Visualizations","title":"Quantikz.jl","text":"","category":"section"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"With the Quantikz library you can visualize gates or sequences of gates.","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Quantikz\ncircuit = [sCNOT(1,2), SparseGate(random_clifford(4), [1,4,5,6]), sMZ(4)]","category":"page"},{"location":"#QuantumClifford.jl","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"DocTestSetup = quote\n using QuantumClifford\nend","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"QuantumClifford.jl is a Julia library for simulation of Clifford circuits, which are a subclass of quantum circuits that can be efficiently simulated on a classical computer.","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"This library uses the tableaux formalism[1] with the destabilizer improvements[2]. Pauli frames are supported for faster repeated simulation of noisy circuits. Various symbolic and algebraic tools for manipulating, converting, and visualizing states and circuits are also implemented. ","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"[1]: (Gottesman, 1998)","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"[2]: (Aaronson and Gottesman, 2004)","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The library consists of two main parts: Tools for working with the algebra of Stabilizer Tableaux and tools specifically for efficient Circuit Simulation.","category":"page"},{"location":"#Stabilizer-Tableau-Algebra","page":"QuantumClifford.jl","title":"Stabilizer Tableau Algebra","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The Stabilizer Tableau Algebra component of QuantumClifford.jl efficiently handles pure and mixed stabilizer states of thousands of qubits, along with support for sparse or dense Clifford operations acting upon them. It provides operations such as canonicalization, projection, generation , and partial traces. The code is vectorized and multithreaded, offering fast, in-place, and allocation-free implementations. Tools for conversion to graph states and for visualization of tableaux are available.","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"See the Stabilizer Tableau Algebra manual or the curated list of useful functions.","category":"page"},{"location":"#Example-Usage","page":"QuantumClifford.jl","title":"Example Usage","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"julia> using QuantumClifford\n\njulia> P\"X\" * P\"Z\"\n-iY\n\njulia> P\"X\" ⊗ P\"Z\"\n+ XZ\n\njulia> S\"-XX\n +ZZ\"\n- XX\n+ ZZ\n\njulia> tCNOT * S\"-XX\n +ZZ\"\n- X_\n+ _Z","category":"page"},{"location":"#Circuit-Simulation","page":"QuantumClifford.jl","title":"Circuit Simulation","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The circuit simulation component of QuantumClifford.jl enables Monte Carlo (or symbolic) simulations of noisy Clifford circuits. It provides three main simulation methods: mctrajectories, pftrajectories, and petrajectories. These methods offer varying levels of efficiency, accuracy, and insight.","category":"page"},{"location":"#Monte-Carlo-Simulations-with-Stabilizer-Tableaux-(mctrajectories)","page":"QuantumClifford.jl","title":"Monte Carlo Simulations with Stabilizer Tableaux (mctrajectories)","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The mctrajectories method runs Monte Carlo simulations using a Stabilizer tableau representation for the quantum states.","category":"page"},{"location":"#Monte-Carlo-Simulations-with-Pauli-Frames-(pftrajectories)","page":"QuantumClifford.jl","title":"Monte Carlo Simulations with Pauli Frames (pftrajectories)","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The pftrajectories method runs Monte Carlo simulations of Pauli frames over a single reference Stabilizer tableau simulation. This approach is much more efficient but supports a smaller class of circuits.","category":"page"},{"location":"#Symbolic-Depth-First-Traversal-of-Quantum-Trajectories-(petrajectories)","page":"QuantumClifford.jl","title":"Symbolic Depth-First Traversal of Quantum Trajectories (petrajectories)","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The petrajectories method performs a depth-first traversal of the most probable quantum trajectories, providing a fixed-order approximation of the circuit's behavior. This approach gives symbolic expressions for various figures of merit instead of just a numeric value.","category":"page"}] +[{"location":"references/#Suggested-reading","page":"Suggested Readings & References","title":"Suggested reading","text":"","category":"section"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"For the basis of the tableaux methods first read (Gottesman, 1998) followed by the more efficient approach described in (Aaronson and Gottesman, 2004).","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"The tableaux can be canonicalized (i.e. Gaussian elimination can be performed on them) in a number of different ways, and considering the different approaches provides useful insight. The following methods are implemented in this library:","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"The default one: (Garcia et al., 2012)\nUseful when in need of tracing out a set of qubits: (Audenaert and Plenio, 2005)\nUseful when defining logical operators of codes: (Gottesman, 1997)","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"For the use of these methods in error correction and the subtle overlap between the two fields consider these resources. They are also useful in defining some of the specific constraints in commutation between rows in the tableaux:","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"(Steane, 2007)\n(Calderbank et al., 1998)\n(MacKay et al., 2004)\n(Wilde, 2009)","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"These publications describe the uniform sampling of random stabilizer states:","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"(Koenig and Smolin, 2014)\n(Bravyi and Maslov, 2021)\n(Van Den Berg, 2021)\n(Li et al., 2019)","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"For circuit construction routines (for stabilizer measurements for a given code):","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"(Cleve and Gottesman, 1997)\n(Gottesman, 1997) (and its erratum)\n(Grassl, 2002)\n(Grassl, 2011)","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"For quantum code construction routines:","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"(Cleve and Gottesman, 1997)\n(Gottesman, 1996)\n(Gottesman, 1997)\n(Yu et al., 2013)\n(Chao and Reichardt, 2017)","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"For classical code construction routines:","category":"page"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"(Muller, 1954)\n(Reed, 1954)\n(Raaphorst, 2003)\n(Abbe et al., 2020)\n(Djordjevic, 2021)","category":"page"},{"location":"references/#References","page":"Suggested Readings & References","title":"References","text":"","category":"section"},{"location":"references/","page":"Suggested Readings & References","title":"Suggested Readings & References","text":"Aaronson, S. and Gottesman, D. (2004). Improved simulation of stabilizer circuits. Physical Review A 70, 052328.\n\n\n\nAbbe, E.; Shpilka, A. and Ye, M. (2020). Reed–Muller codes: Theory and algorithms. IEEE Transactions on Information Theory 67, 3251–3277.\n\n\n\nAudenaert, K. M. and Plenio, M. B. (2005). Entanglement on mixed stabilizer states: normal forms and reduction procedures. New Journal of Physics 7, 170.\n\n\n\nBravyi, S. and Maslov, D. (2021). Hadamard-free circuits expose the structure of the Clifford group. IEEE Transactions on Information Theory 67, 4546–4563.\n\n\n\nCalderbank, A. R.; Rains, E. M.; Shor, P. and Sloane, N. J. (1998). Quantum error correction via codes over GF (4). IEEE Transactions on Information Theory 44, 1369–1387.\n\n\n\nChao, R. and Reichardt, B. W. (2017). Quantum Error Correction with Only Two Extra Qubits. Physical review letters 121 5, 050502.\n\n\n\nCleve, R. and Gottesman, D. (1997). Efficient computations of encodings for quantum error correction. Physical Review A 56, 76.\n\n\n\nDjordjevic, I. B. (2021). Quantum information processing, quantum computing, and quantum error correction: an engineering approach (Academic Press).\n\n\n\nGarcia, H. J.; Markov, I. L. and Cross, A. W. (2012). Efficient inner-product algorithm for stabilizer states, arXiv preprint arXiv:1210.6646.\n\n\n\nGottesman, D. (1996). Class of quantum error-correcting codes saturating the quantum Hamming bound. Physical Review A 54, 1862.\n\n\n\nGottesman, D. (1997). Stabilizer codes and quantum error correction. Ph.D. Thesis, California Institute of Technology.\n\n\n\nGottesman, D. (1998). The Heisenberg representation of quantum computers. In: International Conference on Group Theoretic Methods in Physics (Citeseer).\n\n\n\nGrassl, M. (2002). Algorithmic aspects of quantum error-correcting codes. Mathematics of Quantum Computation, 223–252.\n\n\n\nGrassl, M. (2011). Variations on encoding circuits for stabilizer quantum codes. In: International Conference on Coding and Cryptology (Springer); pp. 142–158.\n\n\n\nGullans, M. J.; Krastanov, S.; Huse, D. A.; Jiang, L. and Flammia, S. T. (2020). Quantum coding with low-depth random circuits, arXiv preprint arXiv:2010.09775.\n\n\n\nKoenig, R. and Smolin, J. A. (2014). How to efficiently select an arbitrary Clifford group element. Journal of Mathematical Physics 55, 122202.\n\n\n\nKrastanov, S.; de la Cerda, A. S. and Narang, P. (2020). Heterogeneous Multipartite Entanglement Purification for Size-Constrained Quantum Devices, arXiv preprint arXiv:2011.11640.\n\n\n\nLi, Y.; Chen, X. and Fisher, M. P. (2019). Measurement-driven entanglement transition in hybrid quantum circuits. Physical Review B 100, 134306.\n\n\n\nMacKay, D. J.; Mitchison, G. and McFadden, P. L. (2004). Sparse-graph codes for quantum error correction. IEEE Transactions on Information Theory 50, 2315–2330.\n\n\n\nMuller, D. E. (1954). Application of Boolean algebra to switching circuit design and to error detection. Transactions of the IRE professional group on electronic computers, 6–12.\n\n\n\nNahum, A.; Ruhman, J.; Vijay, S. and Haah, J. (2017). Quantum Entanglement Growth under Random Unitary Dynamics. Physical Review X 7, 031016.\n\n\n\nRaaphorst, S. (2003). Reed-muller codes. Carleton University, May 9.\n\n\n\nReed, I. S. (1954). A class of multiple-error-correcting codes and the decoding scheme. IEEE Transactions on Information Theory 4, 38–49.\n\n\n\nSteane, A. M. (2007). A tutorial on quantum error correction. In: PROCEEDINGS-INTERNATIONAL SCHOOL OF PHYSICS ENRICO FERMI, Vol. 162 (IOS Press; Ohmsha; 1999); p. 1.\n\n\n\nVan Den Berg, E. (2021). A simple method for sampling random Clifford operators. In: 2021 IEEE International Conference on Quantum Computing and Engineering (QCE) (IEEE); pp. 54–59.\n\n\n\nWilde, M. M. (2009). Logical operators of quantum codes. Physical Review A 79, 062322.\n\n\n\nYu, S.; Bierbrauer, J.; Dong, Y.; Chen, Q. and Oh, C. H. (2013). All the Stabilizer Codes of Distance 3. IEEE Transactions on Information Theory 59, 5179–5185.\n\n\n\n","category":"page"},{"location":"noise/#noise","page":"Noise Processes","title":"Noise Processes","text":"","category":"section"},{"location":"noise/","page":"Noise Processes","title":"Noise Processes","text":"DocTestSetup = quote\n using QuantumClifford\n using StableRNGs\n rng = StableRNG(42)\nend","category":"page"},{"location":"noise/","page":"Noise Processes","title":"Noise Processes","text":"As seen in the list of possible gates, the simulator is capable of modeling different types of noise. If that is your goal, please consider using the available Monte Carlo simulator or the Symbolic Perturbative Expansion system.","category":"page"},{"location":"noise/","page":"Noise Processes","title":"Noise Processes","text":"The implemented types of noise include:","category":"page"},{"location":"noise/","page":"Noise Processes","title":"Noise Processes","text":"UnbiasedUncorrelatedNoise\nPauliNoise","category":"page"},{"location":"noise/","page":"Noise Processes","title":"Noise Processes","text":"The low-level functionality to work with noise is applynoise!, but most of the time you would probably just want to use PauliError, NoisyGate, NoiseOp and NoiseOpAll.","category":"page"},{"location":"ecc_example_sim/#noisycircuits_pf_ecc_example","page":"ECC example","title":"ECC example with Pauli Frames","text":"","category":"section"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"DocTestSetup = quote\n using QuantumClifford\n using Quantikz\nend\nCurrentModule = QuantumClifford.Experimental.NoisyCircuits","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"warning: The documentation is incomplete\nWaiting for a better documentation than the small example below. Check out also the page on ECC performance evaluators","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"Consider Steane 7-qubit code:","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"using QuantumClifford\nusing QuantumClifford.ECC: Steane7, naive_syndrome_circuit, naive_encoding_circuit, parity_checks, code_s, code_n\nusing Quantikz\n\ncode = Steane7()\nH = parity_checks(code)","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"... and the corresponding encoding circuit","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"ecirc = naive_encoding_circuit(code)","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"... and the corresponding syndrome measurement circuit (the non-fault tolerant one)","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"scirc, _ = naive_syndrome_circuit(code)","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"The most straightforward way to start sampling syndromes is to set up a table of Pauli frames.","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"circuit = [ecirc..., scirc...]\nnframes = 4\nframes = pftrajectories(circuit; trajectories=nframes) # run the sims\npfmeasurements(frames) # extract the measurements","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"The pftrajectories function is multithreaded. If you want more low-level control over these Pauli frame simulations, check out the PauliFrame structure, the other methods of pftrajectories, and the circuit compactifaction function compactify_circuit.","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"If you want to model Pauli errors, use:","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"The helper PauliError for unbiased Pauli noise operation acting on a given qubit\nThe lower level NoiseOp (for a single qubit) or NoiseOpAll (for all qubits) parameterized with a particular noise type, e.g. UnbiasedUncorrelatedNoise","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"errprob = 0.1\nerrors = [PauliError(i,errprob) for i in 1:code_n(code)]\nfullcircuit = [ecirc..., errors..., scirc...]","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"And running this noisy simulation:","category":"page"},{"location":"ecc_example_sim/","page":"ECC example","title":"ECC example","text":"frames = pftrajectories(fullcircuit; trajectories=nframes)\npfmeasurements(frames)","category":"page"},{"location":"stab-algebra-manual/#Stabilizer-Tableau-Algebra-Manual","page":"Manual","title":"Stabilizer Tableau Algebra Manual","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"DocTestSetup = quote\n using QuantumClifford\nend","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The library consists of two main parts: Tools for working with the algebra of Stabilizer tableaux and tools specifically for efficient Circuit Simulation. This chapter discusses the former \"lower level\" Stabilizer tableau algebra tools.","category":"page"},{"location":"stab-algebra-manual/#Pauli-Operators","page":"Manual","title":"Pauli Operators","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The PauliOperator object represents multi-qubit Pauli operator (1iIZXY^otimes n). It is stored in memory as a phase (a single byte where 0x0,0x1,0x2,0x3 corresponds to 1i-1-i) and two bit-arrays, for X and for Z components.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"You can create them with a P string.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> P\"-iXZ\"\n-iXZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Or by specifying phase and X/Z components:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> PauliOperator(0x0,Bool[0,1,0],Bool[0,0,1])\n+ _XZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Both underscore and I can be used for identity.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> P\"I_XYZ\"\n+ __XYZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Multiplication with scalars or other Pauli operators works as expected, as well as tensor products of Pauli operators.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> -1im*P\"X\"\n-iX\n\njulia> P\"X\" * P\"Z\"\n-iY\n\njulia> P\"X\" ⊗ P\"Z\"\n+ XZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"One can check for commutativity with comm.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> comm(P\"X\",P\"Z\")\n0x01\n\njulia> comm(P\"XX\",P\"ZZ\")\n0x00","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"And check the phase of a product with prodphase.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> prodphase(P\"X\", P\"Z\")\n0x03\n\njulia> prodphase(P\"X\", P\"iZ\")\n0x00\n\njulia> prodphase(P\"X\",P\"Y\")\n0x01","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Indexing operations are available.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> p = P\"IXYZ\";\n\njulia> p[1], p[2], p[3], p[4]\n((false, false), (true, false), (true, true), (false, true))\n\njulia> p = P\"III\";\n\njulia> p[2] = (true, true);\n\njulia> p\n+ _Y_","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Including fancy indexing:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> P\"IXYZ\"[[2,3]]\n+ XY\n\njulia> P\"IXYZ\"[[false,true,true,false]]\n+ XY","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The operator is represented in memory by bit arrays (much denser than using byte arrays).","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> p = P\"-IXYZ\";\n\njulia> p.nqubits, p.xz\n(4, UInt64[0x0000000000000006, 0x000000000000000c])","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Views that give just the X or Z components of the xz bitarray are available through xview and zview.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> xview(P\"XYZI\")\n1-element view(::Vector{UInt64}, 1:1) with eltype UInt64:\n 0x0000000000000003","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The convenience methods xbit and zbit give you Bool (GF2) vectors.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> xbit(P\"XYZI\")\n4-element Vector{Bool}:\n 1\n 1\n 0\n 0","category":"page"},{"location":"stab-algebra-manual/#Stabilizers","page":"Manual","title":"Stabilizers","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"A Stabilizer object is a tableau of Pauli operators. When the tableau is meant to represent a (pure or mixed) stabilizer state, all of these operators should commute (but that is not enforced, rather Stabilizer is a generic tableau data structure). It is stored in memory as a phase list and a bit-matrix for X and Z components. It can be instantiated by an S string, or with a number of different constructors.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"tip: Stabilizers and Destabilizers\nIn many cases you probably would prefer to use the MixedDestabilizer data structure, as it caries a lot of useful additional information, like tracking rank and destabilizer operators. Stabilizer has mostly a pedagogical value, and it is also used for slightly faster simulation of a particular subset of Clifford operations. See also the data structures discussion page.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> S\"-XX\n +ZZ\"\n- XX\n+ ZZ\n\njulia> Stabilizer([P\"-XX\",P\"+ZZ\"])\n- XX\n+ ZZ\n\njulia> Stabilizer([0x2, 0x0],\n Bool[1 1;\n 0 0],\n Bool[0 0;\n 1 1])\n- XX\n+ ZZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Direct sums can be performed,","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> S\"-XX\" ⊗ S\"ZZ\"\n- XX__\n+ __ZZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Indexing operations are available, including fancy indexing. Be careful about how phase information gets transferred during sub-indexing.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XYZ\n -ZIX\n +XIZ\";\n\njulia> s[1]\n- XYZ\n\njulia> s[1,2]\n(true, true)\n\njulia> s[[3,1]]\n+ X_Z\n- XYZ\n\njulia> s[[3,1],[2]]\n+ _\n- Y","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Consistency at creation is not verified so nonsensical stabilizers can be created, both in terms of content and shape.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> S\"iX\n +Z\"\n+iX\n+ Z","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Similarly to the Pauli operators, a bit array representation is used.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZI\n -IZZ\"\n- XXX\n+ ZZ_\n- _ZZ\n\njulia> phases(s), tab(s).xzs\n(UInt8[0x02, 0x00, 0x02], UInt64[0x0000000000000007 0x0000000000000000 0x0000000000000000; 0x0000000000000000 0x0000000000000003 0x0000000000000006])","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"And there are convenience functions that can extract the corresponding binary check matrix.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> stab_to_gf2(s)\n3×6 Matrix{Bool}:\n 1 1 1 0 0 0\n 0 0 0 1 1 0\n 0 0 0 0 1 1","category":"page"},{"location":"stab-algebra-manual/#Canonicalization-of-Stabilizers","page":"Manual","title":"Canonicalization of Stabilizers","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Canonicalization (akin to Gaussian elimination over F(2,2)) is implemented in the canonicalize! function. Besides the default canonicalization prescription, alternative ones are available as described in the canonicalization page.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZX\n +III\";\n\njulia> canonicalize!(s)\n+ YY_\n+ ZZX\n+ ___","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"If phases are inconsequential, the operations can be faster by not tracking and updating them.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZX\n +III\";\n\njulia> canonicalize!(s; phases=false)\n- YY_\n+ ZZX\n+ ___","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"These operations are in place (as customarily signified by \"!\").","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZX\n +III\";\n\njulia> canonicalize!(s; phases=false);\n\njulia> s\n- YY_\n+ ZZX\n+ ___","category":"page"},{"location":"stab-algebra-manual/#Projective-Measurements","page":"Manual","title":"Projective Measurements","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The project! function is used to perform generic projective measurements.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"tip: Single qubit projections\nIf you know your Pauli measurement operator acts on a single qubit, there are much faster projection functions available, discussed in the next section. Namely projectX!, projectY!, and projectZ!.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"To observe the effect of different projections, we will start with a GHZ state.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZI\n -IZZ\";","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The project! function returns the new stabilizer, the index where the anticommutation was detected, and the result of the projection (nothing being an undetermined result). For instance here we project on an operator that does not commute with all stabilizer generators.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(copy(s), P\"ZII\")[1]\n+ Z__\n+ ZZ_\n- _ZZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Importantly, when there is an undetermined result, we return nothing and leave the phase of the new stabilizer the same as the phase of the projection operator. If you want to perform a Monte Carlo simulation, you need to randomize the phase of the stabilizer at the anticommuting index yourself. For instance, one can do:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> newstate, anticomindex, result = project!(copy(s), P\"XII\")\n if isnothing(result)\n phases(newstate)[anticomindex] = rand([0x0,0x2])\n end\n result, anticomindex\n(nothing, 2)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Of course, this is a rather cumbersome way to run a simulation, so we also provide projectrand! which does the necessary randomization automatically, for cases where you do not need the fine grained control of project!.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"We can project on a commuting operator, hence no anticommuting terms (the index is zero), and the result is perfectly determined (-1, or in our convention to represent the phase, 0x2).","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(copy(s), P\"-ZZI\")\n(Stabilizer 3×3, 0, 0x02)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"When the projection is consistent with the stabilizer (i.e. the measurement result is not nothing), this would trigger an expensive canonicalization procedure in order to calculate the measurement result (unless we are using more advanced data structures to represent the state, which are discussed later). If all you want to know is whether the projection is consistent with the stabilizer, but you do not care about the measurement result, you can skip the canonicalization and calculation of the result.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(copy(s), P\"-ZZI\", keep_result=false)\n(Stabilizer 3×3, 0, nothing)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Lastly, in either case, you can skip the calculation of the phases as well, if they are unimportant.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(copy(s), P\"ZZI\", phases=false)\n(Stabilizer 3×3, 0, 0x00)","category":"page"},{"location":"stab-algebra-manual/#Sparse-single-qubit-measurements","page":"Manual","title":"Sparse single-qubit measurements","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"In many circumstances only a single-qubit operator is being measured. In that case one should use the projectX!, projectY!, and projectZ! functions as they are much faster thanks to tracking only a single qubit. They have versions that randomize the phase as necessary as well: projectXrand!, projectYrand!, and projectZrand!.","category":"page"},{"location":"stab-algebra-manual/#Gate-like-interface","page":"Manual","title":"Gate-like interface","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"If you do not need all this boilerplate, and especially if you want to perform the randomization automatically, you can use the gate-like \"symbolic\" objects sMX, sMY, and sMZ, that perform the measurement and the necessary randomization of phase. If the measurement result is to be stored, you can use the Register structure that stores both stabilizer tableaux and bit values.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> state = Register(ghz(3), [false,false])\nRegister{Vector{UInt8}, Matrix{UInt64}}(Rank 3 stabilizer\n+ Z__\n+ _X_\n+ __X\n═════\n+ XXX\n+ ZZ_\n+ Z_Z\n═════\n, Bool[0, 0])\n\njulia> apply!(state, sMX(3,2)) # which qubit is measured and in which bit it is stored\nRegister{Vector{UInt8}, Matrix{UInt64}}(Rank 3 stabilizer\n+ Z__\n+ _X_\n+ Z_Z\n═════\n+ XXX\n+ ZZ_\n- __X\n═════\n, Bool[0, 1])\n\njulia> bitview(state)\n2-element Vector{Bool}:\n 0\n 1","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Or you can use the projectXrand!, projectYrand!, and projectZrand! if you prefer a function-call interface.","category":"page"},{"location":"stab-algebra-manual/#Partial-Traces","page":"Manual","title":"Partial Traces","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Partial trace (using traceout!) over even a single qubit might cause many of them to decohere due to entanglement.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> ghz = S\"XXX\n ZZ_\n _ZZ\";\n\njulia> traceout!(ghz, [1])\n+ _ZZ\n+ ___\n+ ___","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"This is somewhat more elegant when the datastructure being used explicitly supports mixed states.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> ghz = MixedStabilizer(S\"XXX\n ZZ_\n _ZZ\");\n\njulia> traceout!(ghz, [1])\n+ _ZZ","category":"page"},{"location":"stab-algebra-manual/#Generating-a-Pauli-Operator-with-Stabilizer-Generators","page":"Manual","title":"Generating a Pauli Operator with Stabilizer Generators","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The generate! function attempts to generate a Pauli operator by multiplying together the operators belonging to a given stabilizer (or reports their independence). This particular function requires the stabilizer to be already canonicalized.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"-XXX\n +ZZI\n -IZZ\";\n\njulia> s = canonicalize!(s)\n- XXX\n- Z_Z\n- _ZZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"It modifies the Pauli operator in place, reducing it to identity if possible. The leftover phase is present to indicate if the phase itself could not have been canceled. The list of indices specifies which rows of the stabilizer were used to generated the desired Pauli operator.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> generate!(P\"XYY\", s)\n(- ___, [1, 3])","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Phases can be neglected, for higher performance.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> generate!(P\"XYY\", s, phases=false)\n(+ ___, [1, 3])","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"If the Pauli operator can not be generated by the stabilizer, nothing value is returned.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> generate!(P\"ZZZ\", s)\n\njulia> generate!(P\"XZX\", s)\n\njulia> generate!(P\"YYY\", s)","category":"page"},{"location":"stab-algebra-manual/#Clifford-Operators","page":"Manual","title":"Clifford Operators","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The CliffordOperator structure represents a linear mapping between stabilizers (which should also preserve commutation relationships, but that is not checked at instantiation). These are n-qubit dense tableaux, representing an operation on n-qubit states. For single- or two-qubit gates, it is much more efficient to use small sparse symbolic clifford operators. A number of predefined Clifford operators are available, their name prefixed with t to mark them as dense tableaux.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> tHadamard\nX₁ ⟼ + Z\nZ₁ ⟼ + X\n\njulia> tPhase\nX₁ ⟼ + Y\nZ₁ ⟼ + Z\n\njulia> tCNOT\nX₁ ⟼ + XX\nX₂ ⟼ + _X\nZ₁ ⟼ + Z_\nZ₂ ⟼ + ZZ\n\njulia> tId1\nX₁ ⟼ + X\nZ₁ ⟼ + Z","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Chaining and tensor products are possible. Same for qubit permutations.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> tHadamard ⊗ tPhase\nX₁ ⟼ + Z_\nX₂ ⟼ + _Y\nZ₁ ⟼ + X_\nZ₂ ⟼ + _Z\n\njulia> tHadamard * tPhase\nX₁ ⟼ - Y\nZ₁ ⟼ + X\n\njulia> permute(tCNOT, [2,1])\nX₁ ⟼ + X_\nX₂ ⟼ + XX\nZ₁ ⟼ + ZZ\nZ₂ ⟼ + _Z","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"You can create custom Clifford operators with C-strings or with a list of Pauli operators.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> C\"-ZZ\n +_Z\n -X_\n +XX\"\nX₁ ⟼ - ZZ\nX₂ ⟼ + _Z\nZ₁ ⟼ - X_\nZ₂ ⟼ + XX\n\njulia> CliffordOperator([P\"-ZZ\", P\"_Z\", P\"-X_\", P\"XX\"])\nX₁ ⟼ - ZZ\nX₂ ⟼ + _Z\nZ₁ ⟼ - X_\nZ₂ ⟼ + XX","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Naturally, the operators can be applied to stabilizer states. This includes high performance in-place operations (and the phase can be neglected with phases=false for faster computation).","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> tCNOT * S\"X_\"\n+ XX\n\njulia> s = S\"X_\";\n\njulia> apply!(s,tCNOT)\n+ XX","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Sparse applications where a small Clifford operator is applied only on a particular subset of a larger stabilizer is also possible, but in such circumstances it is useful to consider using symbolic operators too.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s = S\"Z_YX\";\n\njulia> apply!(s, tCNOT, [4,2]) # Apply the CNOT on qubits 4 and 2\n+ ZXYX","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Pauli operators act as Clifford operators too (but they are rather boring, as they only change signs).","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> P\"XII\" * S\"ZXX\"\n- ZXX","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Internally, the CliffordOperator structure simply stores the tableau representation of the operation.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The apply! function is efficiently multithreaded for CliffordOperators. To start multithreaded Julia, use julia -t where specifies the number of threads.","category":"page"},{"location":"stab-algebra-manual/#Symbolic-Clifford-Operators","page":"Manual","title":"Symbolic Clifford Operators","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Much faster implementations for a number of common Clifford operators are available. They are stored as special named structs, instead of as a full tableau. These are the subtypes of AbstractSingleQubitOperator and AbstractTwoQubitOperator. Currently these are:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"using QuantumClifford # hide\nusing InteractiveUtils # hide\nsubtypes(QuantumClifford.AbstractSingleQubitOperator)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"subtypes(QuantumClifford.AbstractTwoQubitOperator)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Generally, they have the prefix s for symbolic/small/sparse. They are used slightly differently, as one needs to specify the qubits on which they act while instantiating them:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> sHadamard(2)\nsHadamard on qubit 2\nX₁ ⟼ + Z\nZ₁ ⟼ + X\n\njulia> sHadamard(2)*S\"XXX\"\n+ XZX\n\njulia> sCNOT(2,3)*S\"XYY\"\n- XXZ","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"The apply! function is efficiently multithreaded for these symbolic operators as well. To start multithreaded Julia, use julia -t where specifies the number of threads.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Symbolic projectors on single qubits also exist: sMX, sMY, sMZ. When used with the Register state representation, they can store the measurement results in the corresponding classical register.","category":"page"},{"location":"stab-algebra-manual/#Destabilizers","page":"Manual","title":"Destabilizers","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Slightly abusing the name: What we call \"destabilizers\" here is a stabilizer and its destabilizing operators saved together. They are implemented with the Destabilizer object and are initialized from a stabilizer.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> s=S\"-XXX\n -ZZI\n +IZZ\";\n\njulia> d = Destabilizer(s)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ _X_\n+ __X\n𝒮𝓉𝒶𝒷━\n- XXX\n- ZZ_\n- Z_Z","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"They have convenience methods to extract only the stabilizer and destabilizer pieces:","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> stabilizerview(d)\n- XXX\n- ZZ_\n- Z_Z\n\njulia> destabilizerview(d)\n+ Z__\n+ _X_\n+ __X","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Importantly commuting projections are much faster when tracking the destabilizer as canonicalization is not necessary (an mathcalO(n^2) complexity because it avoids the expensive mathcalO(n^3) canonicalization operation).","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(d,P\"ZZI\")\n(Destablizer 3×3, 0, 0x02)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Non-commuting projections are just as fast as when using only stabilizers.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> project!(d,P\"ZZZ\")\n(Destablizer 3×3, 1, nothing)","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Clifford operations can be applied the same way they are applied to stabilizers.","category":"page"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"julia> apply!(d,tCNOT⊗tHadamard)\n𝒟ℯ𝓈𝓉𝒶𝒷\n- X_Z\n+ XXZ\n+ X__\n𝒮𝓉𝒶𝒷━\n+ _ZX\n- _Z_\n- Z_X","category":"page"},{"location":"stab-algebra-manual/#Mixed-States","page":"Manual","title":"Mixed States","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"Both the Stabilizer and Destabilizer structures have more general forms that enable work with mixed stabilizer states. They are the MixedStabilizer and MixedDestabilizer structures, described in Mixed States. More information that can be seen in the data structures page, which expands upon the algorithms available for each structure.","category":"page"},{"location":"stab-algebra-manual/#Random-States-and-Circuits","page":"Manual","title":"Random States and Circuits","text":"","category":"section"},{"location":"stab-algebra-manual/","page":"Manual","title":"Manual","text":"random_clifford, random_stabilizer, and enumerate_cliffords can be used for the generation of random states.","category":"page"},{"location":"tutandpub/#tutandpub","page":"Tutorials and Publications","title":"Tutorials and Publications","text":"","category":"section"},{"location":"tutandpub/","page":"Tutorials and Publications","title":"Tutorials and Publications","text":"This list has a number of notebooks with tutorials, examples, and reproduction of published results (some of these results originally obtained with this very library).","category":"page"},{"location":"tutandpub/#On-the-topic-of-explicit-use-of-the-Tableaux-formalism-for-Stabilizer-states","page":"Tutorials and Publications","title":"On the topic of explicit use of the Tableaux formalism for Stabilizer states","text":"","category":"section"},{"location":"tutandpub/","page":"Tutorials and Publications","title":"Tutorials and Publications","text":"Quantum coding with low-depth random circuits reproducing results from (Gullans et al., 2020). view on nbviewer.jupyter.org","category":"page"},{"location":"tutandpub/#On-the-Monte-Carlo-and-Perturbative-Expansions-for-**Noisy**-Clifford-circuits","page":"Tutorials and Publications","title":"On the Monte Carlo and Perturbative Expansions for Noisy Clifford circuits","text":"","category":"section"},{"location":"tutandpub/","page":"Tutorials and Publications","title":"Tutorials and Publications","text":"In-depth study of multi-partite entanglement purification circuits reproducing results from (Krastanov et al., 2020). view on nbviewer.jupyter.org\nComparing the Monte Carlo and Perturbative method for noisy circuit simulations. view on nbviewer.jupyter.org\nShowcasing symbolic perturbative expansions of noisy circuits. view on nbviewer.jupyter.org","category":"page"},{"location":"noisycircuits_API/#Full-API-(autogenerated)","page":"API","title":"Full API (autogenerated)","text":"","category":"section"},{"location":"noisycircuits_API/","page":"API","title":"API","text":"warning: Unstable\nThis is experimental functionality with an unstable API.","category":"page"},{"location":"noisycircuits_API/","page":"API","title":"API","text":"Modules = [QuantumClifford.Experimental.NoisyCircuits]\nPrivate = false","category":"page"},{"location":"noisycircuits_API/#QuantumClifford.Experimental.NoisyCircuits.ConditionalGate","page":"API","title":"QuantumClifford.Experimental.NoisyCircuits.ConditionalGate","text":"A conditional gate that either performs truegate or falsegate, depending on the value of controlbit.\n\n\n\n\n\n","category":"type"},{"location":"noisycircuits_API/#QuantumClifford.Experimental.NoisyCircuits.DecisionGate","page":"API","title":"QuantumClifford.Experimental.NoisyCircuits.DecisionGate","text":"A conditional gate that performs one of the gates, depending on the output of decisionfunction applied to the entire classical bit register.\n\n\n\n\n\n","category":"type"},{"location":"noisycircuits_API/#QuantumClifford.Experimental.NoisyCircuits.NoisyBellMeasurement","page":"API","title":"QuantumClifford.Experimental.NoisyCircuits.NoisyBellMeasurement","text":"A Bell measurement in which each of the measured qubits has a chance to have flipped.\n\n\n\n\n\n","category":"type"},{"location":"noisycircuits_perturb/#noisycircuits_perturb","page":"Perturbative Expansions","title":"Perturbative expansions for simulating noisy Clifford circuits","text":"","category":"section"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"DocTestSetup = quote\n using QuantumClifford\n using QuantumClifford.Experimental.NoisyCircuits\n using Quantikz\nend\nCurrentModule = QuantumClifford.Experimental.NoisyCircuits","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"warning: Unstable\nThis is experimental functionality with an unstable API.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"Import with using QuantumClifford.Experimental.NoisyCircuits.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"This module enables the simulation of noisy Clifford circuits through a perturbative expansion in the noise parameter (assuming the noise is small). Instead of simulating many Monte Carlo trajectories, only the leading order trajectories are exhaustively enumerated and simulated.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"Here is an example of a purification circuit (the same circuit seen in the Monte Carlo example)","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\ngood_bell_state = S\"XX\n ZZ\"\ncanonicalize_rref!(good_bell_state)\ninitial_state = MixedDestabilizer(good_bell_state⊗good_bell_state)\n\ng1 = sCNOT(1,3) # CNOT between qubit 1 and qubit 3 (both with Alice)\ng2 = sCNOT(2,4) # CNOT between qubit 2 and qubit 4 (both with Bob)\nm = BellMeasurement([sMX(3),sMX(4)]) # Bell measurement on qubit 3 and 4\nv = VerifyOp(good_bell_state,[1,2]) # Verify that qubit 1 and 2 indeed form a good Bell pair\nepsilon = 0.01 # The error rate\nn = NoiseOpAll(UnbiasedUncorrelatedNoise(epsilon))\n\n# This circuit performs a depolarization at rate `epsilon` to all qubits,\n# then bilater CNOT operations\n# then a Bell measurement\n# followed by checking whether the final result indeed corresponds to the correct Bell pair.\ncircuit = [n,g1,g2,m,v]\n\npetrajectories(initial_state, circuit)","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"For more examples, see the notebook comparing the Monte Carlo and Perturbative method or this tutorial on entanglement purification.","category":"page"},{"location":"noisycircuits_perturb/#Symbolic-expansions","page":"Perturbative Expansions","title":"Symbolic expansions","text":"","category":"section"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"The perturbative expansion method works with symbolic variables as well. One can use any of the symbolic libraries available in Julia and simply plug symbolic parameters in lieu of numeric parameters. A detailed example is available as a Jupyter notebook.","category":"page"},{"location":"noisycircuits_perturb/#Interface-for-custom-operations","page":"Perturbative Expansions","title":"Interface for custom operations","text":"","category":"section"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"If you want to create a custom gate type (e.g. calling it Operation), you need to definite the following methods.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"applyop_branches!(s::T, g::Operation; max_order=1)::Vector{Tuple{T,Symbol,Real,Int}} where T is a tableaux type like Stabilizer or a Register. The Symbol is the status of the operation, the Real is the probability for that branch, and the Int is the order of that branch.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"There is also applynoise_branches! which is convenient for use in NoisyGate, but you can also just make up your own noise operator simply by implementing applyop_branches! for it.","category":"page"},{"location":"noisycircuits_perturb/","page":"Perturbative Expansions","title":"Perturbative Expansions","text":"You can also consult the list of implemented operators.","category":"page"},{"location":"graphs/#Graph-States","page":"Graph States","title":"Graph States","text":"","category":"section"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"warning: The `graphstate` API is not considered stable\ngraphstate returns a lot of information about encoding a given stabilizer state in a graph. A different API is being designed that streamlines the work with graph states.","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"Conversion to and from graph states is possible.","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"Consider a GHZ state:","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"using QuantumClifford # hide\nghz(4)","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"It can be converted to a graph state with graphstate","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"graphstate(ghz(4))[1]","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"using Random; Random.seed!(1); using QuantumClifford, GraphMakie, CairoMakie;\nf = Figure(resolution=(200,200))\na = Axis(f[1,1])\ngraphplot!(a,graphstate(ghz(4))[1])\nhidedecorations!(a); hidespines!(a)\na.aspect = DataAspect()\nsave(\"ghz4graph.png\", f); nothing","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"(Image: )","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"Notice that the initial GHZ state was not in the typical graph state form. We can see that explicitly by converting back and forth between the two forms:","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"julia> using Graphs, QuantumClifford\n\njulia> ghz(4)\n+ XXXX\n+ ZZ__\n+ _ZZ_\n+ __ZZ\n\njulia> Stabilizer(Graph(ghz(4)))\n+ XZZZ\n+ ZX__\n+ Z_X_\n+ Z__X","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"There is a set of single-qubit operations that can convert any stabilizer tableau into a state representable as a graph. These transformations are performed implicitly by the Graph constructor when converting from a Stabilizer. If you need the explicit transformation you can use the graphstate function that specifies which qubits require a Hadamard, Inverse Phase, or Phase Flip gate. The graph_gatesequence or graph_gate helper functions can be used to generate the exact operations:","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"julia> s = ghz(4)\n+ XXXX\n+ ZZ__\n+ _ZZ_\n+ __ZZ\n\njulia> g, h_idx, ip_idx, z_idx = graphstate(s);\n\njulia> gate = graph_gate(h_idx, ip_idx, z_idx, nqubits(s))\nX₁ ⟼ + X___\nX₂ ⟼ + _Z__\nX₃ ⟼ + __Z_\nX₄ ⟼ + ___Z\nZ₁ ⟼ + Z___\nZ₂ ⟼ + _X__\nZ₃ ⟼ + __X_\nZ₄ ⟼ + ___X\n\njulia> canonicalize!(apply!(s,gate)) == canonicalize!(Stabilizer(g))\ntrue","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"These converters also provides for a convenient way to create graph and cluster states, by using the helper constructors provided in Graphs.jl.","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"julia> Stabilizer(grid([4,1])) # Linear cluster state\n+ XZ__\n+ ZXZ_\n+ _ZXZ\n+ __ZX\n\njulia> Stabilizer(grid([2,2])) # Small 2D cluster state\n+ XZZ_\n+ ZX_Z\n+ Z_XZ\n+ _ZZX","category":"page"},{"location":"graphs/","page":"Graph States","title":"Graph States","text":"Graphs are represented with the Graphs.jl package and plotting can be done both in Plots.jl and Makie.jl (with GraphMakie).","category":"page"},{"location":"ECC_API/#Full-ECC-API-(autogenerated)","page":"API","title":"Full ECC API (autogenerated)","text":"","category":"section"},{"location":"ECC_API/","page":"API","title":"API","text":"Modules = [QuantumClifford.ECC]\nPrivate = false","category":"page"},{"location":"ECC_API/#QuantumClifford.ECC.CSS","page":"API","title":"QuantumClifford.ECC.CSS","text":"An arbitrary CSS error correcting code defined by its X and Z checks.\n\njulia> CSS([0 1 1 0; 1 1 0 0], [1 1 1 1]) |> parity_checks\n+ _XX_\n+ XX__\n+ ZZZZ\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.Cleve8","page":"API","title":"QuantumClifford.ECC.Cleve8","text":"A pedagogical example of a quantum error correcting [8,3] code used in (Cleve and Gottesman, 1997).\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.CommutationCheckECCSetup","page":"API","title":"QuantumClifford.ECC.CommutationCheckECCSetup","text":"Configuration for ECC evaluator that does not simulate any ECC circuits, rather it simply checks the commutation of the parity check and the Pauli error.\n\nThis is much faster than any other simulation method, but it is incapable of noisy-circuit simulations and thus useless for fault-tolerance studies.\n\nSee also: NaiveSyndromeECCSetup, ShorSyndromeECCSetup\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.Gottesman","page":"API","title":"QuantumClifford.ECC.Gottesman","text":"The family of [[2ʲ, 2ʲ - j - 2, 3]] Gottesman codes, also known as quantum Hamming codes, as described in Gottesman's 1997 PhD thesis and in (Gottesman, 1996).\n\nYou might be interested in consulting (Yu et al., 2013) and (Chao and Reichardt, 2017) as well.\n\nThe ECC Zoo has an entry for this family\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.NaiveSyndromeECCSetup","page":"API","title":"QuantumClifford.ECC.NaiveSyndromeECCSetup","text":"Configuration for ECC evaluator that runs the simplest syndrome measurement circuit.\n\nThe circuit is being simulated (as opposed to doing only a quick commutation check). This circuit would give poor performance if there is non-zero gate noise.\n\nSee also: CommutationCheckECCSetup, ShorSyndromeECCSetup\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.ShorSyndromeECCSetup","page":"API","title":"QuantumClifford.ECC.ShorSyndromeECCSetup","text":"Configuration for ECC evaluators that simulate the Shor-style syndrome measurement (without a flag qubit).\n\nThe simulated circuit includes:\n\nperfect noiseless encoding (encoding and its fault tolerance are not being studied here)\none round of \"memory noise\" after the encoding but before the syndrome measurement\nperfect preparation of entangled ancillary qubits\nnoisy Shor-style syndrome measurement (only two-qubit gate noise)\nnoiseless \"logical state measurement\" (providing the comparison data when evaluating the decoder)\n\nSee also: CommutationCheckECCSetup, NaiveSyndromeECCSetup\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.TableDecoder","page":"API","title":"QuantumClifford.ECC.TableDecoder","text":"A simple look-up table decoder for error correcting codes.\n\nThe lookup table contains only weight=1 errors, thus it is small, but at best it provides only for distance=3 decoding.\n\nThe size of the lookup table would grow exponentially quickly for higher distances.\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.Toric","page":"API","title":"QuantumClifford.ECC.Toric","text":"The Toric code.\n\n\n\n\n\n","category":"type"},{"location":"ECC_API/#QuantumClifford.ECC.BeliefPropDecoder-Tuple","page":"API","title":"QuantumClifford.ECC.BeliefPropDecoder","text":"A simple Belief Propagation decoder built around tools from LDPCDecoders.jl.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.PyBeliefPropDecoder-Tuple","page":"API","title":"QuantumClifford.ECC.PyBeliefPropDecoder","text":"A Belief Propagation decoder built around tools from the python package ldpc available from the julia package PyQDecoders.jl.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.PyBeliefPropOSDecoder-Tuple","page":"API","title":"QuantumClifford.ECC.PyBeliefPropOSDecoder","text":"A Belief Propagation decoder with ordered statistics decoding, built around tools from the python package ldpc available from the julia package PyQDecoders.jl.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.PyMatchingDecoder-Tuple","page":"API","title":"QuantumClifford.ECC.PyMatchingDecoder","text":"A perfect matching decoder built around tools from the python package pymatching available from the julia package PyQDecoders.jl.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.code_k-Tuple{Any}","page":"API","title":"QuantumClifford.ECC.code_k","text":"The number of logical qubits in a code.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.code_n","page":"API","title":"QuantumClifford.ECC.code_n","text":"The number of physical qubits in a code.\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.code_s","page":"API","title":"QuantumClifford.ECC.code_s","text":"The number of stabilizer checks in a code.\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.distance","page":"API","title":"QuantumClifford.ECC.distance","text":"The distance of a code.\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.evaluate_decoder-Tuple{QuantumClifford.ECC.AbstractSyndromeDecoder, QuantumClifford.ECC.AbstractECCSetup, Int64}","page":"API","title":"QuantumClifford.ECC.evaluate_decoder","text":"Evaluate the performance of a given decoder (e.g. TableDecoder) and a given style of running an ECC code (e.g. ShorSyndromeECCSetup)\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.evaluate_decoder-Tuple{QuantumClifford.ECC.AbstractSyndromeDecoder, Vararg{Any, 5}}","page":"API","title":"QuantumClifford.ECC.evaluate_decoder","text":"Evaluate the performance of an error-correcting circuit.\n\nThis method requires you give the circuit that performs both syndrome measurements and (probably noiseless) logical state measurements. The faults matrix that translates an error vector into corresponding logical errors is necessary as well.\n\nThis is a relatively barebones method that assumes the user prepares necessary circuits, etc. It is a method that is used internally by more user-frienly methods providing automatic conversion of codes and noise models to the necessary noisy circuits.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.faults_matrix-Tuple{Stabilizer}","page":"API","title":"QuantumClifford.ECC.faults_matrix","text":"Error-to-logical-observable map (a.k.a. fault matrix) of a code.\n\nFor a code with n physical qubits and k logical qubits this function returns a 2k × 2n binary matrix O such that O[i,j] is true if the logical observable of index i is flipped by the single physical qubit error of index j. Indexing is such that:\n\nO[1:k,:] is the error-to-logical-X-observable map (logical X observable, i.e. triggered by logical Z errors)\nO[k+1:2k,:] is the error-to-logical-Z-observable map\nO[:,1:n] is the X-physical-error-to-logical-observable map\nO[n+1:2n,:] is the Z-physical-error-to-logical-observable map\n\nE.g. for k=1, n=10, then if O[2,5] is true, then the logical Z observable is flipped by a X₅ error; and if O[1,12] is true, then the logical X observable is flipped by a Z₂ error.\n\nOf note is that there is a lot of freedom in choosing the logical operations! A logical operator multiplied by a stabilizer operator is still a logical operator. Similarly there is a different fault matrix for each choice of logical operators. But once the logical operators are picked, the fault matrix is fixed.\n\nBelow we show an example that uses the Shor code. While it is not the smallest code, it is a convenient choice to showcase the importance of the fault matrix when dealing with degenerate codes where a correction operation and an error do not need to be the same.\n\nFirst, consider a single-qubit error, potential correction operations, and their effect on the Shor code:\n\njulia> using QuantumClifford.ECC: faults_matrix, Shor9\n\njulia> state = MixedDestabilizer(Shor9())\n𝒟ℯ𝓈𝓉𝒶𝒷━━━━━\n+ Z________\n+ ___Z_____\n+ _X_______\n+ __X______\n+ ____X____\n+ _____X___\n+ ______X__\n+ _______X_\n𝒳ₗ━━━━━━━━━\n+ ______XXX\n𝒮𝓉𝒶𝒷━━━━━━━\n+ XXX___XXX\n+ ___XXXXXX\n+ ZZ_______\n+ Z_Z______\n+ ___ZZ____\n+ ___Z_Z___\n+ ______Z_Z\n+ _______ZZ\n𝒵ₗ━━━━━━━━━\n+ Z__Z____Z\n\njulia> err_Z₁ = single_z(9,1) # the error we will simulate\n+ Z________\n\njulia> cor_Z₂ = single_z(9,2) # the correction operation we will perform\n+ _Z_______\n\njulia> err_Z₁ * state # observe that one of the syndrome bits is now flipped\n𝒟ℯ𝓈𝓉𝒶𝒷━━━━━\n+ Z________\n+ ___Z_____\n+ _X_______\n+ __X______\n+ ____X____\n+ _____X___\n+ ______X__\n+ _______X_\n𝒳ₗ━━━━━━━━━\n+ ______XXX\n𝒮𝓉𝒶𝒷━━━━━━━\n- XXX___XXX\n+ ___XXXXXX\n+ ZZ_______\n+ Z_Z______\n+ ___ZZ____\n+ ___Z_Z___\n+ ______Z_Z\n+ _______ZZ\n𝒵ₗ━━━━━━━━━\n+ Z__Z____Z\n\njulia> cor_Z₂ * err_Z₁ * state # we are back to a good code state\n𝒟ℯ𝓈𝓉𝒶𝒷━━━━━\n+ Z________\n+ ___Z_____\n- _X_______\n+ __X______\n+ ____X____\n+ _____X___\n+ ______X__\n+ _______X_\n𝒳ₗ━━━━━━━━━\n+ ______XXX\n𝒮𝓉𝒶𝒷━━━━━━━\n+ XXX___XXX\n+ ___XXXXXX\n+ ZZ_______\n+ Z_Z______\n+ ___ZZ____\n+ ___Z_Z___\n+ ______Z_Z\n+ _______ZZ\n𝒵ₗ━━━━━━━━━\n+ Z__Z____Z\n\njulia> bad_Z₆Z₉ = single_z(9,6) * single_z(9,9) # a different \"correction\" operation\n+ _____Z__Z\n\njulia> bad_Z₆Z₉ * err_Z₁ * state # the syndrome is trivial, but now we have a logical error\n𝒟ℯ𝓈𝓉𝒶𝒷━━━━━\n+ Z________\n+ ___Z_____\n+ _X_______\n+ __X______\n+ ____X____\n- _____X___\n+ ______X__\n+ _______X_\n𝒳ₗ━━━━━━━━━\n- ______XXX\n𝒮𝓉𝒶𝒷━━━━━━━\n+ XXX___XXX\n+ ___XXXXXX\n+ ZZ_______\n+ Z_Z______\n+ ___ZZ____\n+ ___Z_Z___\n+ ______Z_Z\n+ _______ZZ\n𝒵ₗ━━━━━━━━━\n+ Z__Z____Z\n\nThe success of cor_Z₂ and the failure of bad_Z₆Z₉ can be immediately seen through the fault matrix, as the wrong \"correction\" does not result in the same logical flips ad the error:\n\njulia> O = faults_matrix(Shor9())\n2×18 BitMatrix:\n 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 1\n 1 0 0 1 0 0 0 0 1 0 0 0 0 0 0 0 0 0\n\njulia> O * stab_to_gf2(err_Z₁)\n2-element Vector{Int64}:\n 0\n 0\n\njulia> O * stab_to_gf2(cor_Z₂)\n2-element Vector{Int64}:\n 0\n 0\n\njulia> O * stab_to_gf2(bad_Z₆Z₉)\n2-element Vector{Int64}:\n 1\n 0\n\nWhile its use in this situation is rather contrived, the fault matrix is incredibly useful when running large scale simulations in which we want a separate fast error sampling process, (e.g. with Pauli frames) and a syndrome decoding process, without coupling between them. We just gather all our syndrome measurement and logical observables from the Pauli frame simulations, and then use them with the fault matrix in the syndrome decoding simulation.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.isdegenerate","page":"API","title":"QuantumClifford.ECC.isdegenerate","text":"Check if the code is degenerate with respect to a given set of error or with respect to all \"up to d physical-qubit\" errors (defaulting to d=1).\n\njulia> using QuantumClifford.ECC\n\njulia> isdegenerate(Shor9(), [single_z(9,1), single_z(9,2)])\ntrue\n\njulia> isdegenerate(Shor9(), [single_z(9,1), single_x(9,1)])\nfalse\n\njulia> isdegenerate(Steane7(), 1)\nfalse\n\njulia> isdegenerate(Steane7(), 2)\ntrue\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.naive_encoding_circuit-Tuple{Any}","page":"API","title":"QuantumClifford.ECC.naive_encoding_circuit","text":"Encoding physical qubits into a larger logical code.\n\nThe initial physical qubits to be encoded have to be at indices n-k+1:n.\n\ninfo: Encoding circuits are not fault-tolerant\nEncoding circuits are not fault-tolerant, and thus should not be used in practice. Instead, you should measure the stabilizers of the code and the logical observables, thus projecting into the code space (which can be fault-tolerant).\n\nThe canonicalization operation performed on the code may permute the qubits (see canonicalize_gott!). That permutation is corrected for with SWAP gates by default (controlled by the undoperm keyword argument).\n\nBased on (Cleve and Gottesman, 1997) and (Gottesman, 1997), however it seems the published algorithm has some errors. Consult the erratum, as well as the more recent (Grassl, 2002) and (Grassl, 2011), and be aware that this implementation also uses H instead of Z gates.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.naive_syndrome_circuit","page":"API","title":"QuantumClifford.ECC.naive_syndrome_circuit","text":"Generate the non-fault-tolerant stabilizer measurement cicuit for a given code instance or parity check tableau.\n\nUse the ancillary_index and bit_index arguments to offset where the corresponding part the circuit starts.\n\nReturns the circuit, the number of ancillary qubits that were added, and a list of bit indices that will store the measurement results.\n\nSee also: shor_syndrome_circuit\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.parity_checks","page":"API","title":"QuantumClifford.ECC.parity_checks","text":"Parity check tableau of a code.\n\nSee also: parity_checks_x and parity_checks_z\n\n\n\n\n\n","category":"function"},{"location":"ECC_API/#QuantumClifford.ECC.parity_checks_x-Tuple{QuantumClifford.ECC.AbstractECC}","page":"API","title":"QuantumClifford.ECC.parity_checks_x","text":"Parity check boolean matrix of a code (only the X entries in the tableau, i.e. the checks for Z errors).\n\nOnly CSS codes have this method.\n\nSee also: parity_checks\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.parity_checks_z-Tuple{QuantumClifford.ECC.AbstractECC}","page":"API","title":"QuantumClifford.ECC.parity_checks_z","text":"Parity check boolean matrix of a code (only the Z entries in the tableau, i.e. the checks for X errors).\n\nOnly CSS codes have this method.\n\nSee also: parity_checks\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.rate-Tuple{Any}","page":"API","title":"QuantumClifford.ECC.rate","text":"The rate of a code.\n\n\n\n\n\n","category":"method"},{"location":"ECC_API/#QuantumClifford.ECC.shor_syndrome_circuit","page":"API","title":"QuantumClifford.ECC.shor_syndrome_circuit","text":"Generate the Shor fault-tolerant stabilizer measurement cicuit for a given code instance or parity check tableau.\n\nUse the ancillary_index and bit_index arguments to offset where the corresponding part the circuit starts. Ancillary qubits\n\nReturns:\n\nThe ancillary cat state preparation circuit.\nThe Shor syndrome measurement circuit.\nThe number of ancillary qubits that were added.\nThe list of bit indices that store the final measurement results.\n\nSee also: naive_syndrome_circuit\n\n\n\n\n\n","category":"function"},{"location":"allops/#all-operations","page":"All Gates","title":"Operations - Gates, Measurements, and More","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"DocTestSetup = quote\n using QuantumClifford\n using StableRNGs\n rng = StableRNG(42)\nend","category":"page"},{"location":"allops/#Operations","page":"All Gates","title":"Operations","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Acting on quantum states can be performed either:","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"In a \"linear algebra\" language where unitaries, measurements, and other operations have separate interfaces. This is an explicitly deterministic lower-level interface, which provides a great deal of control over how tableaux are manipulated. See the Stabilizer Tableau Algebra Manual as a primer on these approaches.\nOr in a \"circuit\" language, where the operators (and measurements and noise) are represented as circuit gates. This is a higher-level interface in which the outcome of an operation can be stochastic. The API for it is centered around the apply! function. Particularly useful for Monte Carlo simulations and Perturbative Expansion Symbolic Results.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"In the circuit language, all operations can be applied on a state with the apply! function. Whether they are deterministic and their computational complexity is listed in the table below. A list of lower-level \"linear algebra style\" functions for more control over how an operation is performed is also given.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Type Deterministic 𝒪(nˣ) Low-level functions\nAbstractOperation \n├─ AbstractCliffordOperator \n│ ├─ AbstractSymbolicOperator \n│ │ ├─ AbstractSingleQubitOperator \n│ │ │ ├─ SingleQubitOperator ✔️ n \n│ │ │ ├─ sHadamard ✔️ n \n│ │ │ ├─ sId1 ✔️ n \n│ │ │ ├─ sInvPhase ✔️ n \n│ │ │ ├─ sPhase ✔️ n \n│ │ │ ├─ sX ✔️ n \n│ │ │ ├─ sY ✔️ n \n│ │ │ └─ sZ ✔️ n \n│ │ └─ AbstractTwoQubitOperator \n│ │ ├─ sCNOT ✔️ n \n│ │ ├─ sCPHASE ✔️ n \n│ │ └─ sSWAP ✔️ n \n│ │ \n│ ├─ CliffordOperator ✔️ n³ \n│ ├─ PauliOperator ✔️ n² \n│ └─ SparseGate ✔️ kn² \n├─ AbstractMeasurement \n│ ├─ PauliMeasurement ❌ n² project!, projectrand!\n│ ├─ sMX ❌ n² projectX!\n│ ├─ sMY ❌ n² projectY!\n│ └─ sMZ ❌ n² projectZ!\n│ \n├─ BellMeasurement ❌ n² \n├─ NoiseOp ❌ ? applynoise!\n├─ NoiseOpAll ❌ ? applynoise!\n├─ NoisyGate ❌ ? applynoise!\n└─ Reset ✔️ kn² reset_qubits!","category":"page"},{"location":"allops/#Details-of-Operations-Supported-by-[apply!](@ref)","page":"All Gates","title":"Details of Operations Supported by apply!","text":"","category":"section"},{"location":"allops/#Unitary-Gates","page":"All Gates","title":"Unitary Gates","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"We distinguish between symbolic gates like sCNOT that have specialized (fast) apply! methods (usually just for single and two qubit gates) and general tableau representation of gates like CliffordOperator that can represent any multi-qubit gate.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Predefined unitary gates are available, like sCNOT, sHadamard, etc.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\n[sCNOT(2,4),sHadamard(2),sCPHASE(1,3),sSWAP(2,4)]","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Any arbitrary tableaux can be used as a gate too. ","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"They can be specified by giving a Clifford operator tableaux and the indices on which it acts (particularly useful for gates acting on a small part of a circuit):","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\nSparseGate(tCNOT, [2,4])","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"The Clifford operator tableaux can be completely arbitrary.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"SparseGate(random_clifford(3), [2,4,5])","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"If the Clifford operator acts on all qubits, we do not need to specify indices, just use the operator.","category":"page"},{"location":"allops/#Noisy-Gates","page":"All Gates","title":"Noisy Gates","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Each gate can be followed by noise applied to the qubits on which it has acted. This is done by wrapping the given gate into a NoisyGate","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"ε = 0.03 # X/Y/Z error probability\nnoise = UnbiasedUncorrelatedNoise(ε)\nnoisy_gate = NoisyGate(SparseGate(tCNOT, [2,4]), noise)","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"In circuit diagrams the noise is not depicted, but after each application of the gate defined in noisy_gate, a noise operator will also be applied. The example above is of Pauli Depolarization implemented by UnbiasedUncorrelatedNoise.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"One can also apply only the noise operator by using NoiseOp which acts only on specified qubits. Or alternatively, one can use NoiseOpAll in order to apply noise to all qubits.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"[NoiseOp(noise, [4,5]), NoiseOpAll(noise)]","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"The machinery behind noise processes and different types of noise is detailed in the section on noise","category":"page"},{"location":"allops/#Coincidence-Measurements","page":"All Gates","title":"Coincidence Measurements","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"Global parity measurements involving single-qubit projections and classical communication are implemented with BellMeasurement. One needs to specify the axes of measurement and the qubits being measured. If the parity is trivial, the circuit continues, if the parity is non-trivial, the circuit ends and reports a detected failure. This operator is frequently used in the simulation of entanglement purification.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"BellMeasurement([sMX(1), sMY(3), sMZ(4)])","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"There is also NoisyBellMeasurement that takes the bit-flip probability of a single-qubit measurement as a third argument.","category":"page"},{"location":"allops/#Stabilizer-Measurements","page":"All Gates","title":"Stabilizer Measurements","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"A measurement over one or more qubits can also be performed, e.g., a direct stabilizer measurement on multiple qubits without the use of ancillary qubits. When applied to multiple qubits, this differs from BellMeasurement as it performs a single projection, unlike BellMeasurement which performs a separate projection for every single qubit involved. This measurement is implemented in PauliMeasurement which requires a Pauli operator on which to project and the index of the classical bit in which to store the result. Alternatively, there are sMX, sMZ, sMY if you are measuring a single qubit.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"[PauliMeasurement(P\"XYZ\", 1), sMZ(2, 2)]","category":"page"},{"location":"allops/#Reset-Operations","page":"All Gates","title":"Reset Operations","text":"","category":"section"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"The Reset operations lets you trace out the specified qubits and set their state to a specific tableau.","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"new_state = random_stabilizer(3)\nqubit_indices = [1,2,3]\nReset(new_state, qubit_indices)","category":"page"},{"location":"allops/","page":"All Gates","title":"All Gates","text":"It can be done anywhere in a circuit, not just at the beginning.","category":"page"},{"location":"noisycircuits/#Simulation-of-Noisy-Clifford-Circuits","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Clifford Circuits","text":"","category":"section"},{"location":"noisycircuits/","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Circuits","text":"DocTestSetup = quote\n using QuantumClifford\n using QuantumClifford.Experimental.NoisyCircuits\nend\nCurrentModule = QuantumClifford.Experimental.NoisyCircuits","category":"page"},{"location":"noisycircuits/","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Circuits","text":"warning: Unstable\nThis is unfinished experimental functionality that will change significantly.","category":"page"},{"location":"noisycircuits/","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Circuits","text":"We have experimental support for simulation of noisy Clifford circuits which can be imported with using QuantumClifford.Experimental.NoisyCircuits.","category":"page"},{"location":"noisycircuits/","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Circuits","text":"Both Monte Carlo and Perturbative Expansion approaches are supported. When performing a perturbative expansion in the noise parameter, the expansion can optionally be performed symbolically, to arbitrary high orders.","category":"page"},{"location":"noisycircuits/","page":"Simulation of Noisy Circuits","title":"Simulation of Noisy Circuits","text":"Multiple notebooks with examples are also available. For instance, see this tutorial on entanglement purification for many examples.","category":"page"},{"location":"ECC_evaluating/#ecc_evaluating","page":"Evaluating codes and decoders","title":"Evaluating an ECC code and decoders","text":"","category":"section"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"DocTestSetup = quote\n using QuantumClifford\n using QuantumClifford.ECC\nend\nCurrentModule = QuantumClifford.ECC","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"warning: The documentation is incomplete\nWhile waiting for a better documentation than the small example below, consider looking into evaluate_decoder, TableDecoder, BeliefPropDecoder, PyBeliefPropDecoder, PyMatchingDecoder, CommutationCheckECCSetup, NaiveSyndromeECCSetup, ShorSyndromeECCSetup","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"This is a quick and durty example on how to use some of the decoders.","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"A function to plot the results of ","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"using CairoMakie\n\nfunction make_decoder_figure(phys_errors, results, title=\"\")\n minlim = min(minimum(phys_errors),minimum(results[results.!=0]))\n maxlim = min(1, max(maximum(phys_errors),maximum(results[results.!=0])))\n\n fresults = copy(results)\n fresults[results.==0] .= NaN\n\n f = Figure()\n a = Axis(f[1,1],\n xscale=log10, yscale=log10,\n limits=(minlim,maxlim,minlim,maxlim),\n aspect=DataAspect(),\n xlabel=\"physical error rate\",\n ylabel=\"logical error rate\",\n title=title)\n lines!(a, [minlim,maxlim],[minlim,maxlim], color=:black)\n for (i,sresults) in enumerate(eachslice(fresults, dims=1))\n scatter!(a, phys_errors, sresults[:,1], marker=:+, color=Cycled(i))\n scatter!(a, phys_errors, sresults[:,2], marker=:x, color=Cycled(i))\n end\n f\nend","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"Testing out a lookup table decoder on a small code.","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"using QuantumClifford\nusing QuantumClifford.ECC\n\nmem_errors = 0.001:0.0005:0.01\ncodes = [Shor9()]\nresults = zeros(length(codes), length(mem_errors), 2)\n\nfor (ic, c) in pairs(codes)\n for (i,m) in pairs(mem_errors)\n setup = CommutationCheckECCSetup(m)\n decoder = TableDecoder(c)\n r = evaluate_decoder(decoder, setup, 10000)\n results[ic,i,:] .= r\n end\nend\n\nmake_decoder_figure(mem_errors, results, \"Shor's code with a lookup table decoder\")","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"Testing out the toric code with a decoder provided by the python package pymatching (provided in julia by the meta package PyQDecoders.jl).","category":"page"},{"location":"ECC_evaluating/","page":"Evaluating codes and decoders","title":"Evaluating codes and decoders","text":"import PyQDecoders\n\nmem_errors = 0.001:0.005:0.1\ncodes = [Toric(4,4), Toric(6,6)]\nresults = zeros(length(codes), length(mem_errors), 2)\n\nfor (ic, c) in pairs(codes)\n for (i,m) in pairs(mem_errors)\n setup = ShorSyndromeECCSetup(m, 0)\n decoder = PyMatchingDecoder(c)\n r = evaluate_decoder(decoder, setup, 1000)\n results[ic,i,:] .= r\n end\nend\n\nmake_decoder_figure(mem_errors, results, \"Toric code with a MWPM decoder\")","category":"page"},{"location":"canonicalization/#Canonicalization-operations","page":"Canonicalization","title":"Canonicalization operations","text":"","category":"section"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"Different types of canonicalization operations are implemented. All of them are types of Gaussian elimination.","category":"page"},{"location":"canonicalization/#[canonicalize!](@ref)","page":"Canonicalization","title":"canonicalize!","text":"","category":"section"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"First do elimination on all X components and only then perform elimination on the Z components. Based on (Garcia et al., 2012). It is used in logdot for inner products of stabilizer states.","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"The final tableaux, if square should look like the following (Image: )","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"If the tableaux is shorter than a square, the diagonals might not reach all the way to the right.","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"using QuantumClifford, CairoMakie\nf=Figure()\nstabilizerplot_axis(f[1,1], canonicalize!(random_stabilizer(20,30)))\nf","category":"page"},{"location":"canonicalization/#[canonicalize_rref!](@ref)","page":"Canonicalization","title":"canonicalize_rref!","text":"","category":"section"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"Cycle between elimination on X and Z for each qubit. Particularly useful for tracing out qubits. Based on (Audenaert and Plenio, 2005). For convenience reasons, the canonicalization starts from the bottom row, and you can specify as a second argument which columns to be canonicalized (useful for tracing out arbitrary qubits, e.g., in traceout!).","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"The tableau canonicalization is done in recursive steps, each one of which results in something akin to one of these three options (Image: )","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"using QuantumClifford, CairoMakie\nf=Figure()\nstabilizerplot_axis(f[1,1], canonicalize_rref!(random_stabilizer(20,30),1:30)[1])\nf","category":"page"},{"location":"canonicalization/#[canonicalize_gott!](@ref)","page":"Canonicalization","title":"canonicalize_gott!","text":"","category":"section"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"First do elimination on all X components and only then perform elimination on the Z components, but without touching the qubits that were eliminated during the X pass. Unlike other canonicalization operations, qubit columns are reordered, providing for a straight diagonal in each block. Particularly useful as certain blocks of the new created matrix are related to logical operations of the corresponding code, e.g. computing the logical X and Z operators of a MixedDestabilizer. Based on (Gottesman, 1997).","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"A canonicalized tableau would look like the following (the right-most block does not exist for square tableaux). (Image: )","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"using QuantumClifford, CairoMakie\nf=Figure()\nstabilizerplot_axis(f[1,1], canonicalize_gott!(random_stabilizer(30))[1])\nf","category":"page"},{"location":"canonicalization/#[canonicalize_clip!](@ref)","page":"Canonicalization","title":"canonicalize_clip!","text":"","category":"section"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"Convert to the \"clipped\" gauge of a stabilizer state resulting in a \"river\" of non-identity operators around the diagonal.","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"using QuantumClifford, CairoMakie\nf=Figure()\nstabilizerplot_axis(f[1,1], canonicalize_clip!(random_stabilizer(30)))\nf","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"The properties of the clipped gauge are:","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"Each qubit is the left/right \"endpoint\" of exactly two stabilizer rows.\nFor the same qubit the two endpoints are always different Pauli operators.","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"This canonicalization is used to derive the bigram a stabilizer state, which is also related to entanglement entropy in the state.","category":"page"},{"location":"canonicalization/","page":"Canonicalization","title":"Canonicalization","text":"Introduced in (Nahum et al., 2017), with a more detailed explanation of the algorithm in Appendix A of (Li et al., 2019).","category":"page"},{"location":"noisycircuits_mc/#noisycircuits_mc","page":"Monte Carlo","title":"Monte Carlo simulations of noisy Clifford circuits","text":"","category":"section"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"DocTestSetup = quote\n using QuantumClifford\n using QuantumClifford.Experimental.NoisyCircuits\n using Quantikz\nend\nCurrentModule = QuantumClifford.Experimental.NoisyCircuits","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"warning: Unstable\nThis is experimental functionality with an unstable API.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"Import with using QuantumClifford.Experimental.NoisyCircuits.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"This module enables the simulation of noisy Clifford circuits through a Monte Carlo method where the same circuit is evaluated multiple times with random errors interspersed through it as prescribed by a given error model.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"Below is an example of a purification circuit. We first prepare the circuit we desire to use, including a noise model. Quantikz.jl was is used to visualize the circuit.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\ngood_bell_state = S\"XX\n ZZ\"\ninitial_state = MixedDestabilizer(good_bell_state⊗good_bell_state)\n\ng1 = sCNOT(1,3) # CNOT between qubit 1 and qubit 3 (both with Alice)\ng2 = sCNOT(2,4) # CNOT between qubit 2 and qubit 4 (both with Bob)\nm = BellMeasurement([sMX(3),sMX(4)]) # Bell measurement on qubit 3 and 4\nv = VerifyOp(good_bell_state,[1,2]) # Verify that qubit 1 and 2 indeed form a good Bell pair\nepsilon = 0.01 # The error rate\nn = NoiseOpAll(UnbiasedUncorrelatedNoise(epsilon))\n\n# This circuit performs a depolarization at rate `epsilon` to all qubits,\n# then bilater CNOT operations\n# then a Bell measurement\n# followed by checking whether the final result indeed corresponds to the correct Bell pair.\ncircuit = [n,g1,g2,m,v]","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"And we can run a Monte Carlo simulation of that circuit with mctrajectories.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"mctrajectories(initial_state, circuit, trajectories=500)","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"For more examples, see the notebook comparing the Monte Carlo and Perturbative method or this tutorial on entanglement purification for many examples.","category":"page"},{"location":"noisycircuits_mc/#Interface-for-custom-operations","page":"Monte Carlo","title":"Interface for custom operations","text":"","category":"section"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"If you want to create a custom gate type (e.g. calling it Operation), you need to definite the following methods.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"applywstatus!(s::T, g::Operation)::Tuple{T,Symbol} where T is a tableaux type like Stabilizer or a Register. The Symbol is the status of the operation. Predefined statuses are kept in the registered_statuses list, but you can add more. Be sure to expand this list if you want the trajectory simulators using your custom statuses to output all trajectories.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"There is also applynoise! which is convenient wait to create a noise model that can then be plugged into the NoisyGate struct, letting you reuse the predefined perfect gates and measurements. However, you can also just make up your own noise operator simply by implementing applywstatus! for it.","category":"page"},{"location":"noisycircuits_mc/","page":"Monte Carlo","title":"Monte Carlo","text":"You can also consult the list of implemented operators.","category":"page"},{"location":"commonstates/#Useful-States-and-Operators","page":"Useful States","title":"Useful States and Operators","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"DocTestSetup = quote\n using QuantumClifford\n using StableRNGs\n rng = StableRNG(42)\nend","category":"page"},{"location":"commonstates/#States","page":"Useful States","title":"States","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Stabilizer states can be represented with the Stabilizer, Destabilizer, MixedStabilizer, and MixedDestabilizer tableau data structures. You probably want to use MixedDestabilizer which supports the widest set of operations.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Moreover, a MixedDestabilizer can be stored inside a Register together with a set of classical bits in which measurement results can be written.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Below are convenience constructors for common types of states and operators, already implemented in this library.","category":"page"},{"location":"commonstates/#Pauli-Operators","page":"Useful States","title":"Pauli Operators","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Single qubit PauliOperator is implemented in [single_z] and [single_x].","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> single_z(4,2)\n+ _Z__\n\njulia> single_x(4,3)\n+ __X_","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"All identity operators use zero.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> zero(PauliOperator, 3)\n+ ___\n\njulia> zero(P\"XYZXYZ\")\n+ ______","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Random Pauli operators are implemented as well (with or without a random phase).","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> using StableRNGs; rng = StableRNG(42);\n\njulia> random_pauli(rng, 4)\n+ ZYY_\n\njulia> random_pauli(rng, 4; nophase=false)\n- YZ_X","category":"page"},{"location":"commonstates/#Stabilizer-States","page":"Useful States","title":"Stabilizer States","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"An all-identity stabilizer can be created with zero.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> zero(Stabilizer, 3)\n+ ___\n+ ___\n+ ___\n\njulia> zero(Stabilizer, 2, 3)\n+ ___\n+ ___\n\njulia> zero(S\"XIZ\n YZX\")\n+ ___\n+ ___","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Diagonal stabilizers in different bases are available as well, through one.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> one(Stabilizer, 3)\n+ Z__\n+ _Z_\n+ __Z\n\njulia> one(Stabilizer, 3; basis=:Y)\n+ Y__\n+ _Y_\n+ __Y\n\njulia> one(S\"XX\n ZZ\")\n+ Z_\n+ _Z","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"A random stabilizer (or destabilizers or Clifford operators) can be created as well. We use the algorithm described in (Bravyi and Maslov, 2021).","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> random_stabilizer(rng, 2,5)\n+ YZXZZ\n- XZYYY","category":"page"},{"location":"commonstates/#Mixed-States","page":"Useful States","title":"Mixed States","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Similarly, one can create a diagonal mixed state.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> one(MixedDestabilizer, 2, 3)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ X__\n+ _X_\n𝒳ₗ━━━\n+ __X\n𝒮𝓉𝒶𝒷━\n+ Z__\n+ _Z_\n𝒵ₗ━━━\n+ __Z","category":"page"},{"location":"commonstates/#Enumerating-all-Clifford-Operations","page":"Useful States","title":"Enumerating all Clifford Operations","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"The algorithm from (Koenig and Smolin, 2014) can be used to enumerate all Clifford operations on a given number of qubits through enumerate_cliffords. Or one can use random_clifford, random_stabilizer to directly sample from that set.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> length(enumerate_cliffords(1))\n6\n\njulia> length(enumerate_cliffords(2))\n720","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"To also enumerate possible phases, you can use enumerate_phases.","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> length(collect(enumerate_phases(tCNOT)))\n16\n\njulia> length(collect(enumerate_phases(enumerate_cliffords(2))))\n11520","category":"page"},{"location":"commonstates/#Common-entangled-states","page":"Useful States","title":"Common entangled states","text":"","category":"section"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"Bell states and GHZ states have convenience constructors:","category":"page"},{"location":"commonstates/","page":"Useful States","title":"Useful States","text":"julia> bell()\n+ XX\n+ ZZ\n\njulia> bell(2)\n+ XX__\n+ ZZ__\n+ __XX\n+ __ZZ\n\njulia> ghz(4)\n+ XXXX\n+ ZZ__\n+ _ZZ_\n+ __ZZ","category":"page"},{"location":"API/#Full-API","page":"API","title":"Full API","text":"","category":"section"},{"location":"API/","page":"API","title":"API","text":"","category":"page"},{"location":"API/#States","page":"API","title":"States","text":"","category":"section"},{"location":"API/","page":"API","title":"API","text":"Stabilizer states can be represented with the Stabilizer, Destabilizer, MixedStabilizer, and MixedDestabilizer tableau data structures. You probably want to use MixedDestabilizer which supports the widest set of operations.","category":"page"},{"location":"API/","page":"API","title":"API","text":"Moreover, a MixedDestabilizer can be stored inside a Register together with a set of classical bits in which measurement results can be written.","category":"page"},{"location":"API/","page":"API","title":"API","text":"Lastly, for Pauli frame simulations there is the PauliFrame type, a tableau in which each row represents a different Pauli frame.","category":"page"},{"location":"API/","page":"API","title":"API","text":"There are convenience constructors for common types of states and operators.","category":"page"},{"location":"API/#Operations","page":"API","title":"Operations","text":"","category":"section"},{"location":"API/","page":"API","title":"API","text":"Acting on quantum states can be performed either:","category":"page"},{"location":"API/","page":"API","title":"API","text":"In a \"linear algebra\" language where unitaries, measurements, and other operations have separate interfaces. This is an explicitly deterministic lower-level interface, which provides a great deal of control over how tableaux are manipulated. See the Stabilizer Tableau Algebra Manual as a primer on these approaches.\nOr in a \"circuit\" language, where the operators (and measurements and noise) are represented as circuit gates. This is a higher-level interface in which the outcome of an operation can be stochastic. The API for it is centered around the apply! function. Particularly useful for Monte Carlo simulations and Perturbative Expansion Symbolic Results.","category":"page"},{"location":"API/","page":"API","title":"API","text":"See the full list of operations for a list of implemented operations.","category":"page"},{"location":"API/#Autogenerated-API-list","page":"API","title":"Autogenerated API list","text":"","category":"section"},{"location":"API/","page":"API","title":"API","text":"Modules = [QuantumClifford]\nPrivate = false","category":"page"},{"location":"API/#QuantumClifford.QuantumClifford","page":"API","title":"QuantumClifford.QuantumClifford","text":"A module for using the Stabilizer formalism and simulating Clifford circuits.\n\n\n\n\n\n","category":"module"},{"location":"API/#QuantumClifford.continue_stat","page":"API","title":"QuantumClifford.continue_stat","text":"Returned by applywstatus! if the circuit simulation should continue.\n\n\n\n\n\n","category":"constant"},{"location":"API/#QuantumClifford.failure_stat","page":"API","title":"QuantumClifford.failure_stat","text":"Returned by applywstatus! if the circuit reports a failure.\n\nSee also: VerifyOp, BellMeasurement.\n\n\n\n\n\n","category":"constant"},{"location":"API/#QuantumClifford.false_success_stat","page":"API","title":"QuantumClifford.false_success_stat","text":"Returned by applywstatus! if the circuit reports a success, but it is a false positive (i.e., there was an undetected error).\n\nSee also: VerifyOp, BellMeasurement.\n\n\n\n\n\n","category":"constant"},{"location":"API/#QuantumClifford.true_success_stat","page":"API","title":"QuantumClifford.true_success_stat","text":"Returned by applywstatus! if the circuit reports a success and there is no undetected error.\n\nSee also: VerifyOp, BellMeasurement.\n\n\n\n\n\n","category":"constant"},{"location":"API/#QuantumClifford.AbstractSingleQubitOperator","page":"API","title":"QuantumClifford.AbstractSingleQubitOperator","text":"Supertype of all single-qubit symbolic operators.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.AbstractSymbolicOperator","page":"API","title":"QuantumClifford.AbstractSymbolicOperator","text":"Supertype of all symbolic operators. Subtype of AbstractCliffordOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.AbstractTwoQubitOperator","page":"API","title":"QuantumClifford.AbstractTwoQubitOperator","text":"Supertype of all two-qubit symbolic operators.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.BellMeasurement","page":"API","title":"QuantumClifford.BellMeasurement","text":"A Bell measurement performing the correlation measurement corresponding to the given pauli projections on the qubits at the selected indices.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.CircuitStatus","page":"API","title":"QuantumClifford.CircuitStatus","text":"A convenience struct to represent the status of a circuit simulated by mctrajectories\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.ClassicalXOR","page":"API","title":"QuantumClifford.ClassicalXOR","text":"Applies an XOR gate to classical bits. Currently only implemented for functionality with pauli frames.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.CliffordOperator","page":"API","title":"QuantumClifford.CliffordOperator","text":"Clifford Operator specified by the mapping of the basis generators.\n\njulia> tCNOT\nX₁ ⟼ + XX\nX₂ ⟼ + _X\nZ₁ ⟼ + Z_\nZ₂ ⟼ + ZZ\n\njulia> phase_gate = C\"Y\n Z\"\nX₁ ⟼ + Y\nZ₁ ⟼ + Z\n\njulia> stab = S\"XI\n IZ\";\n\n\njulia> entangled = tCNOT*stab\n+ XX\n+ ZZ\n\njulia> CliffordOperator(T\"YY\")\nERROR: DimensionMismatch: Input tableau should be of size 2n×n (top half is the X mappings and the bottom half are the Z mappings).\n[...]\n\nDestabilizer can also be converted.\n\njulia> d = Destabilizer(S\"Y\")\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z\n𝒮𝓉𝒶𝒷\n+ Y\n\njulia> CliffordOperator(d)\nX₁ ⟼ + Z\nZ₁ ⟼ + Y\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Destabilizer","page":"API","title":"QuantumClifford.Destabilizer","text":"A tableau representation of a pure stabilizer state. The tableau tracks the destabilizers as well, for efficient projections. On initialization there are no checks that the provided state is indeed pure. This enables the use of this data structure for mixed stabilizer state, but a better choice would be to use MixedDestabilizer.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.MixedDestabilizer","page":"API","title":"QuantumClifford.MixedDestabilizer","text":"A tableau representation for mixed stabilizer states that keeps track of the destabilizers in order to provide efficient projection operations.\n\nThe rank r of the n-qubit tableau is tracked, either so that it can be used to represent a mixed stabilizer state, or so that it can be used to represent an n-r logical-qubit code over n physical qubits. The \"logical\" operators are tracked as well.\n\nWhen the constructor is called on an incomplete Stabilizer it automatically calculates the destabilizers and logical operators, following chapter 4 of (Gottesman, 1997). Under the hood the conversion uses the canonicalize_gott! canonicalization. That canonicalization permutes the columns of the tableau, but we automatically undo the column permutation in the preparation of a MixedDestabilizer so that qubits are not reindexed. The boolean keyword arguments undoperm and reportperm can be used to control this behavior and to report the permutations explicitly.\n\nSee also: stabilizerview, destabilizerview, logicalxview, logicalzview\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.MixedStabilizer","page":"API","title":"QuantumClifford.MixedStabilizer","text":"A slight improvement of the Stabilizer data structure that enables more naturally and completely the treatment of mixed states, in particular when the project! function is used.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.NoiseOp","page":"API","title":"QuantumClifford.NoiseOp","text":"An operator that applies the given noise model to the qubits at the selected indices.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.NoiseOpAll","page":"API","title":"QuantumClifford.NoiseOpAll","text":"An operator that applies the given noise model to all qubits.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.NoisyGate","page":"API","title":"QuantumClifford.NoisyGate","text":"A gate consisting of the given noise applied after the given perfect Clifford gate.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.PauliChannel","page":"API","title":"QuantumClifford.PauliChannel","text":"A Pauli channel datastructure, mainly for use with StabMixture\n\nSee also: UnitaryPauliChannel\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.PauliFrame","page":"API","title":"QuantumClifford.PauliFrame","text":"struct PauliFrame{T, S} <: QuantumClifford.AbstractQCState\n\nThis is a wrapper around a tableau. This \"frame\" tableau is not to be viewed as a normal stabilizer tableau, although it does conjugate the same under Clifford operations. Each row in the tableau refers to a single frame. The row represents the Pauli operation by which the frame and the reference differ.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.PauliFrame-Tuple{Any, Any, Any}","page":"API","title":"QuantumClifford.PauliFrame","text":"PauliFrame(\n frames,\n qubits,\n measurements\n) -> PauliFrame{Stabilizer{QuantumClifford.Tableau{Vector{UInt8}, LinearAlgebra.Adjoint{UInt64, Matrix{UInt64}}}}}\n\n\nPrepare an empty set of Pauli frames with the given number of frames and qubits. Preallocates spaces for measurement number of measurements.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.PauliMeasurement","page":"API","title":"QuantumClifford.PauliMeasurement","text":"A Stabilizer measurement on the entirety of the quantum register.\n\nprojectrand!(state, pauli) and apply!(state, PauliMeasurement(pauli)) give the same (possibly non-deterministic) result. Particularly useful when acting on Register.\n\nSee also: apply!, projectrand!.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.PauliOperator","page":"API","title":"QuantumClifford.PauliOperator","text":"A multi-qubit Pauli operator (1iIZXY^otimes n).\n\nA Pauli can be constructed with the P custom string macro or by building up one through products and tensor products of smaller operators.\n\njulia> pauli3 = P\"-iXYZ\"\n-iXYZ\n\njulia> pauli4 = 1im * pauli3 ⊗ X\n+ XYZX\n\njulia> Z*X\n+iY\n\nWe use a typical F(2,2) encoding internally. The X and Z bits are stored in a single concatenated padded array of UInt chunks of a bit array.\n\njulia> p = P\"-IZXY\";\n\n\njulia> p.xz\n2-element Vector{UInt64}:\n 0x000000000000000c\n 0x000000000000000a\n\nYou can access the X and Z bits through getters and setters or through the xview, zview, xbit, and zbit functions.\n\njulia> p = P\"XYZ\"; p[1]\n(true, false)\n\njulia> p[1] = (true, true); p\n+ YYZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Register","page":"API","title":"QuantumClifford.Register","text":"A register, representing the state of a computer including both a tableaux and an array of classical bits (e.g. for storing measurement results)\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Reset","page":"API","title":"QuantumClifford.Reset","text":"Reset the specified qubits to the given state.\n\nBe careful, this operation implies first tracing out the qubits, which can lead to mixed states if these qubits were entangled with the rest of the system.\n\nSee also: sMRZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.SingleQubitOperator","page":"API","title":"QuantumClifford.SingleQubitOperator","text":"A \"symbolic\" general single-qubit operator which permits faster multiplication than an operator expressed as an explicit tableau.\n\njulia> op = SingleQubitOperator(2, true, true, true, false, true, true) # Tableau components and phases\nSingleQubitOperator on qubit 2\nX₁ ⟼ - Y\nZ₁ ⟼ - X\n\njulia> typeof(op)\nSingleQubitOperator\n\njulia> t_op = CliffordOperator(op, 3) # Transforming it back into an explicit tableau representation (specifying the size)\nX₁ ⟼ + X__\nX₂ ⟼ - _Y_\nX₃ ⟼ + __X\nZ₁ ⟼ + Z__\nZ₂ ⟼ - _X_\nZ₃ ⟼ + __Z\n\njulia> typeof(t_op)\nCliffordOperator{QuantumClifford.Tableau{Vector{UInt8}, Matrix{UInt64}}}\n\njulia> CliffordOperator(op, 1, compact=true) # You can also extract just the non-trivial part of the tableau\nX₁ ⟼ - Y\nZ₁ ⟼ - X\n\nSee also: sHadamard, sPhase, sId1, sX, sY, sZ, CliffordOperator\n\nOr simply consult subtypes(QuantumClifford.AbstractSingleQubitOperator) and subtypes(QuantumClifford.AbstractTwoQubitOperator) for a full list. You can think of the s prefix as \"symbolic\" or \"sparse\".\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.SparseGate","page":"API","title":"QuantumClifford.SparseGate","text":"A Clifford gate, applying the given cliff operator to the qubits at the selected indices.\n\napply!(state, cliff, indices) and apply!(state, SparseGate(cliff, indices)) give the same result.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.StabMixture","page":"API","title":"QuantumClifford.StabMixture","text":"mutable struct StabMixture{T, F}\n\nRepresents mixture ∑ ϕᵢⱼ Pᵢ ρ Pⱼ† where ρ is a pure stabilizer state.\n\njulia> StabMixture(S\"-X\")\nA mixture ∑ ϕᵢⱼ Pᵢ ρ Pⱼ† where ρ is\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z\n𝒮𝓉𝒶𝒷\n- X\nwith ϕᵢⱼ | Pᵢ | Pⱼ:\n 1.0+0.0im | + _ | + _\n\njulia> pcT\nA unitary Pauli channel P = ∑ ϕᵢ Pᵢ with the following branches:\nwith ϕᵢ | Pᵢ\n 0.853553+0.353553im | + _\n 0.146447-0.353553im | + Z\n\njulia> apply!(StabMixture(S\"-X\"), pcT)\nA mixture ∑ ϕᵢⱼ Pᵢ ρ Pⱼ† where ρ is\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z\n𝒮𝓉𝒶𝒷\n- X\nwith ϕᵢⱼ | Pᵢ | Pⱼ:\n 0.0+0.353553im | + _ | + Z\n 0.0-0.353553im | + Z | + _\n 0.853553+0.0im | + _ | + _\n 0.146447+0.0im | + Z | + Z\n\nSee also: PauliChannel\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Stabilizer","page":"API","title":"QuantumClifford.Stabilizer","text":"Stabilizer, i.e. a list of commuting multi-qubit Hermitian Pauli operators.\n\nInstances can be created with the S custom string macro or as direct sum of other stabilizers.\n\ntip: Stabilizers and Destabilizers\nIn many cases you probably would prefer to use the MixedDestabilizer data structure, as it caries a lot of useful additional information, like tracking rank and destabilizer operators. Stabilizer has mostly a pedagogical value, and it is also used for slightly faster simulation of a particular subset of Clifford operations.\n\njulia> s = S\"XXX\n ZZI\n IZZ\"\n+ XXX\n+ ZZ_\n+ _ZZ\n\njulia> s⊗s\n+ XXX___\n+ ZZ____\n+ _ZZ___\n+ ___XXX\n+ ___ZZ_\n+ ____ZZ\n\nIt has an indexing API, looking like a list of PauliOperators.\n\njulia> s[2]\n+ ZZ_\n\nPauli operators can act directly on the a stabilizer.\n\njulia> P\"YYY\" * s\n- XXX\n+ ZZ_\n+ _ZZ\n\nThere are a number of ways to create a Stabilizer, including:\n\ngenerate Stabilizers from a list of Pauli operators\n\njulia> Stabilizer([P\"XX\", P\"ZZ\"])\n+ XX\n+ ZZ\n\ngenerate Stabilizers from boolean matrices\n\njulia> a = [true true; false false]; b = [false true; true true];\n\njulia> Stabilizer(a, b)\n+ XY\n+ ZZ\n\njulia> Stabilizer([0x0, 0x2], a, b)\n+ XY\n- ZZ\n\ninitialize an empty Stabilizer and fill it through indexing\n\njulia> s = zero(Stabilizer, 2)\n+ __\n+ __\n\njulia> s[1,1] = (true, false); s\n+ X_\n+ __\n\nThere are no automatic checks for correctness (i.e. independence of all rows, commutativity of all rows, hermiticity of all rows). The rank (number of rows) is permitted to be less than the number of qubits (number of columns): canonilization, projection, etc. continue working in that case. To great extent this library uses the Stabilizer data structure simply as a tableau. This might be properly abstracted away in future versions.\n\nSee also: PauliOperator, canonicalize!\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Stabilizer-Tuple{Graphs.SimpleGraphs.SimpleGraph}","page":"API","title":"QuantumClifford.Stabilizer","text":"Convert a graph representing a stabilizer state to an explicit Stabilizer.\n\nSee also: graphstate\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.UnbiasedUncorrelatedNoise","page":"API","title":"QuantumClifford.UnbiasedUncorrelatedNoise","text":"Depolarization noise model with total probability of error p.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.UnitaryPauliChannel","page":"API","title":"QuantumClifford.UnitaryPauliChannel","text":"A Pauli channel datastructure, mainly for use with StabMixture.\n\nMore convenient to use than PauliChannel when you know your Pauli channel is unitary.\n\njulia> Tgate = UnitaryPauliChannel(\n (I, Z),\n ((1+exp(im*π/4))/2, (1-exp(im*π/4))/2)\n )\nA unitary Pauli channel P = ∑ ϕᵢ Pᵢ with the following branches:\nwith ϕᵢ | Pᵢ\n 0.853553+0.353553im | + _\n 0.146447-0.353553im | + Z\n\njulia> PauliChannel(Tgate)\nPauli channel ρ ↦ ∑ ϕᵢⱼ Pᵢ ρ Pⱼ† with the following branches:\nwith ϕᵢⱼ | Pᵢ | Pⱼ:\n 0.853553+0.0im | + _ | + _\n 0.0+0.353553im | + _ | + Z\n 0.0-0.353553im | + Z | + _\n 0.146447+0.0im | + Z | + Z\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.VerifyOp","page":"API","title":"QuantumClifford.VerifyOp","text":"A \"probe\" to verify that the state of the qubits corresponds to a desired good_state, e.g. at the end of the execution of a circuit.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sCNOT","page":"API","title":"QuantumClifford.sCNOT","text":"A \"symbolic\" CNOT. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sCPHASE","page":"API","title":"QuantumClifford.sCPHASE","text":"A \"symbolic\" CPHASE. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sHadamard","page":"API","title":"QuantumClifford.sHadamard","text":"A \"symbolic\" single-qubit Hadamard. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sId1","page":"API","title":"QuantumClifford.sId1","text":"A \"symbolic\" single-qubit Identity operation.\n\nSee also: SingleQubitOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sInvPhase","page":"API","title":"QuantumClifford.sInvPhase","text":"A \"symbolic\" single-qubit InvPhase. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMRX","page":"API","title":"QuantumClifford.sMRX","text":"Measure a qubit in the X basis and reset to the |+⟩ state.\n\nSee also: sMRZ, Reset, sMZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMRY","page":"API","title":"QuantumClifford.sMRY","text":"Measure a qubit in the Y basis and reset to the |i₊⟩ state.\n\nSee also: sMRZ, Reset, sMZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMRZ","page":"API","title":"QuantumClifford.sMRZ","text":"Measure a qubit in the Z basis and reset to the |0⟩ state.\n\nwarning: It does not trace out the qubit!\nAs described below there is a difference between measuring the qubit (followed by setting it to a given known state) and \"tracing out\" the qubit. By reset here we mean \"measuring and setting to a known state\", not \"tracing out\".\n\njulia> s = MixedDestabilizer(S\"XXX ZZI IZZ\") # |000⟩+|111⟩\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ _X_\n+ __X\n𝒮𝓉𝒶𝒷━\n+ XXX\n+ ZZ_\n+ Z_Z\n\njulia> traceout!(copy(s), 1) # = I⊗(|00⟩⟨00| + |11⟩⟨11|)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ _X_\n𝒳ₗ━━━\n+ _XX\n+ Z__\n𝒮𝓉𝒶𝒷━\n+ _ZZ\n𝒵ₗ━━━\n+ Z_Z\n+ XXX\n\njulia> projectZ!(traceout!(copy(s), 1), 1)[1] # = |000⟩⟨000|+|011⟩⟨011| or |100⟩⟨100|+|111⟩⟨111| (use projectZrand! to actually get a random result)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ _X_\n+ XXX\n𝒳ₗ━━━\n+ _XX\n𝒮𝓉𝒶𝒷━\n+ _ZZ\n+ Z__\n𝒵ₗ━━━\n+ Z_Z\n\njulia> projectZ!(copy(s), 1)[1] # = |000⟩ or |111⟩ (use projectZrand! to actually get a random result)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ XXX\n+ _X_\n+ __X\n𝒮𝓉𝒶𝒷━\n+ Z__\n+ ZZ_\n+ Z_Z\n\njulia> apply!(Register(copy(s)), sMRZ(1)) |> quantumstate # |000⟩ or |011⟩, depending on randomization\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ XXX\n+ _X_\n+ __X\n𝒮𝓉𝒶𝒷━\n+ Z__\n- ZZ_\n- Z_Z\n\nSee also: Reset, sMZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMX","page":"API","title":"QuantumClifford.sMX","text":"Symbolic single qubit X measurement. See also Register, projectXrand!, sMY, sMZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMY","page":"API","title":"QuantumClifford.sMY","text":"Symbolic single qubit Y measurement. See also Register, projectYrand!, sMX, sMZ\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sMZ","page":"API","title":"QuantumClifford.sMZ","text":"Symbolic single qubit Z measurement. See also Register, projectZrand!, sMX, sMY\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sPhase","page":"API","title":"QuantumClifford.sPhase","text":"A \"symbolic\" single-qubit Phase. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sSWAP","page":"API","title":"QuantumClifford.sSWAP","text":"A \"symbolic\" SWAP. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sX","page":"API","title":"QuantumClifford.sX","text":"A \"symbolic\" single-qubit X. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sXCX","page":"API","title":"QuantumClifford.sXCX","text":"A \"symbolic\" XCX. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sXCY","page":"API","title":"QuantumClifford.sXCY","text":"A \"symbolic\" XCY. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sXCZ","page":"API","title":"QuantumClifford.sXCZ","text":"A \"symbolic\" XCZ. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sY","page":"API","title":"QuantumClifford.sY","text":"A \"symbolic\" single-qubit Y. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sYCX","page":"API","title":"QuantumClifford.sYCX","text":"A \"symbolic\" YCX. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sYCY","page":"API","title":"QuantumClifford.sYCY","text":"A \"symbolic\" YCY. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sYCZ","page":"API","title":"QuantumClifford.sYCZ","text":"A \"symbolic\" YCZ. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sZ","page":"API","title":"QuantumClifford.sZ","text":"A \"symbolic\" single-qubit Z. See also: SingleQubitOperator, AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sZCX","page":"API","title":"QuantumClifford.sZCX","text":"A \"symbolic\" ZCX. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sZCY","page":"API","title":"QuantumClifford.sZCY","text":"A \"symbolic\" ZCY. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sZCZ","page":"API","title":"QuantumClifford.sZCZ","text":"A \"symbolic\" ZCZ. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.sZCrY","page":"API","title":"QuantumClifford.sZCrY","text":"A \"symbolic\" ZCrY. See also: AbstractSymbolicOperator\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.PauliError","page":"API","title":"QuantumClifford.PauliError","text":"A convenient constructor for various types of Pauli errors, that can be used as circuit gates in simulations. Returns more specific types when necessary.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.PauliError-Tuple{Any, Any}","page":"API","title":"QuantumClifford.PauliError","text":"\"Construct a gate operation that applies an unbiased Pauli error on all qubits, each with independent probability p.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.PauliError-Tuple{Int64, Any}","page":"API","title":"QuantumClifford.PauliError","text":"\"Construct a gate operation that applies an unbiased Pauli error on qubit q with probability p.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.PauliNoise","page":"API","title":"QuantumClifford.PauliNoise","text":"A convenient constructor for various types of Pauli noise models. Returns more specific types when necessary.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.PauliNoise-Tuple{Any}","page":"API","title":"QuantumClifford.PauliNoise","text":"Constructs an unbiased Pauli noise model with total probability of error p.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.affectedqubits","page":"API","title":"QuantumClifford.affectedqubits","text":"A method giving the qubits acted upon by a given operation. Part of the Noise interface.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.applybranches","page":"API","title":"QuantumClifford.applybranches","text":"Compute all possible new states after the application of the given operator. Reports the probability of each one of them. Deterministic (as it reports all branches of potentially random processes), part of the Perturbative Expansion interface.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.applynoise!","page":"API","title":"QuantumClifford.applynoise!","text":"A method modifying a given state by applying the corresponding noise model. It is non-deterministic, part of the Noise interface.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.applywstatus!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.applywstatus!","text":"Used for mctrajectories.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.bell","page":"API","title":"QuantumClifford.bell","text":"Prepare one or more Bell pairs (with optional phases).\n\njulia> bell()\n+ XX\n+ ZZ\n\njulia> bell(2)\n+ XX__\n+ ZZ__\n+ __XX\n+ __ZZ\n\njulia> bell((true, false))\n- XX\n+ ZZ\n\njulia> bell([true, false, true, true])\n- XX__\n+ ZZ__\n- __XX\n- __ZZ\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.bigram-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.bigram","text":"bigram(\n state::QuantumClifford.AbstractStabilizer;\n clip\n) -> Matrix{Int64}\n\n\nGet the bigram of a tableau.\n\nIt is the list of endpoints of a tableau in the clipped gauge.\n\nIf clip=true (the default) the tableau is converted to the clipped gauge in-place before calculating the bigram. Otherwise, the clip gauge conversion is skipped (for cases where the input is already known to be in the correct gauge).\n\nIntroduced in (Nahum et al., 2017), with a more detailed explanation of the algorithm in (Li et al., 2019) and (Gullans et al., 2020).\n\nSee also: canonicalize_clip!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.bitview","page":"API","title":"QuantumClifford.bitview","text":"A view of the classical bits stored with the state\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.canonicalize!-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.canonicalize!","text":"canonicalize!(\n state::QuantumClifford.AbstractStabilizer;\n phases,\n ranks\n) -> Any\n\n\nCanonicalize a stabilizer (in place).\n\nAssumes the input is a valid stabilizer (all operators commute and have real phases). It permits redundant generators and identity generators.\n\njulia> ghz = S\"XXXX\n ZZII\n IZZI\n IIZZ\";\n\n\njulia> canonicalize!(ghz)\n+ XXXX\n+ Z__Z\n+ _Z_Z\n+ __ZZ\n\njulia> canonicalize!(S\"XXXX\n IZZI\n IIZZ\")\n+ XXXX\n+ _Z_Z\n+ __ZZ\n\nNot all rows in the tableau in the next example are independent:\n\njulia> canonicalize!(S\"XXXX\n ZZII\n IZZI\n IZIZ\n IIZZ\")\n+ XXXX\n+ Z__Z\n+ _Z_Z\n+ __ZZ\n+ ____\n\nIn cases of lower rank, more advanced tableau structures might be better. For instance the MixedStabilizer or MixedDestabilizer structures (you can read more about them in the Data Structures section of the documentation).\n\nIf phases=false is set, the canonicalization does not track the phases in the tableau, leading to significant (constant factor) speedup.\n\njulia> s = S\"-ZX\n XZ\"\n- ZX\n+ XZ\n\njulia> canonicalize!(copy(s), phases=false)\n- XZ\n+ ZX\n\njulia> canonicalize!(copy(s))\n+ XZ\n- ZX\n\nIf ranks=true is set, the last pivot indices for the X and Z stage of the canonicalization are returned as well.\n\njulia> s = S\"XXXX\n ZZII\n IZIZ\n ZIIZ\";\n\n\njulia> _, ix, iz = canonicalize!(s, ranks=true); ix, iz\n(1, 3)\n\njulia> s\n+ XXXX\n+ Z__Z\n+ _Z_Z\n+ ____\n\nBased on (Garcia et al., 2012).\n\nSee also: canonicalize_rref!, canonicalize_gott!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.canonicalize_clip!-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.canonicalize_clip!","text":"canonicalize_clip!(\n state::QuantumClifford.AbstractStabilizer;\n phases\n) -> QuantumClifford.AbstractStabilizer\n\n\nFix the clipped gauge of a stabilizer (in place).\n\nAssumes the input is a valid full-rank stabilizer (all operators commute and have real phases).\n\njulia> s = S\"- X_ZX_X\n + XXYZ__\n - YZ_Z_X\n - XZX__Y\n + _Z_Y_Y\n - ____Z_\";\n\n\njulia> canonicalize_clip!(s)\n- X_XY__\n+ YZY___\n+ _XZX__\n- _ZYX_Z\n- __YZ_X\n- ____Z_\n\nIf phases=false is set, the canonicalization does not track the phases in the tableau, leading to a significant speedup.\n\nIntroduced in (Nahum et al., 2017), with a more detailed explanation of the algorithm in Appendix A of (Li et al., 2019)\n\nSee also: canonicalize!, canonicalize_rref!, canonicalize_gott!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.canonicalize_gott!-Tuple{Stabilizer}","page":"API","title":"QuantumClifford.canonicalize_gott!","text":"Inplace Gottesman canonicalization of a tableau.\n\nThis uses different canonical form from canonicalize!. It is used in the computation of the logical X and Z operators of a MixedDestabilizer.\n\nIt returns the (in place) modified state, the indices of the last pivot of both Gaussian elimination steps, and the permutations that have been used to put the X and Z tableaux in standard form.\n\nBased on (Gottesman, 1997).\n\nSee also: canonicalize!, canonicalize_rref!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.canonicalize_rref!-Tuple{QuantumClifford.AbstractStabilizer, Any}","page":"API","title":"QuantumClifford.canonicalize_rref!","text":"canonicalize_rref!(\n state::QuantumClifford.AbstractStabilizer,\n colindices;\n phases\n) -> Tuple{QuantumClifford.AbstractStabilizer, Any}\n\n\nCanonicalize a stabilizer (in place) along only some columns.\n\nThis uses different canonical form from canonicalize!. It also indexes in reverse in order to make its use in traceout! more efficient. Its use in traceout! is its main application.\n\nIt returns the (in place) modified state and the index of the last pivot.\n\nBased on (Audenaert and Plenio, 2005).\n\nSee also: canonicalize!, canonicalize_gott!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.canonicalize_rref!-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.canonicalize_rref!","text":"canonicalize_rref!(\n state::QuantumClifford.AbstractStabilizer;\n phases\n) -> Tuple{QuantumClifford.AbstractStabilizer, Any}\n\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.clifford_cardinality-Tuple{Int64}","page":"API","title":"QuantumClifford.clifford_cardinality","text":"The size of the Clifford group over a given number of qubits, possibly modulo the phases.\n\nFor n qubits, not accounting for phases is 2ⁿⁿΠⱼ₌₁ⁿ(4ʲ-1). There are 4ⁿ different phase configurations.\n\nSee also: enumerate_cliffords.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.comm","page":"API","title":"QuantumClifford.comm","text":"Check whether two operators commute.\n\n0x0 if they commute, 0x1 if they anticommute.\n\njulia> P\"XX\"*P\"ZZ\", P\"ZZ\"*P\"XX\"\n(- YY, - YY)\n\njulia> comm(P\"ZZ\", P\"XX\")\n0x00\n\njulia> comm(P\"IZ\", P\"XX\")\n0x01\n\nSee also: comm!\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.comm!","page":"API","title":"QuantumClifford.comm!","text":"An in-place version of comm, storing its output in the given buffer.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.compactify_circuit-Tuple{Any}","page":"API","title":"QuantumClifford.compactify_circuit","text":"Convert a list of gates to a more optimized \"sum type\" format which permits faster dispatch.\n\nGenerally, this should be called on a circuit before it is used in a simulation.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.destabilizerview-Tuple{Destabilizer}","page":"API","title":"QuantumClifford.destabilizerview","text":"A view of the subtableau corresponding to the destabilizer. See also tab, stabilizerview, logicalxview, logicalzview\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_cliffords-Tuple{Any, Any}","page":"API","title":"QuantumClifford.enumerate_cliffords","text":"Give the i-th n-qubit Clifford operation, where i∈{1..2ⁿⁿΠⱼ₌₁ⁿ(4ʲ-1)}\n\nThe algorithm is detailed in (Koenig and Smolin, 2014).\n\nSee also: symplecticGS, clifford_cardinality.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_cliffords-Tuple{Any}","page":"API","title":"QuantumClifford.enumerate_cliffords","text":"Give all n-qubit Clifford operations.\n\nThe algorithm is detailed in (Koenig and Smolin, 2014).\n\nSee also: symplecticGS, clifford_cardinality.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_phases-Tuple{CliffordOperator}","page":"API","title":"QuantumClifford.enumerate_phases","text":"Given an operator, return all operators that have the same tableau but different phases.\n\njulia> length(collect(enumerate_phases(tCNOT)))\n16\n\nSee also: enumerate_cliffords, clifford_cardinality.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_phases-Tuple{Union{Base.Generator, AbstractVector}}","page":"API","title":"QuantumClifford.enumerate_phases","text":"Given a set of operators, return all operators that have the same tableaux but different phases.\n\njulia> length(collect(enumerate_phases(enumerate_cliffords(2))))\n11520\n\nSee also: enumerate_cliffords, clifford_cardinality.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_single_qubit_gates-Tuple{Any}","page":"API","title":"QuantumClifford.enumerate_single_qubit_gates","text":"Generate a symbolic single-qubit gate given its index. Optionally, set non-trivial phases.\n\nSee also: enumerate_cliffords.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.fastcolumn","page":"API","title":"QuantumClifford.fastcolumn","text":"Convert a tableau to a memory layout that is fast for column operations.\n\nIn this layout a column of the tableau is stored (mostly) contiguously in memory. Due to bitpacking, e.g., packing 64 bits into a single UInt64, the memory layout is not perfectly contiguous, but it is still optimal given that some bitwrangling is required to extract a given bit.\n\nSee also: fastrow\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.fastrow","page":"API","title":"QuantumClifford.fastrow","text":"Convert a tableau to a memory layout that is fast for row operations.\n\nIn this layout a Pauli string (a row of the tableau) is stored contiguously in memory.\n\nSee also: fastrow\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.generate!-Tuple{PauliOperator, Stabilizer}","page":"API","title":"QuantumClifford.generate!","text":"Generate a Pauli operator by using operators from a given the Stabilizer.\n\nIt assumes the stabilizer is already canonicalized. It modifies the Pauli operator in place, generating it in reverse, up to a phase. That phase is left in the modified operator, which should be the identity up to a phase. Returns the new operator and the list of indices denoting the elements of stabilizer that were used for the generation.\n\njulia> ghz = S\"XXXX\n ZZII\n IZZI\n IIZZ\";\n\n\njulia> canonicalize!(ghz)\n+ XXXX\n+ Z__Z\n+ _Z_Z\n+ __ZZ\n\njulia> generate!(P\"-ZIZI\", ghz)\n(- ____, [2, 4])\n\nWhen the Pauli operator can not be generated by the given tableau, nothing is returned.\n\njulia> generate!(P\"XII\",canonicalize!(S\"ZII\")) === nothing\ntrue\n\njulia> generate!(P\"XII\",canonicalize!(S\"XII\")) === nothing\nfalse\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.gf2_H_to_G-Tuple{Any}","page":"API","title":"QuantumClifford.gf2_H_to_G","text":"For a given F(2,2) parity check matrix, return the generator matrix.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.gf2_gausselim!-Tuple{Any}","page":"API","title":"QuantumClifford.gf2_gausselim!","text":"Gaussian elimination over the binary field.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.gf2_invert-Tuple{Any}","page":"API","title":"QuantumClifford.gf2_invert","text":"Invert a binary matrix.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.gf2_isinvertible-Tuple{Any}","page":"API","title":"QuantumClifford.gf2_isinvertible","text":"Check whether a binary matrix is invertible.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.ghz","page":"API","title":"QuantumClifford.ghz","text":"Prepare a GHZ state of n qubits.\n\njulia> ghz()\n+ XXX\n+ ZZ_\n+ _ZZ\n\njulia> ghz(2)\n+ XX\n+ ZZ\n\njulia> ghz(4)\n+ XXXX\n+ ZZ__\n+ _ZZ_\n+ __ZZ\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.graph_gate-NTuple{4, Any}","page":"API","title":"QuantumClifford.graph_gate","text":"A helper function converting the gate indices from graphstate into a Clifford operator.\n\njulia> s = S\" XXX\n YZ_\n -_ZZ\";\n\n\njulia> graph, h_idx, ip_idx, z_idx = graphstate(s);\n\n\njulia> gate = graph_gate(h_idx, ip_idx, z_idx, nqubits(s));\n\n\njulia> apply!(s, gate) # This is now a graph state (notice you need to multiply row 1 by row 2)\n+ YYZ\n+ XZ_\n+ _ZX\n\njulia> canonicalize!(s) == canonicalize!(Stabilizer(graph))\ntrue\n\nSee also: graph_gatesequence\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.graph_gatesequence-Tuple{Vector{Int64}, Vector{Int64}, Vector{Int64}}","page":"API","title":"QuantumClifford.graph_gatesequence","text":"A helper function converting the gate indices from graphstate into a sequence of gates.\n\njulia> s = S\" XXX\n YZ_\n -_ZZ\";\n\n\njulia> graph, h_idx, ip_idx, z_idx = graphstate(s);\n\n\njulia> gates = graph_gatesequence(h_idx, ip_idx, z_idx);\n\n\njulia> for gate in vcat(gates...) apply!(s, gate) end\n\n\njulia> s # This is now a graph state (notice you need to multiply row 1 by row 2)\n+ YYZ\n+ XZ_\n+ _ZX\n\njulia> canonicalize!(s) == canonicalize!(Stabilizer(graph))\ntrue\n\nSee also: graph_gatesequence\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.graphstate!-Tuple{Stabilizer}","page":"API","title":"QuantumClifford.graphstate!","text":"An in-place version of graphstate.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.graphstate-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.graphstate","text":"Convert any stabilizer state to a graph state\n\nGraph states are a special type of entangled stabilizer states that can be represented by a graph. For a graph G=(VE) the corresponding stabilizers are S_v = X_v prod_u N(v) Z_u. Notice that such tableau rows contain only a single X operator. There is a set of single qubit gates that converts any stabilizer state to a graph state.\n\nThis function returns the graph state corresponding to a stabilizer and the gates that might be necessary to convert the stabilizer into a state representable as a graph.\n\nFor a tableau stab you can convert it with:\n\ngraph, hadamard_idx, iphase_idx, flips_idx = graphstate()\n\nwhere graph is the graph representation of stab, and the rest specifies the single-qubit gates converting stab to graph: hadamard_idx are the qubits that require a Hadamard gate (mapping X ↔ Z), iphase_idx are (different) qubits that require an inverse Phase gate (Y → X), and flips_idx are the qubits that require a phase flip (Pauli Z gate), after the previous two sets of gates.\n\njulia> using Graphs\n\njulia> s = S\" XXX\n ZZ_\n -_ZZ\";\n\n\njulia> g, h_idx, ip_idx, z_idx = graphstate(s);\n\n\njulia> collect(edges(g))\n2-element Vector{Graphs.SimpleGraphs.SimpleEdge{Int64}}:\n Edge 1 => 2\n Edge 1 => 3\n\njulia> h_idx\n2-element Vector{Int64}:\n 2\n 3\n\njulia> ip_idx\nInt64[]\n\njulia> z_idx\n1-element Vector{Int64}:\n 3\n\nThe Graphs.jl library provides many graph-theory tools and the MakieGraphs.jl library provides plotting utilities for graphs.\n\nYou can directly call the graph constructor on a stabilizer, if you just want the graph and do not care about the Clifford operation necessary to convert an arbitrary state to a state representable as a graph:\n\njulia> collect(edges( Graph(bell()) ))\n1-element Vector{Graphs.SimpleGraphs.SimpleEdge{Int64}}:\n Edge 1 => 2\n\nFor a version that does not copy the stabilizer, but rather performs transformations in-place, use graphstate!. It would perform canonicalize_gott! on its argument as it finds a way to convert it to a graph state.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.logdot-Tuple{QuantumClifford.AbstractStabilizer, QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumClifford.logdot","text":"Logarithm of the inner product between to Stabilizer states.\n\nIf the result is nothing, the dot inner product is zero. Otherwise the inner product is 2^(-logdot/2).\n\nThe actual inner product can be computed with LinearAlgebra.dot.\n\nBased on (Garcia et al., 2012).\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.logicalxview-Tuple{MixedDestabilizer}","page":"API","title":"QuantumClifford.logicalxview","text":"A view of the subtableau corresponding to the logical X operators. See also tab, stabilizerview, destabilizerview, logicalzview\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.logicalzview-Tuple{MixedDestabilizer}","page":"API","title":"QuantumClifford.logicalzview","text":"A view of the subtableau corresponding to the logical Z operators. See also tab, stabilizerview, destabilizerview, logicalxview\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.mctrajectories-Tuple{Any, Any}","page":"API","title":"QuantumClifford.mctrajectories","text":"Run multiple Monte Carlo trajectories and report the aggregate final statuses of each.\n\nSee also: pftrajectories, petrajectories\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.mctrajectory!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.mctrajectory!","text":"Run a single Monte Carlo sample, starting with (and modifying) state by applying the given circuit. Uses apply! under the hood.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.petrajectories-Tuple{Any, Any}","page":"API","title":"QuantumClifford.petrajectories","text":"Run a perturbative expansion to a given order. This is the main public function for the perturbative expansion approach.\n\nSee also: pftrajectories, mctrajectories\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pfmeasurements-Tuple{PauliFrame}","page":"API","title":"QuantumClifford.pfmeasurements","text":"pfmeasurements(frame::PauliFrame) -> Any\n\n\nReturns the measurement results for each frame in the PauliFrame instance.\n\nwarning: Relative measurements\nThe return measurements are relative to the reference measurements, i.e. they only say whether the reference measurements have been flipped in the given frame.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pfmeasurements-Tuple{Register, PauliFrame}","page":"API","title":"QuantumClifford.pfmeasurements","text":"pfmeasurements(register::Register, frame::PauliFrame) -> Any\n\n\nTakes the references measurements from the given Register and applies the flips as prescribed by the PauliFrame relative measurements. The result is the actual (non-relative) measurement results for each frame.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pfmeasurements-Tuple{Register}","page":"API","title":"QuantumClifford.pfmeasurements","text":"pfmeasurements(register::Register) -> Vector{Bool}\n\n\nReturns the measurements stored in the bits of the given Register.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pftrajectories","page":"API","title":"QuantumClifford.pftrajectories","text":"Perform a \"Pauli frame\" style simulation of a quantum circuit.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.pftrajectories-Tuple{Any}","page":"API","title":"QuantumClifford.pftrajectories","text":"pftrajectories(\n circuit;\n trajectories,\n threads\n) -> PauliFrame{Stabilizer{QuantumClifford.Tableau{Vector{UInt8}, LinearAlgebra.Adjoint{UInt64, Matrix{UInt64}}}}, Matrix{Bool}}\n\n\nThe main method for running Pauli frame simulations of circuits. See the other methods for lower level access.\n\nMultithreading is enabled by default, but can be disabled by setting threads=false. Do not forget to launch Julia with multiple threads enabled, e.g. julia -t4, if you want to use multithreading.\n\nSee also: mctrajectories, petrajectories\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pftrajectories-Tuple{PauliFrame, Any}","page":"API","title":"QuantumClifford.pftrajectories","text":"pftrajectories(state::PauliFrame, circuit) -> PauliFrame\n\n\nEvolve each frame stored in PauliFrame by the given circuit.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pftrajectories-Tuple{Register, Any}","page":"API","title":"QuantumClifford.pftrajectories","text":"pftrajectories(\n register::Register,\n circuit;\n trajectories\n) -> Tuple{Register, PauliFrame{Stabilizer{QuantumClifford.Tableau{Vector{UInt8}, LinearAlgebra.Adjoint{UInt64, Matrix{UInt64}}}}, Matrix{Bool}}}\n\n\nFor a given Register and circuit, simulates the reference circuit acting on the register and then also simulate numerous PauliFrame trajectories. Returns the register and the PauliFrame instance.\n\nUse pfmeasurements to get the measurement results.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.phases-Tuple{QuantumClifford.Tableau}","page":"API","title":"QuantumClifford.phases","text":"The phases of a given tableau. It is a view, i.e. if you modify this array, the original tableau caries these changes.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.prodphase-Union{Tuple{T}, Tuple{AbstractVector{T}, AbstractVector{T}}} where T<:Unsigned","page":"API","title":"QuantumClifford.prodphase","text":"Get the phase of the product of two Pauli operators.\n\nPhase is encoded as F(4) in the low qubits of an UInt8.\n\njulia> P\"ZZZ\"*P\"XXX\"\n-iYYY\n\njulia> prodphase(P\"ZZZ\", P\"XXX\")\n0x03\n\njulia> prodphase(P\"XXX\", P\"ZZZ\")\n0x01\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.projectXrand!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.projectXrand!","text":"projectXrand!(state, qubit) -> Tuple{Register, UInt8}\n\n\nProject qubit of state along the X axis and randomize the phase if necessary.\n\nLower boilerplate version of project!.\n\nSee also: project!, projectX!, projectZrand!, projectYrand!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.projectYrand!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.projectYrand!","text":"projectYrand!(state, qubit) -> Tuple{Register, UInt8}\n\n\nProject qubit of state along the Y axis and randomize the phase if necessary.\n\nLower boilerplate version of project!.\n\nSee also: project!, projectY!, projectXrand!, projectZrand!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.projectZrand!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.projectZrand!","text":"projectZrand!(state, qubit) -> Tuple{Register, UInt8}\n\n\nProject qubit of state along the Z axis and randomize the phase if necessary.\n\nLower boilerplate version of project!.\n\nSee also: project!, projectZ!, projectXrand!, projectYrand!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.projectrand!-Tuple{Any, Any}","page":"API","title":"QuantumClifford.projectrand!","text":"projectrand!(state, pauli) -> Tuple{Register, Any}\n\n\nMeasure pauli operator on state and randomize the phase if necessary.\n\nLower boilerplate version of project!.\n\nSee also: project!, projectXrand!, projectZrand!, projectYrand!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.puttableau!-Union{Tuple{M2}, Tuple{M1}, Tuple{T}, Tuple{V2}, Tuple{V1}, Tuple{B}, Tuple{QuantumClifford.Tableau{V1, M1}, QuantumClifford.Tableau{V2, M2}, Int64, Int64}} where {B, V1, V2, T<:Unsigned, M1<:AbstractMatrix{T}, M2<:AbstractMatrix{T}}","page":"API","title":"QuantumClifford.puttableau!","text":"Put source tableau in target tableau at given row and column. Assumes target location is zeroed out.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.quantumstate","page":"API","title":"QuantumClifford.quantumstate","text":"Only the quantum part of the state (excluding classical bits)\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.random_clifford-Tuple{Random.AbstractRNG, Int64}","page":"API","title":"QuantumClifford.random_clifford","text":"A random Clifford operator generated by the Bravyi-Maslov Algorithm 2 from (Bravyi and Maslov, 2021).\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.random_clifford1-Tuple{Random.AbstractRNG, Any}","page":"API","title":"QuantumClifford.random_clifford1","text":"Random symbolic single-qubit Clifford applied to qubit at index qubit.\n\nSee also: SingleQubitOperator, random_clifford\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.random_destabilizer-Tuple{Random.AbstractRNG, Int64}","page":"API","title":"QuantumClifford.random_destabilizer","text":"A random Stabilizer/Destabilizer tableau generated by the Bravyi-Maslov Algorithm 2 from (Bravyi and Maslov, 2021).\n\nrandom_destabilizer(n) gives a n-qubit tableau of rank n. random_destabilizer(r,n) gives a n-qubit tableau of rank r.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.random_pauli","page":"API","title":"QuantumClifford.random_pauli","text":"A random Pauli operator on n qubits.\n\nUse nophase=false to randomize the phase. Use realphase=false to get operators with phases including ±i.\n\nOptionally, a \"flip\" probability p can be provided specified, in which case each bit is set to I with probability 1-p and to X or Y or Z with probability p. Useful for simulating unbiased Pauli noise.\n\nSee also random_pauli!\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.random_pauli!","page":"API","title":"QuantumClifford.random_pauli!","text":"An in-place version of random_pauli\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.random_stabilizer-Tuple{Random.AbstractRNG, Int64}","page":"API","title":"QuantumClifford.random_stabilizer","text":"A random Stabilizer tableau generated by the Bravyi-Maslov Algorithm 2 from (Bravyi and Maslov, 2021).\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.single_x-Tuple{Any, Any}","page":"API","title":"QuantumClifford.single_x","text":"A multiqubit operator corresponding to all identities except for Pauli X at i. See also: sX, sMX\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.single_y-Tuple{Any, Any}","page":"API","title":"QuantumClifford.single_y","text":"A multiqubit operator corresponding to all identities except for Pauli Y at i. See also: sY, sMY\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.single_z-Tuple{Any, Any}","page":"API","title":"QuantumClifford.single_z","text":"A multiqubit operator corresponding to all identities except for Pauli Z at i. See also: sY, sMY\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.stab_to_gf2-Tuple{QuantumClifford.Tableau}","page":"API","title":"QuantumClifford.stab_to_gf2","text":"The F(2,2) matrix of a given tableau, represented as the concatenation of two binary matrices, one for X and one for Z.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.stabilizerplot","page":"API","title":"QuantumClifford.stabilizerplot","text":"A Makie.jl recipe for pictorial representation of a tableau.\n\nRequires a Makie.jl backend to be loaded, e.g. using CairoMakie.\n\nAlternatively, you can use the Plots.jl plotting ecosystem, e.g. using Plots; plot(S\"XXX ZZZ\").\n\nConsult the documentation for more details on visualization options.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.stabilizerplot_axis","page":"API","title":"QuantumClifford.stabilizerplot_axis","text":"A Makie.jl recipe for pictorial representation of a tableau.\n\nRequires a Makie.jl backend to be loaded, e.g. using CairoMakie.\n\nAlternatively, you can use the Plots.jl plotting ecosystem, e.g. using Plots; plot(S\"XXX ZZZ\").\n\nConsult the documentation for more details on visualization options.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.stabilizerview-Tuple{Stabilizer}","page":"API","title":"QuantumClifford.stabilizerview","text":"A view of the subtableau corresponding to the stabilizer. See also tab, destabilizerview, logicalxview, logicalzview\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.symplecticGS-Tuple{PauliOperator}","page":"API","title":"QuantumClifford.symplecticGS","text":"Perform the Symplectic Gram-Schmidt procedure that gives a Clifford operator canonically related to a given Pauli operator.\n\nThe algorithm is detailed in (Koenig and Smolin, 2014).\n\nSee also: enumerate_cliffords, clifford_cardinality.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.tab-Union{Tuple{Stabilizer{T}}, Tuple{T}} where T","page":"API","title":"QuantumClifford.tab","text":"Extract the underlying tableau structure.\n\njulia> s = S\"X\"\n+ X\n\njulia> tab(s)\n+ X\n\njulia> tab(Destabilizer(s))\n+ Z\n+ X\n\njulia> tab(MixedDestabilizer(s))\n+ Z\n+ X\n\njulia> tab(tHadamard)\n+ Z\n+ X\n\njulia> typeof(tab(tHadamard))\nQuantumClifford.Tableau{Vector{UInt8}, Matrix{UInt64}}\n\nSee also: stabilizerview, destabilizerview, logicalxview, logicalzview\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.xbit-Tuple{PauliOperator}","page":"API","title":"QuantumClifford.xbit","text":"Extract as a new bit array the X part of the UInt array of packed qubits of a given Pauli operator.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.xview-Tuple{PauliOperator}","page":"API","title":"QuantumClifford.xview","text":"Get a view of the X part of the UInt array of packed qubits of a given Pauli operator.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.zbit-Tuple{PauliOperator}","page":"API","title":"QuantumClifford.zbit","text":"Extract as a new bit array the Z part of the UInt array of packed qubits of a given Pauli operator.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.zview-Tuple{PauliOperator}","page":"API","title":"QuantumClifford.zview","text":"Get a view of the Y part of the UInt array of packed qubits of a given Pauli operator.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.apply!","page":"API","title":"QuantumInterface.apply!","text":"In QuantumClifford the apply! function is used to apply any quantum operation to a stabilizer state, including unitary Clifford operations, Pauli measurements, and noise. Thus, this function may result in a random/stochastic result (e.g. with measurements or noise).\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumInterface.embed-Tuple{Int64, Int64, PauliOperator}","page":"API","title":"QuantumInterface.embed","text":"Embed a Pauli operator in a larger Pauli operator.\n\njulia> embed(5, 3, P\"-Y\")\n- __Y__\n\njulia> embed(5, (3,5), P\"-YX\")\n- __Y_X\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.entanglement_entropy","page":"API","title":"QuantumInterface.entanglement_entropy","text":"Get bipartite entanglement entropy of a subsystem\n\nDefined as entropy of the reduced density matrix.\n\nIt can be calculated with multiple different algorithms, the most performant one depending on the particular case.\n\nCurrently implemented are the :clip (clipped gauge), :graph (graph state), and :rref (Gaussian elimination) algorithms. Benchmark your particular case to choose the best one.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumInterface.entanglement_entropy-Tuple{QuantumClifford.AbstractStabilizer, AbstractVector, Val{:graph}}","page":"API","title":"QuantumInterface.entanglement_entropy","text":"Get bipartite entanglement entropy by first converting the state to a graph and computing the rank of the adjacency matrix.\n\nBased on \"Entanglement in graph states and its applications\".\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.entanglement_entropy-Tuple{QuantumClifford.AbstractStabilizer, AbstractVector, Val{:rref}}","page":"API","title":"QuantumInterface.entanglement_entropy","text":"Get bipartite entanglement entropy by converting to RREF form (i.e., partial trace form).\n\nThe state will be partially canonicalized in an RREF form.\n\nSee also: canonicalize_rref!, traceout!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.entanglement_entropy-Tuple{QuantumClifford.AbstractStabilizer, UnitRange, Val{:clip}}","page":"API","title":"QuantumInterface.entanglement_entropy","text":"Get bipartite entanglement entropy of a contiguous subsystem by passing through the clipped gauge.\n\nIf clip=false is set the canonicalization step is skipped, useful if the input state is already in the clipped gauge.\n\nSee also: bigram, canonicalize_clip!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.expect-Tuple{PauliOperator, QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumInterface.expect","text":"expect(p::PauliOperator, st::AbstractStabilizer)\n\nCompute the expectation value of a Pauli operator p on a stabilizer state st. This function will allocate a temporary copy of the stabilizer state st.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.nqubits-Tuple{QuantumClifford.AbstractStabilizer}","page":"API","title":"QuantumInterface.nqubits","text":"The number of qubits of a given state.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.project!-Tuple{Any, PauliOperator}","page":"API","title":"QuantumInterface.project!","text":"project!(\n state,\n pauli::PauliOperator;\n keep_result,\n phases\n) -> Tuple{MixedStabilizer, Any, Any}\n\n\nProject the state of a Stabilizer on the two eigenspaces of a Pauli operator.\n\nAssumes the input is a valid stabilizer. The projection is done inplace on that stabilizer and it does not modify the projection operator.\n\nIt returns\n\na stabilizer that might not be in canonical form\nthe index of the row where the non-commuting operator was (that row is now equal to pauli; its phase is not updated and for a faithful measurement simulation it needs to be randomized by the user)\nand the result of the projection if there was no non-commuting operator (nothing otherwise)\n\nIf keep_result==false that result of the projection in case of anticommutation is not computed, sparing a canonicalization operation. This canonicalization operation is the only one potentially of cubic complexity. The rest of the calculations are of quadratic complexity.\n\nIf you need to measure a single qubit instead of a multiqubit Pauli operator, the faster projectX!, projectY!, and projectZ! are available.\n\nFor less boilerplate and automatic randomization of the phase use projectrand!.\n\nHere is an example of a projection destroying entanglement:\n\njulia> ghz = S\"XXXX\n ZZII\n IZZI\n IIZZ\";\n\n\njulia> canonicalize!(ghz)\n+ XXXX\n+ Z__Z\n+ _Z_Z\n+ __ZZ\n\njulia> state, anticom_index, result = project!(ghz, P\"ZIII\");\n\n\njulia> state\n+ Z___\n+ Z__Z\n+ _Z_Z\n+ __ZZ\n\njulia> canonicalize!(state)\n+ Z___\n+ _Z__\n+ __Z_\n+ ___Z\n\njulia> anticom_index, result\n(1, nothing)\n\nAnd an example of projection consistent with the stabilizer state.\n\njulia> s = S\"ZII\n IXI\n IIY\";\n\n\njulia> canonicalize!(s)\n+ _X_\n+ __Y\n+ Z__\n\njulia> state, anticom_index, result = project!(s, P\"-ZII\");\n\n\njulia> state\n+ _X_\n+ __Y\n+ Z__\n\njulia> anticom_index, result\n(0, 0x02)\n\nWhile not the best choice, Stabilizer can be used for mixed states, simply by providing an incomplete tableau. In that case it is possible to attempt to project on an operator that can not be generated by the provided stabilizer operators. In that case we have anticom_index==rank and result===nothing, where rank is the the new rank of the tableau, one more than the number of rows in the initial tableau. However, if keep_result was set to false, then anticom_index would stay at zero.\n\njulia> s = S\"XZI\n IZI\";\n\n\njulia> project!(s, P\"IIX\")[1]\n+ X__\n+ _Z_\n\nIf we had used MixedStabilizer we would have added the projector to the list of stabilizers.\n\njulia> s = one(MixedStabilizer, 2, 3)\n+ Z__\n+ _Z_\n\njulia> project!(s, P\"IIX\")[1]\n+ Z__\n+ _Z_\n+ __X\n\nHowever, MixedDestabilizer would be an even better choice as it has mathcalO(n^2) complexity instead of the mathcalO(n^3) complexity of *Stabilizer.\n\njulia> s = one(MixedDestabilizer, 2, 3)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ X__\n+ _X_\n𝒳ₗ━━━\n+ __X\n𝒮𝓉𝒶𝒷━\n+ Z__\n+ _Z_\n𝒵ₗ━━━\n+ __Z\n\njulia> project!(s, P\"IIX\")[1]\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ X__\n+ _X_\n+ __Z\n𝒮𝓉𝒶𝒷━\n+ Z__\n+ _Z_\n+ __X\n\nSee the \"Datastructure Choice\" section in the documentation for more details.\n\nSee also: projectX!, projectY!, projectZ!, projectrand!\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.project!-Tuple{MixedStabilizer, PauliOperator}","page":"API","title":"QuantumInterface.project!","text":"project!(\n state::MixedStabilizer,\n pauli::PauliOperator;\n phases\n) -> Tuple{MixedStabilizer, Any, Any}\n\n\nWhen using project! on MixedStabilizer it automates some of the extra steps we encounter when implicitly using the Stabilizer datastructure to represent mixed states. Namely, it helps when the projector is not among the list of stabilizers:\n\njulia> s = S\"XZI\n IZI\";\n\n\njulia> ms = MixedStabilizer(s)\n+ X__\n+ _Z_\n\njulia> project!(ms, P\"IIY\")[1]\n+ X__\n+ _Z_\n+ __Y\n\nSimilarly to project! on Stabilizer, this function has cubic complexity when the Pauli operator commutes with all rows of the tableau. Most of the time it is better to simply use MixedDestabilizer representation.\n\nUnlike other project! methods, this one does not allow for keep_result=false, as the correct rank or anticommutation index can not be calculated without the expensive (cubic) canonicalization operation required by keep_result=true.\n\nSee the \"Datastructure Choice\" section in the documentation for more details.\n\nSee also: projectX!, projectY!, projectZ!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.projectX!-Tuple{MixedDestabilizer, Int64}","page":"API","title":"QuantumInterface.projectX!","text":"Measure a given qubit in the X basis. A faster special-case version of project!.\n\nSee also: project!, projectXrand!, projectY!, projectZ!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.projectY!-Tuple{MixedDestabilizer, Int64}","page":"API","title":"QuantumInterface.projectY!","text":"Measure a given qubit in the Y basis. A faster special-case version of project!.\n\nSee also: project!, projectYrand!, projectX!, projectZ!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.projectZ!-Tuple{MixedDestabilizer, Int64}","page":"API","title":"QuantumInterface.projectZ!","text":"Measure a given qubit in the Z basis. A faster special-case version of project!.\n\nSee also: project!, projectZrand!, projectY!, projectX!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.reset_qubits!-Tuple{MixedDestabilizer, QuantumClifford.AbstractStabilizer, Any}","page":"API","title":"QuantumInterface.reset_qubits!","text":"reset_qubits!(\n s::MixedDestabilizer,\n newstate::QuantumClifford.AbstractStabilizer,\n qubits;\n phases\n) -> MixedDestabilizer\n\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.reset_qubits!-Tuple{MixedStabilizer, Any, Any}","page":"API","title":"QuantumInterface.reset_qubits!","text":"reset_qubits!(\n s::MixedStabilizer,\n newstate,\n qubits;\n phases\n) -> MixedStabilizer\n\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.reset_qubits!-Tuple{Stabilizer, Any, Any}","page":"API","title":"QuantumInterface.reset_qubits!","text":"reset_qubits!(\n s::Stabilizer,\n newstate,\n qubits;\n phases\n) -> Any\n\n\nReset a given set of qubits to be in the state newstate. These qubits are traced out first, which could lead to \"nonlocal\" changes in the tableau.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.tensor","page":"API","title":"QuantumInterface.tensor","text":"Tensor product between operators or tableaux. See also tensor_pow.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumInterface.tensor_pow-Tuple{Union{QuantumClifford.AbstractCliffordOperator, QuantumClifford.AbstractStabilizer}, Any}","page":"API","title":"QuantumInterface.tensor_pow","text":"Repeated tensor product of an operators or a tableau. See also tensor.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.traceout!-Tuple{Stabilizer, Any}","page":"API","title":"QuantumInterface.traceout!","text":"traceout!(s::Stabilizer, qubits; phases, rank) -> Any\n\n\nTrace out a qubit.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumInterface.traceout!-Tuple{Union{MixedDestabilizer, MixedStabilizer}, Any}","page":"API","title":"QuantumInterface.traceout!","text":"traceout!(\n s::Union{MixedDestabilizer, MixedStabilizer},\n qubits;\n phases,\n rank\n) -> Any\n\n\n\n\n\n\n","category":"method"},{"location":"API/#Private-API","page":"API","title":"Private API","text":"","category":"section"},{"location":"API/","page":"API","title":"API","text":"danger: Private Implementation Details\nThese functions are used internally by the library and might be drastically modified or deleted without warning or deprecation.","category":"page"},{"location":"API/","page":"API","title":"API","text":"Modules = [QuantumClifford]\nPrivate = true\nPublic = false","category":"page"},{"location":"API/#QuantumClifford.AbstractMeasurement","page":"API","title":"QuantumClifford.AbstractMeasurement","text":"Supertype of all symbolic single-qubit measurements.\n\n\n\n\n\n","category":"type"},{"location":"API/#QuantumClifford.Tableau","page":"API","title":"QuantumClifford.Tableau","text":"Internal Tableau type for storing a list of Pauli operators in a compact form. No special semantic meaning is attached to this type, it is just a convenient way to store a list of Pauli operators. E.g. it is not used to represent a stabilizer state, or a stabilizer group, or a Clifford circuit.\n\n\n\n\n\n","category":"type"},{"location":"API/#Base.inv-Tuple{CliffordOperator}","page":"API","title":"Base.inv","text":"inv(\n c::CliffordOperator;\n phases\n) -> CliffordOperator{QuantumClifford.Tableau{Vector{UInt8}, Matrix{UInt64}}}\n\n\nInverse of a CliffordOperator\n\n\n\n\n\n","category":"method"},{"location":"API/#Base.permute!-Tuple{QuantumClifford.AbstractStabilizer, AbstractVector}","page":"API","title":"Base.permute!","text":"Permute the qubits (i.e., columns) of the state in place.\n\n\n\n\n\n","category":"method"},{"location":"API/#Base.permute!-Tuple{QuantumClifford.Tableau, AbstractVector}","page":"API","title":"Base.permute!","text":"Permute the qubits (i.e., columns) of the tableau in place.\n\n\n\n\n\n","category":"method"},{"location":"API/#LinearAlgebra.dot-Tuple{QuantumClifford.AbstractStabilizer, QuantumClifford.AbstractStabilizer}","page":"API","title":"LinearAlgebra.dot","text":"The inner product of two Stabilizers.\n\nBased on (Garcia et al., 2012).\n\nSee also: logdot\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._apply_nonthread!-Tuple{QuantumClifford.AbstractStabilizer, CliffordOperator, AbstractVector{Int64}}","page":"API","title":"QuantumClifford._apply_nonthread!","text":"Nonvectorized version of apply! used for unit tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._apply_nonthread!-Tuple{QuantumClifford.AbstractStabilizer, CliffordOperator}","page":"API","title":"QuantumClifford._apply_nonthread!","text":"Nonvectorized version of apply! used for unit tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._mul_left_nonvec!-Union{Tuple{T}, Tuple{AbstractVector{T}, AbstractVector{T}}} where T<:Unsigned","page":"API","title":"QuantumClifford._mul_left_nonvec!","text":"Nonvectorized version of mul_left! used for unit tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._remove_rowcol!-Tuple{MixedDestabilizer, Any, Any}","page":"API","title":"QuantumClifford._remove_rowcol!","text":"Unexported low-level function that removes a row (by shifting all rows up as necessary)\n\nBecause MixedDestabilizer is not mutable we return a new MixedDestabilizer with the same (modified) xzs array.\n\nUsed on its own, this function will break invariants. Meant to be used with projectremove!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._rowmove!-Union{Tuple{B}, Tuple{QuantumClifford.Tableau, Any, Any}} where B","page":"API","title":"QuantumClifford._rowmove!","text":"Unexported low-level function that moves row i to row j.\n\nUsed on its own, this function will break invariants. Meant to be used in _remove_rowcol!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford._stim_prodphase-Union{Tuple{T}, Tuple{AbstractVector{T}, AbstractVector{T}}} where T<:Unsigned","page":"API","title":"QuantumClifford._stim_prodphase","text":"The quantumlib/Stim implementation, which performs the prodphase and mul_left! together. Used for unit tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.apply_single_x!-Tuple{QuantumClifford.AbstractStabilizer, Any}","page":"API","title":"QuantumClifford.apply_single_x!","text":"Apply a Pauli X to the i-th qubit of state s. You should use apply!(stab,sX(i)) instead of this.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.apply_single_y!-Tuple{QuantumClifford.AbstractStabilizer, Any}","page":"API","title":"QuantumClifford.apply_single_y!","text":"Apply a Pauli Y to the i-th qubit of state s. You should use apply!(stab,sY(i)) instead of this.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.apply_single_z!-Tuple{QuantumClifford.AbstractStabilizer, Any}","page":"API","title":"QuantumClifford.apply_single_z!","text":"Apply a Pauli Z to the i-th qubit of state s. You should use apply!(stab,sZ(i)) instead of this.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.applynoise_branches","page":"API","title":"QuantumClifford.applynoise_branches","text":"Compute all possible new states after the application of the given noise model. Reports the probability of each one of them. Deterministic (as it reports all branches of potentially random processes), part of the Noise interface.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.colswap!-Tuple{QuantumClifford.Tableau, Any, Any}","page":"API","title":"QuantumClifford.colswap!","text":"Swap two columns of a stabilizer in place.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.destab_looks_good-Tuple{Any}","page":"API","title":"QuantumClifford.destab_looks_good","text":"Check basic consistency requirements of a destabilizer. Used in tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.enumerate_cliffords_slow-Tuple{Any, Any}","page":"API","title":"QuantumClifford.enumerate_cliffords_slow","text":"The O(n^4) implementation from (Koenig and Smolin, 2014) – their algorithm seems wrong as ⟨w'₁|wₗ⟩=bₗ which is not always zero.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.fill_tril-Tuple{Any, Any, Any}","page":"API","title":"QuantumClifford.fill_tril","text":"Assign (symmetric) random ints to off diagonals of matrix.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.gf2_H_standard_form_indices-Tuple{Any}","page":"API","title":"QuantumClifford.gf2_H_standard_form_indices","text":"The permutation of columns which turns a binary matrix into standard form. It is assumed the matrix has already undergone Gaussian elimination.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.initZ!-Tuple{PauliFrame}","page":"API","title":"QuantumClifford.initZ!","text":"initZ!(frame::PauliFrame) -> PauliFrame\n\n\nInject random Z errors over all frames and qubits for the supplied PauliFrame with probability 0.5.\n\nCalling this after initialization is essential for simulating any non-deterministic circuit. It is done automatically by most PauliFrame constructors.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.make_sumtype-Tuple{Any}","page":"API","title":"QuantumClifford.make_sumtype","text":"julia> make_sumtype([sCNOT])\nquote\n @sum_type CompactifiedGate :hidden begin\n sCNOT(::Int64, ::Int64)\n end\nend\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.make_sumtype_method","page":"API","title":"QuantumClifford.make_sumtype_method","text":"``` julia> makesumtypemethod([sCNOT], :apply!, (:s,)) quote function QuantumClifford.apply!(s, g::CompactifiedGate) @cases g begin sCNOT(q1, q2) => apply!(s, sCNOT(q1, q2)) end end end\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.make_sumtype_variant_constructor-Tuple{Any}","page":"API","title":"QuantumClifford.make_sumtype_variant_constructor","text":"julia> make_sumtype_variant_constructor(sCNOT)\n:(CompactifiedGate(g::sCNOT) = begin\n (CompactifiedGate').sCNOT(g.q1, g.q2)\nend)\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.make_variant-Tuple{DataType}","page":"API","title":"QuantumClifford.make_variant","text":"julia> make_variant(sCNOT)\n:(sCNOT(::Int64, ::Int64))\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.make_variant_deconstruct","page":"API","title":"QuantumClifford.make_variant_deconstruct","text":"julia> make_variant_deconstruct(sCNOT, :apply!, (:s,))\n:(sCNOT(q1, q2) => apply!(s, sCNOT(q1, q2)))\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.mixed_destab_looks_good-Tuple{Any}","page":"API","title":"QuantumClifford.mixed_destab_looks_good","text":"Check basic consistency requirements of a mixed destabilizer. Used in tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.mixed_stab_looks_good-Tuple{Any}","page":"API","title":"QuantumClifford.mixed_stab_looks_good","text":"Check basic consistency requirements of a mixed stabilizer. Used in tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.pfmeasurement","page":"API","title":"QuantumClifford.pfmeasurement","text":"For a given simulated state, e.g. a PauliFrame instance, returns the measurement results.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.precise_inv-Tuple{Any}","page":"API","title":"QuantumClifford.precise_inv","text":"Inverting a binary matrix: uses floating point for small matrices and Nemo for large matrices.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.project_cond!-Union{Tuple{PHASES}, Tuple{RESET}, Tuple{IS}, Tuple{MixedDestabilizer, Int64, Val{IS}, Val{RESET}}} where {IS, RESET, PHASES}","page":"API","title":"QuantumClifford.project_cond!","text":"Internal method used to implement projectX!, projectZ!, and projectY!.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.projectremoverand!-Union{Tuple{F}, Tuple{MixedDestabilizer, F, Any}} where F<:Union{typeof(projectX!), typeof(projectY!), typeof(projectZ!)}","page":"API","title":"QuantumClifford.projectremoverand!","text":"Unexported low-level function that projects a qubit and returns the result while making the tableau smaller by a qubit.\n\nBecause MixedDestabilizer is not mutable we return a new MixedDestabilizer with the same (modified) xzs array.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.quantum_mallows-Tuple{Any, Any}","page":"API","title":"QuantumClifford.quantum_mallows","text":"Sample (h, S) from the distribution P_n(h, S) from Bravyi and Maslov Algorithm 1.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.remove_column!-Union{Tuple{M}, Tuple{T}, Tuple{V}, Tuple{QuantumClifford.Tableau{V, M}, Int64}} where {V, T<:Unsigned, M<:AbstractMatrix{T}}","page":"API","title":"QuantumClifford.remove_column!","text":"Unexported low-level function that removes a column (by shifting all columns to the right of the target by one step to the left)\n\nBecause Tableau is not mutable we return a new Tableau with the same (modified) xzs array.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.rowdecompose-Tuple{Any, Union{Destabilizer, MixedDestabilizer}}","page":"API","title":"QuantumClifford.rowdecompose","text":"Decompose a Pauli P in terms of stabilizer and destabilizer rows from a given tableaux.\n\nFor given tableaux of rows destabilizer rows d_i and stabilizer rows s_i, there are boolean vectors b and c such that P = i^p prod_i d_i^b_i prod_i s_i^c_i.\n\nThis function returns p, b, c.\n\njulia> s = MixedDestabilizer(ghz(2))\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z_\n+ _X\n𝒮𝓉𝒶𝒷\n+ XX\n+ ZZ\n\njulia> phase, destab_rows, stab_rows = QuantumClifford.rowdecompose(P\"XY\", s)\n(3, Bool[1, 0], Bool[1, 1])\n\njulia> im^3 * P\"Z_\" * P\"XX\" * P\"ZZ\"\n+ XY\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.sample_geometric_2-Tuple{Any, Integer}","page":"API","title":"QuantumClifford.sample_geometric_2","text":"This function samples a number from 1 to n where n >= 1 probability of outputting i is proportional to 2^i\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.stab_looks_good-Tuple{Any}","page":"API","title":"QuantumClifford.stab_looks_good","text":"Check basic consistency requirements of a stabilizer. Used in tests.\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.to_cpu","page":"API","title":"QuantumClifford.to_cpu","text":"copies the memory content of the object to CPU\n\nYou can only use this function if CUDA.jl is imported\n\nFor more advanced users to_cpu(data, element_type) will reinterpret elements of data and converts them to element_type. For example based on your CPU architecture, if working with matrices of UInt32 is faster than UInt64, you can use to_cpu(data, UInt32)\n\njulia> using QuantumClifford: to_cpu, to_gpu\n\njulia> using CUDA # without this import, to_cpu, to_gpu are just function\n\njulia> stab = S\"- X_Z\\n+ _ZZ\\n+ __Z\"\n- X_Z\n+ _ZZ\n+ __Z\n\njulia> stab_gpu = to_gpu(stab);\n\njulia> apply!(stab_gpu, sHadamard(1));\n\njulia> stab_result_cpu = to_cpu(stab_gpu)\n- Z_Z\n+ _ZZ\n+ __Z\n\njulia> using QuantumClifford: to_cpu, to_gpu\n\njulia> using CUDA # without this import, to_cpu, to_gpu are just function\n\njulia> pf_gpu = to_gpu(PauliFrame(1000, 2, 2));\njulia> circuit = [sMZ(1, 1), sHadamard(2), sMZ(2, 2)];\njulia> pftrajectories(pf_gpu, circuit);\njulia> measurements = to_cpu(pf_gpu.measurements);\n\nSee also: to_gpu\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.to_gpu","page":"API","title":"QuantumClifford.to_gpu","text":"copies the memory content of the object to GPU\n\nYou can only use this function if CUDA.jl is imported\n\nFor more advanced users to_gpu(data, element_type) will reinterpret elements of data and converts them to element_type. For example based on your GPU architecture, if working with matrices of UInt64 is faster than UInt32, you can use to_gpu(data, UInt64)\n\njulia> using QuantumClifford: to_cpu, to_gpu\n\njulia> using CUDA # without this import, to_cpu, to_gpu are just function\n\njulia> stab = S\"- X_Z\\n+ _ZZ\\n+ __Z\"\n- X_Z\n+ _ZZ\n+ __Z\n\njulia> stab_gpu = to_gpu(stab);\n\njulia> apply!(stab_gpu, sHadamard(1));\n\njulia> stab_result_cpu = to_cpu(stab_gpu)\n- Z_Z\n+ _ZZ\n+ __Z\n\njulia> using QuantumClifford: to_cpu, to_gpu\n\njulia> using CUDA # without this import, to_cpu, to_gpu are just function\n\njulia> pf_gpu = to_gpu(PauliFrame(1000, 2, 2));\njulia> circuit = [sMZ(1, 1), sHadamard(2), sMZ(2, 2)];\njulia> pftrajectories(pf_gpu, circuit);\njulia> measurements = to_cpu(pf_gpu.measurements);\n\nSee also: to_cpu\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.trusted_rank","page":"API","title":"QuantumClifford.trusted_rank","text":"A \"trusted\" rank which returns rank(state) for Mixed[De]Stabilizer and length(state) for [De]Stabilizer.\n\n\n\n\n\n","category":"function"},{"location":"API/#QuantumClifford.zero!-Tuple{QuantumClifford.Tableau, Any}","page":"API","title":"QuantumClifford.zero!","text":"Zero-out a given row of a Tableau\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.zero!-Union{Tuple{PauliOperator{Tz, Tv}}, Tuple{Tv}, Tuple{Tve}, Tuple{Tz}} where {Tz, Tve<:Unsigned, Tv<:AbstractVector{Tve}}","page":"API","title":"QuantumClifford.zero!","text":"Zero-out the phases and single-qubit operators in a PauliOperator\n\n\n\n\n\n","category":"method"},{"location":"API/#QuantumClifford.@qubitop1-Tuple{Any, Any}","page":"API","title":"QuantumClifford.@qubitop1","text":"Macro used to define single qubit symbolic gates and their qubit_kernel methods.\n\n\n\n\n\n","category":"macro"},{"location":"API/#QuantumClifford.@qubitop2-Tuple{Any, Any}","page":"API","title":"QuantumClifford.@qubitop2","text":"Macro used to define 2-qubit symbolic gates and their qubit_kernel methods.\n\n\n\n\n\n","category":"macro"},{"location":"API/#QuantumClifford.@valbooldispatch-Tuple{Any, Vararg{Any}}","page":"API","title":"QuantumClifford.@valbooldispatch","text":"Turns f(Val(x)) into x ? f(Val(true)) : f(Val(false)) in order to avoid dynamic dispatch\n\nSee discourse discussion\n\n\n\n\n\n","category":"macro"},{"location":"mixed/#Mixed-Stabilizer-States","page":"Mixed States","title":"Mixed Stabilizer States","text":"","category":"section"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"DocTestSetup = quote\n using QuantumClifford\nend","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"The Stabilizer and Destabilizer have some support for mixed states (by being initialized with an incomplete list of stabilizer generators), but for most purposes one would use the Mixed* data structures.","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"Mixed stabilizer states are implemented with MixedStabilizer and MixedDestabilizer, the latter of which is the preferred data structure for most tasks as it is much faster by virtue of tracking the destabilizer generators.","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"julia> s = S\"XXX\n IZZ\";\n\njulia> Destabilizer(s)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ _X_\n𝒮𝓉𝒶𝒷━\n+ XXX\n+ _ZZ","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"Unlike Destabilizer, MixedDestabilizer also tracks the logical operation generators.","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"julia> m = MixedDestabilizer(s)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ _X_\n𝒳ₗ━━━\n+ _XX\n𝒮𝓉𝒶𝒷━\n+ XXX\n+ _ZZ\n𝒵ₗ━━━\n+ Z_Z\n\njulia> stabilizerview(m)\n+ XXX\n+ _ZZ\n\njulia> destabilizerview(m)\n+ Z__\n+ _X_\n\njulia> logicalxview(m)\n+ _XX\n\njulia> logicalzview(m)\n+ Z_Z","category":"page"},{"location":"mixed/#Gottesman-Canonicalization","page":"Mixed States","title":"Gottesman Canonicalization","text":"","category":"section"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"To obtain the logical operators we perform a different type of canonicalization, described in Gottesman's thesis and implemented in canonicalize_gott!. Unlike canonicalize! which uses only row operations, canonicalize_gott! performs column swaps as well. MixedDestabilizer undoes those swaps by default when instantiated, but that behavior can be turned off, if you prefer to work with the canonicalized tableau.","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"julia> s = S\"XXX\n ZIZ\";\n\njulia> MixedDestabilizer(s)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ __X\n𝒳ₗ━━━\n+ _X_\n𝒮𝓉𝒶𝒷━\n+ XXX\n+ Z_Z\n𝒵ₗ━━━\n+ ZZ_\n\njulia> MixedDestabilizer(s; undoperm=false)\n𝒟ℯ𝓈𝓉𝒶𝒷\n+ Z__\n+ _X_\n𝒳ₗ━━━\n+ __X\n𝒮𝓉𝒶𝒷━\n+ XXX\n+ ZZ_\n𝒵ₗ━━━\n+ Z_Z","category":"page"},{"location":"mixed/","page":"Mixed States","title":"Mixed States","text":"Destabilizer and MixedStabilizer do not use any column swaps on instantiation as they do not track the logical operators.","category":"page"},{"location":"datastructures/#Data-Structures-Options","page":"Datastructure Choice","title":"Data Structures Options","text":"","category":"section"},{"location":"datastructures/#Choosing-Appropriate-Data-Structure","page":"Datastructure Choice","title":"Choosing Appropriate Tableau Data Structure","text":"","category":"section"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"There are four different data structures used to represent stabilizer states. If you will never need projective measurements you probably would want to use Stabilizer. If you require projective measurements, but only on pure states, Destabilizer should be the appropriate data structure. If mixed stabilizer states are involved, MixedStabilizer would be necessary.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"Stabilizer is simply a list of Pauli operators in a tableau form. As a data structure it does not enforce the requirements for a pure stabilizer state (the rows of the tableau do not necessarily commute, nor are they forced to be Hermitian; the tableau might be underdetermined, redundant, or contradictory). It is up to the user to ensure that the initial values in the tableau are meaningful and consistent.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"canonicalize!, project!, and generate! can accept an under determined (mixed state) Stabilizer instance and operate correctly. canonicalize! can also accept a redundant Stabilizer (i.e. not all rows are independent), leaving as many identity rows at the bottom of the canonicalized tableau as the number of redundant stabilizers in the initial tableau.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"canonicalize! takes mathcalO(n^3) steps. generate! expects a canonicalized input and then takes mathcalO(n^2) steps. project! takes mathcalO(n^3) for projecting on commuting operators due to the need to call canonicalize! and generate!. If the projections is on an anticommuting operator (or if keep_result=false) then it takes mathcalO(n^2) steps.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"MixedStabilizer provides explicit tracking of the rank of the mixed state and works properly when the projection is on a commuting operator not in the stabilizer (see table below for details). Otherwise it has the same performance as Stabilizer.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"The canonicalization can be made unnecessary if we track the destabilizer generators. There are two data structures capable of that.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"Destabilizer stores both the destabilizer and stabilizer states. project! called on it never requires a stabilizer canonicalization, hence it runs in mathcalO(n^2). However, project! will raise an exception if you try to project on a commuting state that is not in the stabilizer as that would be an expensive mathcalO(n^3) operation.","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"MixedDestabilizer tracks both the destabilizer operators and the logical operators in addition to the stabilizer generators. It does not require canonicalization for measurements and its project! operations always takes mathcalO(n^2).","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"For the operation _, anticom_index, result = project!(...) we have the following behavior:","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"projection Stabilizer MixedStabilizer Destabilizer MixedDestabilizer\non anticommuting operator anticom_index>0 result===nothing correct result in mathcalO(n^2) steps same as Stabilizer same as Stabilizer same as Stabilizer\non commuting operator in the stabilizer anticom_index==0 result!==nothing mathcalO(n^3); or mathcalO(n^2) if keep_result=false mathcalO(n^3) mathcalO(n^2) if the state is pure, throws exception otherwise mathcalO(n^2)\non commuting operator out of the stabilizer[1] anticom_index==rank result===nothing mathcalO(n^3), but the user needs to manually include the new operator to the stabilizer; or mathcalO(n^2) if keep_result=false but then result indistinguishable from cell above and anticom_index==0 mathcalO(n^3) and rank goes up by one not applicable if the state is pure, throws exception otherwise mathcalO(n^2) and rank goes up by one","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"Notice the results when the projection operator commutes with the state but is not generated by the stabilizers of the state (the last row of the table). In that case we have _, anticom_index, result = project!(...) where both anticom_index==rank and result===nothing, with rank being the new rank after projection, one more than the number of rows in the tableau before the measurement. ","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"[1]: This can occur only if the state being projected is mixed. Both Stabilizer and Destabilizer can be used for mixed states by simply providing fewer stabilizer generators than qubits at initialization. This can be useful for low-level code that tries to avoid the extra memory cost of using MixedStabilizer and MixedDestabilizer but should be avoided otherwise. project! works correctly or raises an explicit warning on all 4 data structures.","category":"page"},{"location":"datastructures/#Bit-Packing-in-Integers-and-Array-Order","page":"Datastructure Choice","title":"Bit Packing in Integers and Array Order","text":"","category":"section"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"We do not use boolean arrays to store information about the qubits as this would be wasteful (7 out of 8 bits in the boolean would be unused). Instead, we use all 8 qubits in a byte and perform bitwise logical operations as necessary. Implementation details of the object in RAM can matter for performance. The library permits any of the standard UInt types to be used for packing the bits, and larger UInt types (like UInt64) are usually faster as they permit working on 64 qubits at a time (instead of 1 if we used a boolean, or 8 if we used a byte).","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"Moreover, how a tableau is stored in memory can affect performance, as a row-major storage usually permits more efficient use of the CPU cache (for the particular algorithms we use).","category":"page"},{"location":"datastructures/","page":"Datastructure Choice","title":"Datastructure Choice","text":"Both of these parameters are benchmarked (testing the application of a Pauli operator, which is an mathcalO(n^2) operation; and testing the canonicalization of a Stabilizer, which is an mathcalO(n^3) operation). Row-major UInt64 is the best performing and it is used by default in this library.","category":"page"},{"location":"noisycircuits_ops/#noisycircuits_ops","page":"Circuit Operations","title":"Operators in Circuit Simulations","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"DocTestSetup = quote\n using QuantumClifford\n using QuantumClifford.Experimental.NoisyCircuits\n using Quantikz\nend\nCurrentModule = QuantumClifford.Experimental.NoisyCircuits","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"warning: Unstable\nThis is experimental functionality with an unstable API.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Import with using QuantumClifford.Experimental.NoisyCircuits.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Too see a condensed list of all operations check out the API docs.","category":"page"},{"location":"noisycircuits_ops/#Unitary-Gates","page":"Circuit Operations","title":"Unitary Gates","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"We distinguish between symbolic gates like sCNOT that have specialized (fast) apply! methods (usually just for single and two qubit gates) and general tableau representation of gates like CliffordOperator that can represent any multi-qubit gate.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Predefined unitary gates are available, like sCNOT, sHadamard, etc.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\n[sCNOT(2,4),sHadamard(2),sCPHASE(1,3),sSWAP(2,4)]","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Any arbitrary tableaux can be used as a gate too. ","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"They can be specified by giving a Clifford operator tableaux and the indices on which it acts (particularly useful for gates acting on a small part of a circuit):","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"using QuantumClifford # hide\nusing QuantumClifford.Experimental.NoisyCircuits # hide\nusing Quantikz # hide\nSparseGate(tCNOT, [2,4])","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"The Clifford operator tableaux can be completely arbitrary.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"SparseGate(random_clifford(3), [2,4,5])","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"If the Clifford operator acts on all qubits, we do not need to specify indices, just use the operator.","category":"page"},{"location":"noisycircuits_ops/#Noisy-Gates","page":"Circuit Operations","title":"Noisy Gates","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Each gate can be followed by noise applied to the qubits on which it has acted. This is done by wrapping the given gate into a NoisyGate","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"ε = 0.03 # X/Y/Z error probability\nnoise = UnbiasedUncorrelatedNoise(ε)\nnoisy_gate = NoisyGate(SparseGate(tCNOT, [2,4]), noise)","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"In circuit diagrams the noise is not depicted, but after each application of the gate defined in noisy_gate, a noise operator will also be applied. The example above is of Pauli Depolarization implemented by UnbiasedUncorrelatedNoise.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"One can also apply only the noise operator by using NoiseOp which acts only on specified qubits. Or alternatively, one can use NoiseOpAll in order to apply noise to all qubits.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"[NoiseOp(noise, [4,5]), NoiseOpAll(noise)]","category":"page"},{"location":"noisycircuits_ops/#Coincidence-Measurements","page":"Circuit Operations","title":"Coincidence Measurements","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"Global parity measurements involving single-qubit projections and classical communication are implemented with BellMeasurement. One needs to specify the axes of measurement and the qubits being measured. If the parity is trivial, the circuit continues, if the parity is non-trivial, the circuit ends and reports a detected failure. This operator is frequently used in the simulation of entanglement purification.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"BellMeasurement([sMX(1), sMY(3), sMZ(4)])","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"There is also NoisyBellMeasurement that takes the bit-flip probability of a single-qubit measurement as a third argument.","category":"page"},{"location":"noisycircuits_ops/#Stabilizer-Measurements","page":"Circuit Operations","title":"Stabilizer Measurements","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"A measurement over one or more qubits can also be performed, e.g., a direct stabilizer measurement on multiple qubits without the use of ancillary qubits. When applied to multiple qubits, this differs from BellMeasurement as it performs a single projection, unlike BellMeasurement which performs a separate projection for every single qubit involved. This measurement is implemented in PauliMeasurement which requires a Pauli operator on which to project and the index of the classical bit in which to store the result. Alternatively, there are sMX, sMZ, sMY if you are measuring a single qubit.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"[PauliMeasurement(P\"XYZ\", 1), sMZ(2, 2)]","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"TODO: SparseMeasurement, NoisyMeasurement","category":"page"},{"location":"noisycircuits_ops/#Verification-Operations","page":"Circuit Operations","title":"Verification Operations","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"At the end of many circuits one might want to check whether they performed correctly. The VerifyOp operation corresponds to an unphysical perfect tomographic operation, checking whether the state of the qubits at the given indices is indeed what is expected. If it is, the operation reports a success, otherwise it reports an undetected error.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"desired_state = random_stabilizer(5)\nqubit_indices = [1,2,3,4,7]\nVerifyOp(desired_state, qubit_indices)","category":"page"},{"location":"noisycircuits_ops/#Reset-Operations","page":"Circuit Operations","title":"Reset Operations","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"The Reset operations lets you trace out the specified qubits and set their state to a specific tableau.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"new_state = random_stabilizer(3)\nqubit_indices = [1,2,3]\nReset(new_state, qubit_indices)","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"It can be done anywhere in a circuit, not just at the beginning.","category":"page"},{"location":"noisycircuits_ops/#Gates-Conditioned-on-Classical-Bits","page":"Circuit Operations","title":"Gates Conditioned on Classical Bits","text":"","category":"section"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"ConditionalGate is a conditional gate that performs one of two provided gates, depending on the value of a given classical bit.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"DecisionGate is a conditional gate that performs one of the supplied gates, depending on the output of decisionfunction applied to the entire classical bit register.","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"gate1 = SparseGate(tCNOT, [1,2])\ngate2 = sCPHASE(1,2)\ngate3 = SparseGate(tSWAP, [1,3])\ncg = ConditionalGate(gate1, gate2, 2)\ndg = DecisionGate([gate1,gate2,gate3], bit_register->1) # it will always perform gate1\n[sMX(4,1), sMZ(5,2), cg, dg]","category":"page"},{"location":"noisycircuits_ops/","page":"Circuit Operations","title":"Circuit Operations","text":"TODO: Split ConditionalGate into quantum conditional and classical conditional","category":"page"},{"location":"plotting/#Visualizations","page":"Visualizations","title":"Visualizations","text":"","category":"section"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"Stabilizers have a plot recipe that can be used with Plots.jl or Makie.jl. It simply displays the corresponding parity check matrix (extracted with stab_to_gf2) as a bitmap image. Circuits can be visualized with Quantikz.jl.","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"Importing the aforementioned packages together with QuantumClifford is necessary to enable the plotting functionality (implemented as package extensions).","category":"page"},{"location":"plotting/#Plots.jl","page":"Visualizations","title":"Plots.jl","text":"","category":"section"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"In Plots.jl we have a simple recipe plot(s::Stabilizer; xzcomponents=...) where xzcomponents=:split plots the tableau heatmap in a wide form, X bits on the left, Z bits on the right; or xzcomponents=:together plots them overlapping, with different colors for I, X, Z, and Y.","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Plots\nplot(random_stabilizer(20,30), xzcomponents=:split)","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Plots\nplot(canonicalize!(random_stabilizer(20,30)))","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Plots\nplot(canonicalize_gott!(random_stabilizer(30))[1], xzcomponents=:split)","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Plots\nplot(canonicalize_gott!(random_stabilizer(30))[1]; xzcomponents=:together)","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Plots\nplot(canonicalize_rref!(random_stabilizer(20,30),1:30)[1]; xzcomponents=:together)","category":"page"},{"location":"plotting/#Makie.jl","page":"Visualizations","title":"Makie.jl","text":"","category":"section"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"Makie's heatmap can be directly called on Stabilizer.","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, CairoMakie\ns = S\"IIXZ\n ZZIZ\n YYIZ\n IIIZ\n ZZXZ\"\nf, ax, p = CairoMakie.heatmap(s)\nhidedecorations!(ax); hidespines!(ax); # remove ticks and spines\nax.aspect = DataAspect(); # set a one-to-one aspect ratio\nf","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"A full Makie recipe is available as well (supporting xzcomponents)","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, CairoMakie\ns = S\"IIXZ\n ZZIZ\n YYIZ\n IIIZ\n ZZXZ\"\nf, ax, p = stabilizerplot(s, xzcomponents=:together)\nhidedecorations!(ax); hidespines!(ax)\nax.aspect = DataAspect()\nf","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"You can easily add colorbars (and change the colormap) as well:","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, CairoMakie\nfig = Figure()\nax, p = stabilizerplot(fig[1, 1], s, colormap=cgrad(:heat, 4, categorical = true))\nhidedecorations!(ax)\nhidespines!(ax)\nxlims!(ax, 0.5, size(s,2)+0.5) # otherwise there is padding\nylims!(ax, 0.5, size(s,1)+0.5) # otherwise there is padding\n# set the aspect ratio of the plot\nax.aspect = DataAspect()\n# set the aspect ratio of the layout\ncolsize!(fig.layout, 1, Aspect(1, size(s,2)/size(s,1))) \nColorbar(fig[1, 2], p, ticks = (0:3, [\"I\", \"X\", \"Z\", \"Y\"]))\nfig","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"Or set a completely custom set of colors:","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"fig = Figure()\nax, p = stabilizerplot(fig[1, 1], s, colormap=cgrad([:lightgray,RGBf(1,0.4,0.4),RGBf(0.3,1,0.5),RGBf(0.4,0.4,1)], 4, categorical = true))\nhidedecorations!(ax)\nhidespines!(ax)\nxlims!(ax, 0.5, size(s,2)+0.5)\nylims!(ax, 0.5, size(s,1)+0.5)\nax.aspect = DataAspect()\ncolsize!(fig.layout, 1, Aspect(1, size(s,2)/size(s,1))) \nColorbar(fig[2, 1], p, ticks = (0:3, [\"I\", \"X\", \"Z\", \"Y\"]), vertical = false, flipaxis = false)\nfig","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"You might have noticed, Makie recipes do not let you edit the axes or figure, rather they only permit you to set the plot content. Which is why we use hidedecorations!, hidesplines!, and DataAspect to further modify the plot. However, these defaults are also available in stabilizerplot_axis.","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, CairoMakie\nf=Figure()\nstabilizerplot_axis(f[1,1],random_stabilizer(100))\nf","category":"page"},{"location":"plotting/#Quantikz.jl","page":"Visualizations","title":"Quantikz.jl","text":"","category":"section"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"With the Quantikz library you can visualize gates or sequences of gates.","category":"page"},{"location":"plotting/","page":"Visualizations","title":"Visualizations","text":"using QuantumClifford, Quantikz\ncircuit = [sCNOT(1,2), SparseGate(random_clifford(4), [1,4,5,6]), sMZ(4)]","category":"page"},{"location":"#QuantumClifford.jl","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"DocTestSetup = quote\n using QuantumClifford\nend","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"QuantumClifford.jl is a Julia library for simulation of Clifford circuits, which are a subclass of quantum circuits that can be efficiently simulated on a classical computer.","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"This library uses the tableaux formalism[1] with the destabilizer improvements[2]. Pauli frames are supported for faster repeated simulation of noisy circuits. Various symbolic and algebraic tools for manipulating, converting, and visualizing states and circuits are also implemented. ","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"[1]: (Gottesman, 1998)","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"[2]: (Aaronson and Gottesman, 2004)","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The library consists of two main parts: Tools for working with the algebra of Stabilizer Tableaux and tools specifically for efficient Circuit Simulation.","category":"page"},{"location":"#Stabilizer-Tableau-Algebra","page":"QuantumClifford.jl","title":"Stabilizer Tableau Algebra","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The Stabilizer Tableau Algebra component of QuantumClifford.jl efficiently handles pure and mixed stabilizer states of thousands of qubits, along with support for sparse or dense Clifford operations acting upon them. It provides operations such as canonicalization, projection, generation , and partial traces. The code is vectorized and multithreaded, offering fast, in-place, and allocation-free implementations. Tools for conversion to graph states and for visualization of tableaux are available.","category":"page"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"See the Stabilizer Tableau Algebra manual or the curated list of useful functions.","category":"page"},{"location":"#Example-Usage","page":"QuantumClifford.jl","title":"Example Usage","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"julia> using QuantumClifford\n\njulia> P\"X\" * P\"Z\"\n-iY\n\njulia> P\"X\" ⊗ P\"Z\"\n+ XZ\n\njulia> S\"-XX\n +ZZ\"\n- XX\n+ ZZ\n\njulia> tCNOT * S\"-XX\n +ZZ\"\n- X_\n+ _Z","category":"page"},{"location":"#Circuit-Simulation","page":"QuantumClifford.jl","title":"Circuit Simulation","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The circuit simulation component of QuantumClifford.jl enables Monte Carlo (or symbolic) simulations of noisy Clifford circuits. It provides three main simulation methods: mctrajectories, pftrajectories, and petrajectories. These methods offer varying levels of efficiency, accuracy, and insight.","category":"page"},{"location":"#Monte-Carlo-Simulations-with-Stabilizer-Tableaux-(mctrajectories)","page":"QuantumClifford.jl","title":"Monte Carlo Simulations with Stabilizer Tableaux (mctrajectories)","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The mctrajectories method runs Monte Carlo simulations using a Stabilizer tableau representation for the quantum states.","category":"page"},{"location":"#Monte-Carlo-Simulations-with-Pauli-Frames-(pftrajectories)","page":"QuantumClifford.jl","title":"Monte Carlo Simulations with Pauli Frames (pftrajectories)","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The pftrajectories method runs Monte Carlo simulations of Pauli frames over a single reference Stabilizer tableau simulation. This approach is much more efficient but supports a smaller class of circuits.","category":"page"},{"location":"#Symbolic-Depth-First-Traversal-of-Quantum-Trajectories-(petrajectories)","page":"QuantumClifford.jl","title":"Symbolic Depth-First Traversal of Quantum Trajectories (petrajectories)","text":"","category":"section"},{"location":"","page":"QuantumClifford.jl","title":"QuantumClifford.jl","text":"The petrajectories method performs a depth-first traversal of the most probable quantum trajectories, providing a fixed-order approximation of the circuit's behavior. This approach gives symbolic expressions for various figures of merit instead of just a numeric value.","category":"page"}] } diff --git a/dev/stab-algebra-manual/index.html b/dev/stab-algebra-manual/index.html index fa51c2099..651c74937 100644 --- a/dev/stab-algebra-manual/index.html +++ b/dev/stab-algebra-manual/index.html @@ -293,4 +293,4 @@ 𝒮𝓉𝒶𝒷━ + _ZX - _Z_ -- Z_X

Mixed States

Both the Stabilizer and Destabilizer structures have more general forms that enable work with mixed stabilizer states. They are the MixedStabilizer and MixedDestabilizer structures, described in Mixed States. More information that can be seen in the data structures page, which expands upon the algorithms available for each structure.

Random States and Circuits

random_clifford, random_stabilizer, and enumerate_cliffords can be used for the generation of random states.

+- Z_X

Mixed States

Both the Stabilizer and Destabilizer structures have more general forms that enable work with mixed stabilizer states. They are the MixedStabilizer and MixedDestabilizer structures, described in Mixed States. More information that can be seen in the data structures page, which expands upon the algorithms available for each structure.

Random States and Circuits

random_clifford, random_stabilizer, and enumerate_cliffords can be used for the generation of random states.

diff --git a/dev/tutandpub/index.html b/dev/tutandpub/index.html index 0e6334827..a1f66c466 100644 --- a/dev/tutandpub/index.html +++ b/dev/tutandpub/index.html @@ -1,2 +1,2 @@ -Tutorials and Publications · QuantumClifford.jl
+Tutorials and Publications · QuantumClifford.jl