From 767b1142849fe13e46be27a679ebf6ba6c5eafcf Mon Sep 17 00:00:00 2001 From: Anthony Micciche Date: Mon, 4 Sep 2023 00:10:43 -0400 Subject: [PATCH 1/3] initial implementation of naive_encoind_circuit - needs more testing, cleanup, and calculation of B' (that part wasn't necessary for all codes currently available in the library (they all have r2=0)) --- src/ecc/ECC.jl | 143 +++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 143 insertions(+) diff --git a/src/ecc/ECC.jl b/src/ecc/ECC.jl index 0410b5a60..df0052b9b 100644 --- a/src/ecc/ECC.jl +++ b/src/ecc/ECC.jl @@ -2,6 +2,7 @@ module ECC using QuantumClifford using QuantumClifford: AbstractOperation +using LinearAlgebra import QuantumClifford: Stabilizer, MixedDestabilizer abstract type AbstractECC end @@ -398,4 +399,146 @@ include("./bitflipcode.jl") include("./shorcode.jl") include("./steanecode.jl") +"""A pedagogical example of a quantum error correcting [8,3] code used in [cleve1997efficient](@cite).""" +struct Cleve8 <: AbstractECC end + +code_n(c::Cleve8) = 8 + +parity_checks(c::Cleve8) = S"XXXXXXXX + ZZZZZZZZ + XIXIZYZY + XIYZXIYZ + XZIYIYXZ" + +function canonicalize_cleve97(checks::Stabilizer) + d, n = size(checks) + X0 = (checks |> stab_to_gf2)[:,1:n]' + Z0 = (checks |> stab_to_gf2)[:,n+1:2n]' + + # Now let's work on getting X1 and Z1 + b = rank(X0) + r = d-b + k = n-d + + # First let's move the 0 columns to the left: + nullColumns = [] + notNullColumns = [] + for j in 1:d + if count(X0[:,j])==0 # TODO make sure this is condition -> null generator + push!(nullColumns, j) + else + push!(notNullColumns, j) + end + end + + # Reorder the generators/columns so 0 vectors are in front of X0: + column_order = vcat(nullColumns, notNullColumns) + X0_5 = X0[:, column_order] + Z0_5 = Z0[:, column_order] + + # Second let's perform Gaussian elimination to arrive at X1 and Z1 + bank = [] + for j in (r+1):(r+b) + i = 1 + STOP = false + while !STOP + if X0_5[i, j] != 1 + i+=1 + if i > n + print("ERROR") + STOP = true + end + else + STOP = true + end + end + println(i) + push!(bank, i) + for a in (r+1):(r+b) + if a == j + continue + end + if X0_5[i,a] == 1 + X0_5[:,a] = (X0_5[:,j]+X0_5[:,a]).%2 + Z0_5[:,a] = (Z0_5[:,j]+Z0_5[:,a]).%2 + end + end + end + + qubit_order = [] + for i in 1:n + if i ∉ bank + push!(qubit_order, i) + end + end + append!(qubit_order, bank) + println(qubit_order) + + X1 = X0_5[qubit_order, :] + Z1 = Z0_5[qubit_order, :] + + # Now begins the march towards X2 and Z2, starting with B' + r1 = rank(Z1[1:r+k,1:r]) + r2 = r - r1 + println("WARN: If r2 is greater than 0, this will not work. r2: ", r2) + + # TODO implement calculation of B' using the same alogirthm as above + # For the Cleve8 code r2 = 0, so X1= X2 and Z1 = Z2 + X2 = X1; Z2 = Z1; # TODO this is generally wrong - see above comment. + + # Now time for the final steps before arriving at Xstar Zstar + B1 = Z2[1:k,1:r2+r1] + + XI = zeros(Bool, k , k) + for i in 1:k + XI[i,i] = 1 + end + Xs = (vcat(XI, zeros(Bool,r2, k), B1', zeros(Bool, b, k))) + Zs = zeros(Bool, n, k) + + Xstar = hcat(Xs,X2) + Zstar = hcat(Zs,Z2) + return Xstar, Zstar, Stabilizer(X2',Z2') # TODO at some point unreorder the qubits +end + +""" The naive implementation of the encoding circuit by arXiv:quant-ph/9607030 """ +function naive_encoding_circuit(checks::Stabilizer) + d, n = size(checks) + X0 = (checks |> stab_to_gf2)[:,1:n]' + b = rank(X0) + r = d-b + k = n-d + + Xstar, Zstar, standard_tab = canonicalize_cleve97(checks) + println(standard_tab) + + naive_ec = AbstractOperation[] + for i in (r+k+1):(r+k+b) + push!(naive_ec, sHadamard(i)) + end + + for j in 1:n + if count(Xstar[:,j]) == 0 # Secondary generators are ignored + continue + end + if Zstar[j,j]==1 + push!(naive_ec, sZ(j)) + end + for i in 1:n + if i == j + continue + end + if Xstar[i,j] == 1 && Zstar[i,j] == 0 + push!(naive_ec, sZCX(j,i)) + elseif Xstar[i,j] == 0 && Zstar[i,j] == 1 + push!(naive_ec, sZCZ(j,i)) + elseif Xstar[i,j] == 1 && Zstar[i,j] == 1 + push!(naive_ec, sZCY(j,i)) + end + end + end + + return naive_ec, standard_tab +end + end #module From 9f5018419e4c95e09d91bb03639e737fcbd72617 Mon Sep 17 00:00:00 2001 From: Anthony Micciche Date: Mon, 4 Sep 2023 15:03:38 -0400 Subject: [PATCH 2/3] Finished initial implementation of naive_encoding_circuit - Implemented the B' part, and fixed the rank calculations --- src/ecc/ECC.jl | 81 ++++++++++++++++++++++++++++++++++++++++++-------- 1 file changed, 69 insertions(+), 12 deletions(-) diff --git a/src/ecc/ECC.jl b/src/ecc/ECC.jl index df0052b9b..6b9e5d0c6 100644 --- a/src/ecc/ECC.jl +++ b/src/ecc/ECC.jl @@ -3,6 +3,7 @@ module ECC using QuantumClifford using QuantumClifford: AbstractOperation using LinearAlgebra +using Nemo import QuantumClifford: Stabilizer, MixedDestabilizer abstract type AbstractECC end @@ -416,7 +417,9 @@ function canonicalize_cleve97(checks::Stabilizer) Z0 = (checks |> stab_to_gf2)[:,n+1:2n]' # Now let's work on getting X1 and Z1 - b = rank(X0) + Z2field = Nemo.ResidueRing(ZZ, 2) + b = rank(Nemo.matrix(Z2field, X0)) + r = d-b k = n-d @@ -452,7 +455,6 @@ function canonicalize_cleve97(checks::Stabilizer) STOP = true end end - println(i) push!(bank, i) for a in (r+1):(r+b) if a == j @@ -472,22 +474,73 @@ function canonicalize_cleve97(checks::Stabilizer) end end append!(qubit_order, bank) - println(qubit_order) - X1 = X0_5[qubit_order, :] Z1 = Z0_5[qubit_order, :] # Now begins the march towards X2 and Z2, starting with B' - r1 = rank(Z1[1:r+k,1:r]) + r1 = rank(Nemo.matrix(Z2field, Z1[1:r+k,1:r])) r2 = r - r1 - println("WARN: If r2 is greater than 0, this will not work. r2: ", r2) - # TODO implement calculation of B' using the same alogirthm as above - # For the Cleve8 code r2 = 0, so X1= X2 and Z1 = Z2 - X2 = X1; Z2 = Z1; # TODO this is generally wrong - see above comment. + # First let's move the 0 columns of B to the left: + nullColumns = [] + notNullColumns = [] + for j in 1:r + if count(Z1[:,j])==0 # TODO make sure this is condition <-> null generator + push!(nullColumns, j) + else + push!(notNullColumns, j) + end + end + for j in r+1:d + push!(notNullColumns, j) + end + + # Reorder the generators/columns so 0 vectors are in front of B: + column_order = vcat(nullColumns, notNullColumns) + Z1_5 = Z1[:, column_order] + + # Now Gaussian elimination again, this time over B' + bank = [] + for j in (r2+1):(r2+r1) + i = 1 + STOP = false + while !STOP + if Z1_5[i, j] != 1 + i+=1 + if i > n + print("ERROR") + STOP = true + end + else + STOP = true + end + end + push!(bank, i) + for a in (r2+1):(r2+r1) + if a == j + continue + end + if Z1_5[i,a] == 1 + Z1_5[:,a] = (Z1_5[:,j]+Z1_5[:,a]).%2 + end + end + end + + qubit_order = [] + for i in 1:n + if i ∉ bank && i<= k+r + push!(qubit_order, i) + end + end + for i in k+r+1:n # rows not in B + push!(bank, i) + end + append!(qubit_order, bank) + Z2 = Z1_5[qubit_order, :] + X2 = X1[qubit_order, :] # X is unchanged by operations on b, except for reindexing of qubits # Now time for the final steps before arriving at Xstar Zstar - B1 = Z2[1:k,1:r2+r1] + B1 = Z2[1:k,1+r2:r2+r1] XI = zeros(Bool, k , k) for i in 1:k @@ -496,16 +549,20 @@ function canonicalize_cleve97(checks::Stabilizer) Xs = (vcat(XI, zeros(Bool,r2, k), B1', zeros(Bool, b, k))) Zs = zeros(Bool, n, k) + println("k, r2, r1, b: ", k, " ", r2, " ", r1, " ", b) Xstar = hcat(Xs,X2) Zstar = hcat(Zs,Z2) - return Xstar, Zstar, Stabilizer(X2',Z2') # TODO at some point unreorder the qubits + return Xstar, Zstar, Stabilizer(X2',Z2')# TODO at some point unreorder the qubits? Recall they get reordered twice. end """ The naive implementation of the encoding circuit by arXiv:quant-ph/9607030 """ function naive_encoding_circuit(checks::Stabilizer) d, n = size(checks) X0 = (checks |> stab_to_gf2)[:,1:n]' - b = rank(X0) + + Z2field = Nemo.ResidueRing(ZZ, 2) + b = rank(Nemo.matrix(Z2field, X0)) + r = d-b k = n-d From 4174d2a5e5dc0c403d2c908571226d7e72839600 Mon Sep 17 00:00:00 2001 From: Anthony Micciche Date: Sun, 10 Sep 2023 16:01:32 -0400 Subject: [PATCH 3/3] some new codes to test with --- src/ecc/ECC.jl | 23 +++++++--------- src/ecc/cleavecode.jl | 55 +++++++++++++++++++++++++++++++++++++ src/ecc/fiveonethreecode.jl | 9 ++++++ src/ecc/fivetwotwocode.jl | 8 ++++++ 4 files changed, 82 insertions(+), 13 deletions(-) create mode 100644 src/ecc/cleavecode.jl create mode 100644 src/ecc/fiveonethreecode.jl create mode 100644 src/ecc/fivetwotwocode.jl diff --git a/src/ecc/ECC.jl b/src/ecc/ECC.jl index 6b9e5d0c6..903fba255 100644 --- a/src/ecc/ECC.jl +++ b/src/ecc/ECC.jl @@ -399,18 +399,10 @@ end include("./bitflipcode.jl") include("./shorcode.jl") include("./steanecode.jl") +include("./cleavecode.jl") +include("./fiveonethreecode.jl") +include("./fivetwotwocode.jl") -"""A pedagogical example of a quantum error correcting [8,3] code used in [cleve1997efficient](@cite).""" -struct Cleve8 <: AbstractECC end - -code_n(c::Cleve8) = 8 - -parity_checks(c::Cleve8) = S"XXXXXXXX - ZZZZZZZZ - XIXIZYZY - XIYZXIYZ - XZIYIYXZ" - function canonicalize_cleve97(checks::Stabilizer) d, n = size(checks) X0 = (checks |> stab_to_gf2)[:,1:n]' @@ -476,6 +468,8 @@ function canonicalize_cleve97(checks::Stabilizer) append!(qubit_order, bank) X1 = X0_5[qubit_order, :] Z1 = Z0_5[qubit_order, :] + println("Qubit order 1", qubit_order) + checks = checks[:, qubit_order] # Now begins the march towards X2 and Z2, starting with B' r1 = rank(Nemo.matrix(Z2field, Z1[1:r+k,1:r])) @@ -538,6 +532,8 @@ function canonicalize_cleve97(checks::Stabilizer) append!(qubit_order, bank) Z2 = Z1_5[qubit_order, :] X2 = X1[qubit_order, :] # X is unchanged by operations on b, except for reindexing of qubits + println("Qubit order 2", qubit_order) + checks = checks[:, qubit_order] # Now time for the final steps before arriving at Xstar Zstar B1 = Z2[1:k,1+r2:r2+r1] @@ -552,7 +548,9 @@ function canonicalize_cleve97(checks::Stabilizer) println("k, r2, r1, b: ", k, " ", r2, " ", r1, " ", b) Xstar = hcat(Xs,X2) Zstar = hcat(Zs,Z2) - return Xstar, Zstar, Stabilizer(X2',Z2')# TODO at some point unreorder the qubits? Recall they get reordered twice. + + #return X0, X0_5, X1, X2, Xstar, Z0, Z0_5, Z1, Z1_5, Z2, Zstar + return Xstar, Zstar, Stabilizer(X2', Z2')# TODO Return reordered checks or Stab(X2', Z2')? end """ The naive implementation of the encoding circuit by arXiv:quant-ph/9607030 """ @@ -567,7 +565,6 @@ function naive_encoding_circuit(checks::Stabilizer) k = n-d Xstar, Zstar, standard_tab = canonicalize_cleve97(checks) - println(standard_tab) naive_ec = AbstractOperation[] for i in (r+k+1):(r+k+b) diff --git a/src/ecc/cleavecode.jl b/src/ecc/cleavecode.jl new file mode 100644 index 000000000..9006827b0 --- /dev/null +++ b/src/ecc/cleavecode.jl @@ -0,0 +1,55 @@ +"""A pedagogical example of a quantum error correcting [8,3] code used in [cleve1997efficient](@cite).""" +struct Cleve8 <: AbstractECC end + +code_n(c::Cleve8) = 8 +code_k(c::Cleve8) = 3 + +parity_checks(c::Cleve8) = S"XXXXXXXX + ZZZZZZZZ + XIXIZYZY + XIYZXIYZ + XZIYIYXZ" + +function encoding_circuit(c::Cleve8) + c1 = sCNOT(1,4) + c2 = sCNOT(2,4) + c3 = sCNOT(3,4) + h1 = sHadamard(5) + z1 = sZ(5) + h2 = sHadamard(6) + h3 = sHadamard(7) + z3 = sZ(7) + h4 = sHadamard(8) + z4 = sZ(8) + first_part = [c1,c2,c3,h1,z1,h2,h3,z3,h4,z4] + + c1 = sZCX(5, 1) + c2 = sZCX(5, 2) + c3 = sZCY(5, 3) + c4 = sZCZ(5, 4) + c5 = sZCZ(5, 6) + column1 = [c1,c2,c3,c4,c5] # 1st non null column of Zstar + + c1 = sZCX(6, 1) + c2 = sZCY(6, 2) + c3 = sZCY(6, 4) + c4 = sZCZ(6, 5) + c5 = sZCZ(6, 7) + column2 = [c1,c2,c3,c4,c5] + + c1 = sZCX(7, 1) + c2 = sZCY(7, 3) + c3 = sZCX(7, 4) + c4 = sZCZ(7, 5) + c5 = sZCZ(7, 8) + column3 = [c1,c2,c3,c4,c5] + + c1 = sZCY(8, 2) + c2 = sZCX(8, 3) + c3 = sZCX(8, 4) + c4 = sZCZ(8, 5) + c5 = sZCZ(8, 6) + column4 = [c1,c2,c3,c4,c5] + + return vcat(first_part, column1, column2, column3, column4) +end \ No newline at end of file diff --git a/src/ecc/fiveonethreecode.jl b/src/ecc/fiveonethreecode.jl new file mode 100644 index 000000000..6b2d089d3 --- /dev/null +++ b/src/ecc/fiveonethreecode.jl @@ -0,0 +1,9 @@ +"""http://www.codetables.de/QECC.php?q=4&n=5&k=1""" +struct FiveOneThree <: AbstractECC end + +code_n(c::FiveOneThree) = 8 + +parity_checks(c::FiveOneThree) = S"YIZXY + ZXIZY + ZIXYZ + IZZZZ" \ No newline at end of file diff --git a/src/ecc/fivetwotwocode.jl b/src/ecc/fivetwotwocode.jl new file mode 100644 index 000000000..dc36721de --- /dev/null +++ b/src/ecc/fivetwotwocode.jl @@ -0,0 +1,8 @@ +"""http://www.codetables.de/QECC.php?q=4&n=5&k=2""" +struct FiveTwoTwo <: AbstractECC end + +code_n(c::FiveTwoTwo) = 8 + +parity_checks(c::FiveTwoTwo) = S"XXXXI + IIIIX + ZZZZI" \ No newline at end of file