-
Notifications
You must be signed in to change notification settings - Fork 56
/
RPETools.py
695 lines (585 loc) · 30.1 KB
/
RPETools.py
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
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
import sys
import GST
import numpy.random as random
import numpy as np
from GSTCommons import MakeLists_WholeGermPowers
import matplotlib
import GST.ComputeReportables
from scipy import optimize
#%matplotlib inline
class myGate():
def __init__(self,inputArray):
if inputArray.shape[0] != inputArray.shape[1]:
raise ValueError("Gate matrix must be square!")
self.dim = inputArray.shape[0]
self.matrix = inputArray
def copy(self):
return myGate(self.matrix)
def makeParamterizedRPEGateSet(alphaTrue,epsilonTrue,Yrot,SPAMdepol,gateDepol=None,withId = True):
"""
Make a gateset for simulating RPE, paramaterized by rotation angles. Note that output gateset also has thetaTrue, alphaTrue, and epsilonTrue attributes.
Parameters
----------
alphaTrue : Angle of Z rotation (canonical RPE requires alphaTrue to be close to pi/2).
epsilonTrue : Angle of X rotation (canonical RPE requires epsilonTrue to be close to pi/4).
Yrot : Angle of rotation about Y axis that, by similarity transformation, rotates X rotation.
SPAMdepol : Amount to depolarize SPAM by.
gateDepol : Amount to depolarize gates by (defaults to None).
withId : Do we include (perfect) identity or no identity? (Defaults to False; should be False for RPE, True for GST)
Returns
-------
outputGateset
The desired gateset for RPE; gateset also has attributes thetaTrue, alphaTrue, and epsilonTrue, automatically extracted.
"""
if withId:
outputGateset = GST.buildGateset( [2], [('Q0',)],['Gi','Gx','Gz'],
[ "I(Q0)", "X("+str(epsilonTrue)+",Q0)", "Z("+str(alphaTrue)+",Q0)"],
rhoExpressions=["0"], EExpressions=["1"],
spamLabelDict={'plus': (0,0), 'minus': (0,-1) })
else:
outputGateset = GST.buildGateset( [2], [('Q0',)],['Gx','Gz'],
[ "X("+str(epsilonTrue)+",Q0)", "Z("+str(alphaTrue)+",Q0)"],
rhoExpressions=["0"], EExpressions=["1"],
spamLabelDict={'plus': (0,0), 'minus': (0,-1) })
if Yrot != 0:
gatesetAux1 = GST.buildGateset( [2], [('Q0',)],['Gi','Gy','Gz'],
[ "I(Q0)", "Y("+str(Yrot)+",Q0)", "Z(pi/2,Q0)"],
rhoExpressions=["0"], EExpressions=["1"],
spamLabelDict={'plus': (0,0), 'minus': (0,-1) })
outputGateset.set_gate('Gx',GST.Gate.FullyParameterizedGate(np.dot(np.dot(np.linalg.inv(gatesetAux1['Gy']),outputGateset['Gx']),gatesetAux1['Gy'])))
# myGate(np.dot(np.dot(np.linalg.inv(gatesetAux1['Gy']),outputGateset['Gx']),gatesetAux1['Gy'])))
outputGateset = GST.GateSetTools.depolarizeSPAM(outputGateset,noise=SPAMdepol)
if gateDepol:
outputGateset = GST.GateSetTools.depolarizeGateset(outputGateset,noise=gateDepol)
thetaTrue = extractTheta(outputGateset)
outputGateset.thetaTrue = thetaTrue
outputGateset.alphaTrue = extractAlpha(outputGateset)
outputGateset.alphaTrue = alphaTrue
outputGateset.epsilonTrue = extractEpsilon(outputGateset)
outputGateset.epsilonTrue = epsilonTrue
return outputGateset
def makeAlphaStrListsGxGz(kList):
"""
Make alpha cosine and sine gatestring lists for (approx) X pi/4 and Z pi/2 gates.
These gate strings are used to estimate alpha (Z rotation angle).
Parameters
----------
kList : The list of "germ powers" to be used. Typically successive powers of two;
e.g. [1,2,4,8,16].
Returns
-------
cosStrList
The list of "cosine strings" to be used for alpha estimation.
sinStrList
The list of "sine strings" to be used for alpha estimation.
"""
cosStrList = []
sinStrList = []
for k in kList:
cosStrList += [GST.gatestring.GateString(('Gi','Gx','Gx','Gz')+('Gz',)*k + ('Gz','Gz','Gz','Gx','Gx'),
'GiGxGxGzGz^'+str(k)+'GzGzGzGxGx')]
sinStrList += [GST.gatestring.GateString(('Gx','Gx','Gz','Gz')+('Gz',)*k + ('Gz','Gz','Gz','Gx','Gx'),
'GxGxGzGzGz^'+str(k)+'GzGzGzGxGx')]
return cosStrList, sinStrList
def makeEpsilonStrListsGxGz(kList):
"""
Make epsilon cosine and sine gatestring lists for (approx) X pi/4 and Z pi/2 gates.
These gate strings are used to estimate epsilon (X rotation angle).
Parameters
----------
kList : The list of "germ powers" to be used. Typically successive powers of two;
e.g. [1,2,4,8,16].
Returns
-------
epsilonCosStrList
The list of "cosine strings" to be used for epsilon estimation.
epsilonSinStrList
The list of "sine strings" to be used for epsilon estimation.
"""
epsilonCosStrList = []
epsilonSinStrList = []
for k in kList:
epsilonCosStrList += [GST.gatestring.GateString(('Gx',)*k+('Gx',)*4,
'Gx^'+str(k)+'GxGxGxGx')]
epsilonSinStrList += [GST.gatestring.GateString(('Gx','Gx','Gz','Gz')+('Gx',)*k+('Gx',)*4,
'GxGxGzGzGx^'+str(k)+'GxGxGxGx')]
return epsilonCosStrList, epsilonSinStrList
def makeThetaStrListsGxGz(kList):
"""
Make theta cosine and sine gatestring lists for (approx) X pi/4 and Z pi/2 gates.
These gate strings are used to estimate theta (X-Z axes angle).
Parameters
----------
kList : The list of "germ powers" to be used. Typically successive powers of two;
e.g. [1,2,4,8,16].
Returns
-------
thetaCosStrList
The list of "cosine strings" to be used for theta estimation.
thetaSinStrList
The list of "sine strings" to be used for theta estimation.
"""
thetaCosStrList = []
thetaSinStrList = []
for k in kList:
thetaCosStrList += [GST.gatestring.GateString(('Gz','Gx','Gx','Gx','Gx','Gz','Gz','Gx','Gx','Gx','Gx','Gz')*k+('Gx',)*4,
'GzGxGxGxGxGzGzGxGxGxGxGz^'+str(k)+'GxGxGxGx')]
thetaSinStrList += [GST.gatestring.GateString(('Gx','Gx','Gz','Gz')+('Gz','Gx','Gx','Gx','Gx','Gz','Gz','Gx','Gx','Gx','Gx','Gz')*k+('Gx',)*4,
'(GzGxGxGxGxGzGzGxGxGxGxGz)^'+str(k)+'GxGxGxGx')]
return thetaCosStrList, thetaSinStrList
def makeRPEStringListD(log2kMax):
"""
Generates a dictionary that contains gate strings for all RPE cosine and sine experiments for all three angles.
Parameters
----------
log2kMax : Maximum number of times to repeat an RPE "germ"
Returns
-------
totalStrListD
A dictionary containing all gate strings for all sine and cosine experiments for alpha, epsilon, and theta.
The keys of the returned dictionary are:
- 'alpha','cos' : List of gate strings for cosine experiments used to determine alpha.
- 'alpha','sin' : List of gate strings for sine experiments used to determine alpha.
- 'epsilon','cos' : List of gate strings for cosine experiments used to determine epsilon.
- 'epsilon','sin' : List of gate strings for sine experiments used to determine epsilon.
- 'theta','cos' : List of gate strings for cosine experiments used to determine theta.
- 'theta','sin' : List of gate strings for sine experiments used to determine theta.
- 'totalStrList' : All above gate strings combined into one list; duplicates removed.
"""
kList = [2**k for k in range(log2kMax+1)]
alphaCosStrList, alphaSinStrList = makeAlphaStrListsGxGz(kList)
epsilonCosStrList, epsilonSinStrList = makeEpsilonStrListsGxGz(kList)
thetaCosStrList, thetaSinStrList = makeThetaStrListsGxGz(kList)
totalStrList = alphaCosStrList + alphaSinStrList + epsilonCosStrList + epsilonSinStrList + thetaCosStrList + thetaSinStrList
totalStrList = GST.ListTools.remove_duplicates(totalStrList)#This step is probably superfluous.
stringListD = {}
stringListD['alpha','cos'] = alphaCosStrList
stringListD['alpha','sin'] = alphaSinStrList
stringListD['epsilon','cos'] = epsilonCosStrList
stringListD['epsilon','sin'] = epsilonSinStrList
stringListD['theta','cos'] = thetaCosStrList
stringListD['theta','sin'] = thetaSinStrList
stringListD['totalStrList'] = totalStrList
return stringListD
def makeRPEDataSet(gatesetOrDataset,stringListD,nSamples,sampleError='binomial',seed=None):
"""
Generate a fake RPE DataSet using the probabilities obtained from a gateset.. Is a thin wrapper for GST.generateFakeData, changing default behavior of sampleError,
and taking a dictionary of gate strings as input.
Parameters
----------
gatesetOrDataset : GateSet or DataSet object
If a GateSet, the gate set whose probabilities generate the data.
If a DataSet, the data set whose frequencies generate the data.
stringListD : Dictionary of list of (tuples or GateStrings)
Each tuple or GateString contains gate labels and
specifies a gate sequence whose counts are included
in the returned DataSet. The dictionary must have the key 'totalStrList';
easiest if this dictionary is generated by makeRPEStringListD.
nSamples : int or list of ints or None
The simulated number of samples for each gate string. This only
has effect when sampleError == "binomial" or "multinomial". If
an integer, all gate strings have this number of total samples. If
a list, integer elements specify the number of samples for the
corresponding gate string. If None, then gatesetOrDataset must be
a DataSet, and total counts are taken from it (on a per-gatestring
basis).
sampleError : string, optional
What type of sample error is included in the counts. Can be:
- "none" - no sampl error:
counts are floating point numbers such that the exact probabilty
can be found by the ratio of count / total.
- "round" - same as "none", except counts are rounded to the nearest integer.
- "binomial" - the number of counts is taken from a binomial distribution.
Distribution has parameters p = probability of the gate string
and n = number of samples. This can only be used when there
are exactly two SPAM labels in gatesetOrDataset.
- "multinomial" - counts are taken from a multinomial distribution.
Distribution has parameters p_k = probability of the
gate string using the k-th SPAM label and n = number
of samples. This should not be used for RPE.
seed : int, optional
If not None, a seed for numpy's random number generator, which
is used to sample from the binomial or multinomial distribution.
Returns
-------
DataSet
A static data set filled with counts for the specified gate strings.
"""
simDS = GST.generateFakeData(gatesetOrDataset,stringListD['totalStrList'],nSamples,sampleError=sampleError,seed=seed)
return simDS
def extractRotationHat(xhat,yhat,k,Nx,Ny,previousAngle=None):
"""
For a single germ generation (k value), estimate the angle of rotation for either
alpha, epsilon, or Phi. (Warning: Do not use for theta estimate without further processing!)
Parameters
----------
xhat : The number of plus counts for the sin string being used.
yhat : The number of plus counts for the cos string being used.
k : The generation of experiments that xhat and yhat come from.
Nx : The number of sin string clicks.
Ny : The number cos string clicks.
previousAngle : Angle estimate from previous generation; used to refine this generation's estimate. Default is None (for estimation with no previous genereation's data)
Returns
-------
alpha_j
The current angle estimate.
"""
if k!=1 and previousAngle == None:
raise Exception('Need previousAngle!')
if k == 1:
return np.arctan2((xhat-Nx/2.)/Nx,(yhat-Ny/2.)/Ny)
elif k>1:
angle_j = 1./k * np.arctan2((xhat-Nx/2.)/Nx,(yhat-Ny/2.)/Ny)
while not (angle_j > previousAngle - np.pi/k and angle_j <= previousAngle + np.pi/k):
if angle_j <= previousAngle - np.pi/k:
angle_j += 2 * np.pi/k
elif angle_j > previousAngle + np.pi/k:
angle_j -= 2 * np.pi/k
else:
raise Exception('What?!')
return angle_j
def estAngleList(DS,angleSinStrs,angleCosStrs):
"""
For a dataset containing sin and cos strings to estimate either alpha, epsilon, or Phi
return a list of alpha, epsilon, or Phi estimates (one for each generation).
WARNING: At present, kList must be of form [1,2,4,...,2**log2kMax].
Parameters
----------
DS : The dataset from which the angle estimates will be extracted.
angleSinStrs : The list of sin strs that the estimator will use.
angleCosStrs : The list of cos strs that the estimator will use.
Returns
-------
angleHatList
A list of angle estimates, ordered by generation (k).
"""
angleTemp1 = None
angleHatList = []
genNum = len(angleSinStrs)
for i in xrange(genNum):
xhatTemp = DS[angleSinStrs[i]]['plus']
yhatTemp = DS[angleCosStrs[i]]['plus']
Nx = xhatTemp + DS[angleSinStrs[i]]['minus']
Ny = yhatTemp + DS[angleCosStrs[i]]['minus']
angleTemp1 = extractRotationHat(xhatTemp,yhatTemp,2**i,Nx,Ny,angleTemp1)
angleHatList.append(angleTemp1)
return angleHatList
def sinPhi2Func(theta,Phi,epsilon):
"""
Returns the function whose zero, for fixed Phi and epsilon, occurs at the desired value of theta.
(This function exists to be passed to a minimizer to obtain theta.)
WARNING: epsilon gets rescaled to newEpsilon, by dividing by pi/4; will have to change for epsilons far from pi/4.
Parameters
----------
theta : Angle between X and Z axes.
Phi : The auxiliary angle Phi; necessary to calculate theta.
epsilon : Angle of X rotation.
Returns
-------
sinPhi2FuncVal
The value of sinPhi2Func for given inputs. (Must be 0 to achieve "true" theta.)
"""
newEpsilon = (epsilon / (np.pi/4)) - 1
sinPhi2FuncVal = np.abs(2*np.sin(theta)*np.cos(np.pi*newEpsilon/2)*np.sqrt(1-np.sin(theta)**2*np.cos(np.pi*newEpsilon/2)**2)-np.sin(Phi/2))
return sinPhi2FuncVal
def estThetaList(DS,angleSinStrs,angleCosStrs,epsilonList,returnPhiFunList = False):
"""
For a dataset containing sin and cos strings to estimate theta,
along with already-made estimates of epsilon, return a list of theta
(one for each generation).
Parameters
----------
DS : The dataset from which the theta estimates will be extracted.
angleSinStrs : The list of sin strs that the estimator will use.
angleCosStrs : The list of cos strs that the estimator will use.
epsilonList : List of epsilon estimates.
returnPhiFunList : Set to True to obtain measure of how well Eq. III.7 is satisfied. Default is False.
Returns
-------
thetaHatList
A list of theta estimates, ordered by generation (k).
PhiFunList
A list of sinPhi2Func vals at optimal theta values. If not close to 0, constraints unsatisfiable. Only returned if returnPhiFunList is set to True.
"""
PhiList = estAngleList(DS,angleSinStrs,angleCosStrs)
thetaList = []
PhiFunList = []
for index, Phi in enumerate(PhiList):
epsilon = epsilonList[index]
soln = optimize.minimize(lambda x: sinPhi2Func(x,Phi,epsilon),0)
thetaList.append(soln['x'][0])
PhiFunList.append(soln['fun'])
# if soln['fun'] > 1e-2:
# print Phi, epsilon
if returnPhiFunList:
return thetaList, PhiFunList
else:
return thetaList
#def gs2qtys(gateset):
# """
# For a given gateset, extract
# """
def extractAlpha(gateset):
"""
For a given gateset, obtain the angle of rotation about Z axis (for gate "Gz").
WARNING: This is a gauge-covariant parameter! Gauge must be fixed prior to estimating.
Parameters
----------
gateset : The gateset whose "Gz" angle of rotation is to be calculated.
Returns
-------
alphaVal
The value of alpha for the input gateset.
"""
gateLabels = gateset.keys() # gate labels
qtys_to_compute = [ ('%s decomposition' % gl) for gl in gateLabels ]
qtys = GST.ComputeReportables.compute_GateSet_Quantities(qtys_to_compute, gateset, confidenceRegionInfo=None)
alphaVal = qtys['Gz decomposition'].value['pi rotations'] * np.pi
return alphaVal
def extractEpsilon(gateset):
"""
For a given gateset, obtain the angle of rotation about X axis (for gate "Gx").
WARNING: This is a gauge-covariant parameter! Gauge must be fixed prior to estimating.
Parameters
----------
gateset : The gateset whose "Gx" angle of rotation is to be calculated.
Returns
-------
epsilonVal
The value of epsilon for the input gateset.
"""
gateLabels = gateset.keys() # gate labels
qtys_to_compute = [ ('%s decomposition' % gl) for gl in gateLabels ]
qtys = GST.ComputeReportables.compute_GateSet_Quantities(qtys_to_compute, gateset, confidenceRegionInfo=None)
epsilonVal = qtys['Gx decomposition'].value['pi rotations'] * np.pi
return epsilonVal
def extractTheta(gateset):
"""
For a given gateset, obtain the angle between the "X axis of rotation" and the "true" X axis (perpendicular to Z).
(Angle of misalignment between "Gx" axis of rotation and X axis as defined by "Gz".)
WARNING: This is a gauge-covariant parameter! (I think!) Gauge must be fixed prior to estimating.
Parameters
----------
gateset : The gateset whose X axis misaligment is to be calculated.
Returns
-------
thetaVal
The value of theta for the input gateset.
"""
gateLabels = gateset.keys() # gate labels
qtys_to_compute = [ ('%s decomposition' % gl) for gl in gateLabels ]
qtys = GST.ComputeReportables.compute_GateSet_Quantities(qtys_to_compute, gateset, confidenceRegionInfo=None)
thetaVal = np.real_if_close([np.arccos(np.dot(
qtys['Gx decomposition'].getValue()['axis of rotation'],
[0,1,0,0]))])[0]
if thetaVal > np.pi/2:
thetaVal = np.pi - thetaVal
elif thetaVal < -np.pi/2:
thetaVal = np.pi + thetaVal
return thetaVal
def analyzeSimulatedRPEExperiment(inputDataset,trueGateset,stringListD):
"""
Compute angle estimates and compare to true estimates for alpha, epsilon, and theta.
Parameters
----------
inputDataset : The dataset containing the RPE experiments.
trueGateset : The gateset used to generate the RPE data.
stringListD : The dictionary of gate string lists used for the RPE experiments. This should be generated via makeRPEStringListD.
Returns
-------
resultsD
A dictionary of the results
The keys of the dictionary are:
-'alphaHatList' : List (ordered by k) of alpha estimates.
-'epsilonHatList' : List (ordered by k) of epsilon estimates.
-'thetaHatList' : List (ordered by k) of theta estimates.
-'alphaErrorList' : List (ordered by k) of difference between true alpha and RPE estimate of alpha.
-'epsilonErrorList' : List (ordered by k) of difference between true epsilon and RPE estimate of epsilon.
-'thetaErrorList' : List (ordered by k) of difference between true theta and RPE estimate of theta.
-'PhiFunErrorList' : List (ordered by k) of sinPhi2Func values.
"""
alphaCosStrList = stringListD['alpha','cos']
alphaSinStrList = stringListD['alpha','sin']
epsilonCosStrList = stringListD['epsilon','cos']
epsilonSinStrList = stringListD['epsilon','sin']
thetaCosStrList = stringListD['theta','cos']
thetaSinStrList = stringListD['theta','sin']
try:
alphaTrue = trueGateset.alphaTrue
except:
alphaTrue = extractAlpha(trueGateset)
try:
epsilonTrue = trueGateset.epsilonTrue
except:
epsilonTrue = extractEpsilon(trueGateset)
try:
thetaTrue = trueGateset.thetaTrue
except:
thetaTrue = trueGateset(trueGateset)
alphaErrorList = []
epsilonErrorList = []
thetaErrorList = []
# PhiFunErrorList = []
alphaHatList = estAngleList(inputDataset,alphaSinStrList,alphaCosStrList)
epsilonHatList = estAngleList(inputDataset,epsilonSinStrList,epsilonCosStrList)
thetaHatList,PhiFunErrorList = estThetaList(inputDataset,thetaSinStrList,thetaCosStrList,epsilonHatList,returnPhiFunList=True)
for alphaTemp1 in alphaHatList:
alphaErrorList.append(abs(alphaTrue+alphaTemp1))
for epsilonTemp1 in epsilonHatList:
epsilonErrorList.append(abs(epsilonTrue-epsilonTemp1))
# print abs(np.pi/2-abs(alphaTemp1))
for thetaTemp1 in thetaHatList:
thetaErrorList.append(abs(thetaTrue - thetaTemp1))
# for PhiFunTemp1 in PhiFunList:
# PhiFunErrorList.append(PhiFunTemp1)
resultsD = {}
resultsD['alphaHatList'] = alphaHatList
resultsD['epsilonHatList'] = epsilonHatList
resultsD['thetaHatList'] = thetaHatList
resultsD['alphaErrorList'] = alphaErrorList
resultsD['epsilonErrorList'] = epsilonErrorList
resultsD['thetaErrorList'] = thetaErrorList
resultsD['PhiFunErrorList'] = PhiFunErrorList
return resultsD
def ensembleTest(alphaTrue, epsilonTrue, Yrot, SPAMdepol, log2kMax, N, runs, plot=False, savePlot=False):
kList = [2**k for k in range(log2kMax+1)]
alphaCosStrList, alphaSinStrList = makeAlphaStrListsGxGz(kList)
epsilonCosStrList, epsilonSinStrList = makeEpsilonStrListsGxGz(kList)
thetaCosStrList, thetaSinStrList = makeThetaStrListsGxGz(kList)
percentAlphaError = 100*np.abs((np.pi/2-alphaTrue)/alphaTrue)
percentEpsilonError = 100*np.abs((np.pi/4 - epsilonTrue)/epsilonTrue)
simGateset = GST.buildGateset( [2], [('Q0',)],['Gi','Gx','Gz'],
[ "I(Q0)", "X("+str(epsilonTrue)+",Q0)", "Z("+str(alphaTrue)+",Q0)"],
rhoExpressions=["0"], EExpressions=["1"],
spamLabelDict={'plus': (0,0), 'minus': (0,-1) })
gatesetAux1 = GST.buildGateset( [2], [('Q0',)],['Gi','Gy','Gz'],
[ "I(Q0)", "Y("+str(Yrot)+",Q0)", "Z(pi/2,Q0)"],
rhoExpressions=["0"], EExpressions=["1"],
spamLabelDict={'plus': (0,0), 'minus': (0,-1) })
simGateset.set_gate('Gx',myGate(np.dot(np.dot(np.linalg.inv(gatesetAux1['Gy']),simGateset['Gx']),gatesetAux1['Gy'])))
simGateset = GST.GateSetTools.depolarizeSPAM(simGateset,noise=SPAMdepol)
#gateset3 = GST.GateSetTools.depolarizeSPAM(gateset3,noise=0.01)
thetaTrue = extractTheta(simGateset)
SPAMerror = np.dot(simGateset.EVecs[0].T,simGateset.rhoVecs[0])[0,0]
jMax = runs
alphaHatListArray = np.zeros([jMax,log2kMax+1],dtype='object')
epsilonHatListArray = np.zeros([jMax,log2kMax+1],dtype='object')
thetaHatListArray = np.zeros([jMax,log2kMax+1],dtype='object')
alphaErrorArray = np.zeros([jMax,log2kMax+1],dtype='object')
epsilonErrorArray = np.zeros([jMax,log2kMax+1],dtype='object')
thetaErrorArray = np.zeros([jMax,log2kMax+1],dtype='object')
PhiFunErrorArray = np.zeros([jMax,log2kMax+1],dtype='object')
for j in xrange(jMax):
# simDS = GST.generateFakeData(gateset3,alphaCosStrList+alphaSinStrList+epsilonCosStrList+epsilonSinStrList+thetaCosStrList+epsilonSinStrList,
simDS = GST.generateFakeData(simGateset,alphaCosStrList+alphaSinStrList+epsilonCosStrList+epsilonSinStrList+thetaCosStrList+thetaSinStrList,
N,sampleError='binomial',seed=j)
alphaErrorList = []
epsilonErrorList = []
thetaErrorList = []
PhiFunErrorList = []
alphaHatList = estAngleList(simDS,alphaSinStrList,alphaCosStrList)
epsilonHatList = estAngleList(simDS,epsilonSinStrList,epsilonCosStrList)
thetaHatList,PhiFunList = estThetaList(simDS,thetaSinStrList,thetaCosStrList,epsilonHatList,returnPhiFunList=True)
for alphaTemp1 in alphaHatList:
alphaErrorList.append(abs(alphaTrue+alphaTemp1))
for epsilonTemp1 in epsilonHatList:
epsilonErrorList.append(abs(epsilonTrue-epsilonTemp1))
# print abs(np.pi/2-abs(alphaTemp1))
for thetaTemp1 in thetaHatList:
thetaErrorList.append(abs(thetaTrue - thetaTemp1))
for PhiFunTemp1 in PhiFunList:
PhiFunErrorList.append(PhiFunTemp1)
alphaErrorArray[j,:] = np.array(alphaErrorList)
epsilonErrorArray[j,:] = np.array(epsilonErrorList)
thetaErrorArray[j,:] = np.array(thetaErrorList)
PhiFunErrorArray[j,:] = np.array(PhiFunErrorList)
alphaHatListArray[j,:] = np.array(alphaHatList)
epsilonHatListArray[j,:] = np.array(epsilonHatList)
thetaHatListArray[j,:] = np.array(thetaHatList)
#print "True alpha:",alphaTrue
#print "True alpha:",alphaTrue
#print "True alpha:",alphaTrue
#print "True alpha:",alphaTrue
#print "% true alpha deviation from target:", percentAlphaError
if plot:
matplotlib.pyplot.loglog(kList,np.median(alphaErrorArray,axis=0),label='N='+str(N))
matplotlib.pyplot.loglog(kList,np.array(kList)**-1.,'-o',label='1/k')
matplotlib.pyplot.xlabel('k')
matplotlib.pyplot.ylabel(r'$\alpha_z - \widehat{\alpha_z}$')
matplotlib.pyplot.title('RPE error in Z angle\n% error in Z angle '+str(percentAlphaError)+'%, % error in X angle '+str(percentEpsilonError)+'%\n% error in SPAM, '+str(100*SPAMerror)+'%, X-Z axis error '+str(Yrot)+'\nMedian of '+str(jMax)+' Trials')
matplotlib.pyplot.legend()
matplotlib.pyplot.show()
matplotlib.pyplot.loglog(kList,np.median(epsilonErrorArray,axis=0),label='N='+str(N))
matplotlib.pyplot.loglog(kList,np.array(kList)**-1.,'-o',label='1/k')
matplotlib.pyplot.xlabel('k')
matplotlib.pyplot.ylabel(r'$\epsilon_x - \widehat{\epsilon_x}$')
matplotlib.pyplot.title('RPE error in X angle\n% error in Z angle '+str(percentAlphaError)+'%, % error in X angle '+str(percentEpsilonError)+'%\n% error in SPAM, '+str(100*SPAMerror)+'%, X-Z axis error '+str(Yrot)+'\nMedian of '+str(jMax)+' Trials')
matplotlib.pyplot.legend()
matplotlib.pyplot.show()
matplotlib.pyplot.loglog(kList,np.median(thetaErrorArray,axis=0),label='N='+str(N))
matplotlib.pyplot.loglog(kList,np.array(kList)**-1.,'-o',label='1/k')
matplotlib.pyplot.xlabel('k')
matplotlib.pyplot.ylabel(r'$\theta_{xz} - \widehat{\theta_{xz}}$')
matplotlib.pyplot.title('RPE error in X axis angle\n% error in Z angle '+str(percentAlphaError)+'%, % error in X angle '+str(percentEpsilonError)+'%\n% error in SPAM, '+str(100*SPAMerror)+'%, X-Z axis error '+str(Yrot)+'\nMedian of '+str(jMax)+' Trials')
matplotlib.pyplot.legend()
matplotlib.pyplot.show()
matplotlib.pyplot.loglog(kList,np.median(PhiFunErrorArray,axis=0),label='N='+str(N))
# matplotlib.pyplot.loglog(kList,np.array(kList)**-1.,'-o',label='1/k')
matplotlib.pyplot.xlabel('k')
matplotlib.pyplot.ylabel(r'$\Phi func.$')
matplotlib.pyplot.title('RPE error in Phi func.\n% error in Z angle '+str(percentAlphaError)+'%, % error in X angle '+str(percentEpsilonError)+'%\n% error in SPAM, '+str(100*SPAMerror)+'%, X-Z axis error '+str(Yrot)+'\nMedian of '+str(jMax)+' Trials')
matplotlib.pyplot.legend()
outputDict = {}
# outputDict['alphaArray'] = alphaHatListArray
# outputDict['alphaErrorArray'] = alphaErrorArray
# outputDict['epsilonArray'] = epsilonHatListArray
# outputDict['epsilonErrorArray'] = epsilonErrorArray
# outputDict['thetaArray'] = thetaHatListArray
# outputDict['thetaErrorArray'] = thetaErrorArray
# outputDict['PhiFunErrorArray'] = PhiFunErrorArray
# outputDict['alpha'] = alphaTrue
# outputDict['epsilonTrue'] = epsilonTrue
# outputDict['thetaTrue'] = thetaTrue
# outputDict['Yrot'] = Yrot
# outputDict['SPAMdepol'] = SPAMdepol#Input value to depolarize SPAM by
# outputDict['SPAMerror'] = SPAMerror#<<E|rho>>
# outputDict['gs'] = simGateset
# outputDict['N'] = N
return outputDict
'''
def makeRPEDataSet(inputGateset, log2kMax, N, seed = None, returnStringListDict = False):
"""
Generate a fake RPE dataset. At present, only works for kList of form [1,2,4,...,2**log2kMax]
Parameters
----------
inputGateset : The gateset used to generate the data.
log2kMax : Maximum number of times to repeat an RPE "germ"
N : Number of clicks per experiment.
seed : Used to seed numpy's random number generator. Default is None.
returnStringListDict : Do we want a dictionary of the sin and cos experiments for the various angles? Default is False.
Returns
-------
simDS
The simulated dataset containing the RPE experiments.
stringListD
Dictionary of gate string lists for sin and cos experiments; is not returned by default.
"""
kList = [2**k for k in range(log2kMax+1)]
alphaCosStrList, alphaSinStrList = makeAlphaStrListsGxGz(kList)
epsilonCosStrList, epsilonSinStrList = makeEpsilonStrListsGxGz(kList)
thetaCosStrList, thetaSinStrList = makeThetaStrListsGxGz(kList)
totalStrList = alphaCosStrList + alphaSinStrList + epsilonCosStrList + epsilonSinStrList + thetaCosStrList + thetaSinStrList
totalStrList = GST.ListTools.remove_duplicates(totalStrList)#This step is probably superfluous.
simDS = GST.generateFakeData(inputGateset,totalStrList,N,sampleError='binomial',seed=seed)
if returnStringListDict:
stringListD = {}
stringListD['alpha','cos'] = alphaCosStrList
stringListD['alpha','sin'] = alphaSinStrList
stringListD['epsilon','cos'] = epsilonCosStrList
stringListD['epsilon','sin'] = epsilonSinStrList
stringListD['theta','cos'] = thetaCosStrList
stringListD['theta','sin'] = thetaSinStrList
return simDS, stringListD
else:
return simDS, None
'''