-
Notifications
You must be signed in to change notification settings - Fork 27
/
Twirl.m
141 lines (118 loc) · 5.24 KB
/
Twirl.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
%% TWIRL Twirls a bipartite or multipartite operator
% This function has one required argument:
% X: a matrix
%
% TX = Twirl(X) is the Werner twirl of the bipartite operator X.
%
% This function has two optional arguments:
% TYPE (default 'werner')
% P (default 2)
%
% TX = Twirl(X,TYPE,P) is the twirl of the P-partite operator X. TYPE
% must be one of 'werner', 'isotropic', 'real', or 'pauli'. A 'werner'
% twirl is a twirl over all complex unitaries of the form U^{\otimes P},
% an 'isotropic' twirl is a twirl over all complex unitaries of the form
% U \otimes conj(U), a 'real' twirl is a twirl over all real orthogonal
% matrices of the form O^{\otimes P}, and a 'pauli' twirl is a twirl over
% all matrices of the form Q \otimes Q, where Q is a Pauli matrix acting
% on one fewer qubit than X. Note that P = 2 is required if TYPE ==
% 'isotropic' or TYPE == 'pauli'.
%
% URL: http://www.qetlab.com/Twirl
% requires: AntisymmetricProjection.m, BrauerStates.m, iden.m,
% MaxEntangled.m,, opt_args.m, Pauli.m, perm_sign.m,
% perfect_matchings.m, PermutationOperator.m, PermuteSystems.m,
% sporth.m, SymmetricProjection.m, Tensor.m
%
% author: Nathaniel Johnston (nathaniel@njohnston.ca)
% package: QETLAB
% last updated: November 27, 2014
function TX = Twirl(X,varargin)
lX = length(X);
% set optional argument defaults: type='werner', p=2
[type,p] = opt_args({ 'werner', 2 },varargin{:});
if((strcmpi(type,'isotropic') == 1 || strcmpi(type,'pauli') == 1) && p > 2)
error('Twirl:InvalidP','Values of P > 2 are not supported for isotropic or Pauli twirling.');
elseif(strcmpi(type,'pauli') == 1)
[log_dim,~] = log2(lX);
if(abs(log_dim - 0.5) > 0.000001) % log_dim is 0.5 if lX is a power of 2, larger than 0.5 otherwise
error('Twirl:InvalidDim','For Pauli twirling, X must act on qubits (i.e., its dimensions must be powers of 2).');
end
end
d = round(lX^(1/p));
% The Werner twirl is slightly complicated because I'm not aware of a
% general formula for the multipartite case. In the bipartite case it's a
% simple linear combination of the symmetric and antisymmetric projections,
% but in the multipartite case you have to do more work to figure out what
% the coefficients of the permutation operators are.
if(strcmpi(type,'werner'))
% First, construct the operators that Werner twirling projects down
% onto (i.e., the p! permutation operators).
p_fac = factorial(p);
per = perms(1:p);
for j = p_fac:-1:1
ProjOps{j} = PermutationOperator(d*ones(1,p),per(j,:),0,1)';
end
% Now project down.
TX = ProjectOntoOperators(X,ProjOps);
% Computing the isotropic twirl of a bipartite state is easy: we have an
% explicit formula.
elseif(strcmpi(type,'isotropic'))
v = MaxEntangled(d,1);
a = v'*X*v;
TX = a*(v*v') + (trace(X)-a)*(speye(d^2)-v*v')/(d^2-1);
% Computing the real orthogonal twirl of a bipartite state is similarly
% easy: we could just plug into a formula. The multipartite case is a bit
% messier, but we just implement this more general procedure for all p.
elseif(strcmpi(type,'real'))
% First, construct the operators that real orthogonal twirling
% projects down onto.
B = BrauerStates(d,p);
szBlocal = sqrt(size(B,1));
for j = size(B,2):-1:1
ProjOps{j} = reshape(B(:,j),szBlocal,szBlocal);
end
% Now project down.
TX = ProjectOntoOperators(X,ProjOps);
% Computing the Puali twirl isn't too tough either: just project onto
% vectorized Pauli matrices.
elseif(strcmpi(type,'pauli'))
% First, construct the vectorized Pauli operators that Pauli
% twirling projects down onto.
num_qubits = round(log2(d));
% Loop over all Pauli operators.
for j = 4^num_qubits:-1:1
bitind = bitget(j-1,1:2*num_qubits);
ind = bitind(1:2:2*num_qubits) + 2*bitind(2:2:2*num_qubits);
ProjOps{j} = reshape(Pauli(ind,1),4^num_qubits,1);
ProjOps{j} = ProjOps{j}*ProjOps{j}';
end
% Now project down.
TX = ProjectOntoOperators(X,ProjOps);
% If the user tried to do a twirl that we don't understand, get cranky.
else
error('Twirl:InvalidType','TYPE must be one of ''werner'', ''isotropic'', ''real'', or ''pauli''.');
end
end
% This function projects the operator X onto the space spanned by the (not
% necessarily orthogonal) operators in the cell ProjOps.
function TX = ProjectOntoOperators(X,ProjOps)
numops = length(ProjOps);
lX = length(X);
coeff_matrix = zeros(numops);
tr_vec = zeros(numops,1);
% Start by creating a coefficient matrix.
for j = 1:numops
for k = 1:numops
coeff_matrix(j,k) = trace(ProjOps{j}'*ProjOps{k});
end
tr_vec(j) = trace(ProjOps{j}'*X);
end
% Now solve the system of equations.
coeffs = pinv(coeff_matrix)*tr_vec; % more stable than using coeff_matrix\tr_vec
% Finally, just sum everything up.
TX = sparse(lX,lX);
for j = 1:numops
TX = TX + coeffs(j)*ProjOps{j};
end
end