The Quantum Exact Simulation Toolkit v4.0.0
Loading...
Searching...
No Matches
test_decoherence.cpp
1/** @file
2 * Ported tests of the deprecated QuEST v3 interface,
3 * unit testing the "decoherence" module.
4 *
5 * This file should be excluded from doxygen parsing so
6 * as not to conflict with the doc of the v4 unit tests.
7 *
8 * @author Tyson Jones
9 * @author Oliver Thomson Brown (ported to Catch2 v3)
10 * @author Ali Rezaei (tested porting to QuEST v4)
11 */
12
13#include <catch2/catch_test_macros.hpp>
14#include <catch2/matchers/catch_matchers_string.hpp>
15#include <catch2/generators/catch_generators_adapters.hpp>
16#include <catch2/generators/catch_generators_range.hpp>
17#include <catch2/generators/catch_generators_random.hpp>
18
19// must define preprocessors to enable quest's
20// deprecated v3 API, and disable the numerous
21// warnings issued by its compilation
22#define INCLUDE_DEPRECATED_FUNCTIONS 1
23#define DISABLE_DEPRECATION_WARNINGS 1
24
25#include "test_utilities.hpp"
26
27#include <random>
28#include <vector>
29#include <algorithm>
30
31using std::vector;
32
33/** Prepares a density matrix in the debug state, and the reference QMatrix
34 */
35#define PREPARE_TEST(qureg, ref) \
36 Qureg qureg = createForcedDensityQureg(NUM_QUBITS); \
37 initDebugState(qureg); \
38 QMatrix ref = toQMatrix(qureg); \
39 assertQuregAndRefInDebugState(qureg, ref); \
40 setValidationEpsilon(REAL_EPS);
41
42/* allows concise use of ContainsSubstring in catch's REQUIRE_THROWS_WITH */
43using Catch::Matchers::ContainsSubstring;
44
45
46
47/** @sa mixDamping
48 * @ingroup deprecatedtests
49 * @author Tyson Jones
50 */
51TEST_CASE( "mixDamping", "[decoherence]" ) {
52
53 PREPARE_TEST(qureg, ref);
54
55 SECTION( "correctness" ) {
56
57 int target = GENERATE( range(0, NUM_QUBITS) );
58 qreal prob = getRandomReal(0, 1);
59 mixDamping(qureg, target, prob);
60
61 // ref -> kraus0 ref kraus0^dagger + kraus1 ref kraus1^dagger
62 QMatrix kraus0{{1,0},{0,sqrt(1-prob)}};
63 QMatrix rho0 = ref;
64 applyReferenceOp(rho0, target, kraus0);
65 QMatrix kraus1{{0,sqrt(prob)},{0,0}};
66 QMatrix rho1 = ref;
67 applyReferenceOp(rho1, target, kraus1);
68 ref = rho0 + rho1;
69
70 REQUIRE( areEqual(qureg, ref) );
71 }
72 SECTION( "validation ") {
73
74 SECTION( "qubit index" ) {
75
76 int target = GENERATE( -1, NUM_QUBITS );
77 REQUIRE_THROWS_WITH( mixDamping(qureg, target, 0), ContainsSubstring("Invalid target") );
78
79 }
80 SECTION( "probability" ) {
81
82 REQUIRE_THROWS_WITH( mixDamping(qureg, 0, -.1), ContainsSubstring("probability is invalid") );
83 REQUIRE_THROWS_WITH( mixDamping(qureg, 0, 1.1), ContainsSubstring("probability is invalid") );
84 }
85 SECTION( "density-matrix" ) {
86
87 Qureg vec = createQureg(NUM_QUBITS);
88 REQUIRE_NOTHROW( mixDamping(vec, 0, 0) ); // zero-prob allowed in v4
89 REQUIRE_THROWS_WITH( mixDamping(vec, 0, .1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
91 }
92 }
93 destroyQureg(qureg, getQuESTEnv());
94}
95
96
97
98/** @sa mixDensityMatrix
99 * @ingroup deprecatedtests
100 * @author Tyson Jones
101 */
102TEST_CASE( "mixDensityMatrix", "[decoherence]" ) {
103
104 Qureg qureg1 = createForcedDensityQureg(NUM_QUBITS);
105 Qureg qureg2 = createForcedDensityQureg(NUM_QUBITS);
106 initDebugState(qureg1);
107 initDebugState(qureg2);
108 QMatrix ref1 = toQMatrix(qureg1);
109 QMatrix ref2 = toQMatrix(qureg2);
110
111 SECTION( "correctness" ) {
112
113 // test p in {0, 1} and 10 random values in (0,1)
114 qreal prob = GENERATE( 0., 1., take(10, random(0.,1.)) );
115 mixDensityMatrix(qureg1, prob, qureg2);
116
117 // ensure target qureg modified correctly
118 ref1 = (1-prob)*ref1 + (prob)*ref2;
119 REQUIRE( areEqual(qureg1, ref1) );
120
121 // enure other qureg was not modified
122 REQUIRE( areEqual(qureg2, ref2) );
123 }
124 SECTION( "input validation" ) {
125
126 SECTION( "probabilities" ) {
127
128 qreal prob = GENERATE( -0.1, 1.1 );
129 REQUIRE_THROWS_WITH( mixDensityMatrix(qureg1, prob, qureg2), ContainsSubstring("probability is invalid") );
130 }
131 SECTION( "density matrices" ) {
132
133 // one is statevec
134 Qureg state1 = createQureg(qureg1.numQubits, getQuESTEnv());
135 REQUIRE_THROWS_WITH( mixDensityMatrix(state1, 0, qureg1), ContainsSubstring("The first Qureg") && ContainsSubstring("must be a density matrix") );
136
137 // in v4, the 2nd arg can be a statevector
138
139 // REQUIRE_THROWS_WITH( mixDensityMatrix(qureg1, 0, state1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
140
141 // both are statevec
142 Qureg state2 = createQureg(qureg1.numQubits, getQuESTEnv());
143 REQUIRE_THROWS_WITH( mixDensityMatrix(state1, 0, state2), ContainsSubstring("The first Qureg") && ContainsSubstring("must be a density matrix") );
144
145 destroyQureg(state1, getQuESTEnv());
146 destroyQureg(state2, getQuESTEnv());
147 }
148 SECTION( "matching dimensions" ) {
149
150 Qureg qureg3 = createDensityQureg(1 + qureg1.numQubits, getQuESTEnv());
151 REQUIRE_THROWS_WITH( mixDensityMatrix(qureg1, 0, qureg3), ContainsSubstring("inconsistent number of qubits") );
152 REQUIRE_THROWS_WITH( mixDensityMatrix(qureg3, 0, qureg1), ContainsSubstring("inconsistent number of qubits") );
153 destroyQureg(qureg3, getQuESTEnv());
154 }
155 }
156 destroyQureg(qureg1, getQuESTEnv());
157 destroyQureg(qureg2, getQuESTEnv());
158}
159
160
161
162/** @sa mixDephasing
163 * @ingroup deprecatedtests
164 * @author Tyson Jones
165 */
166TEST_CASE( "mixDephasing", "[decoherence]" ) {
167
168 PREPARE_TEST(qureg, ref);
169
170 SECTION( "correctness" ) {
171
172 int target = GENERATE( range(0,NUM_QUBITS) );
173 qreal prob = getRandomReal(0, 1/2.);
174 mixDephasing(qureg, target, prob);
175
176 // ref -> (1 - prob) ref + prob Z ref Z
177 QMatrix phaseRef = ref;
178 applyReferenceOp(phaseRef, target, QMatrix{{1,0},{0,-1}}); // Z ref Z
179 ref = ((1 - prob) * ref) + (prob * phaseRef);
180
181 REQUIRE( areEqual(qureg, ref) );
182 }
183 SECTION( "validation" ) {
184
185 SECTION( "qubit index" ) {
186
187 int target = GENERATE( -1, NUM_QUBITS );
188 REQUIRE_THROWS_WITH( mixDephasing(qureg, target, 0), ContainsSubstring("Invalid target") );
189
190 }
191 SECTION( "probability" ) {
192
193 REQUIRE_THROWS_WITH( mixDephasing(qureg, 0, -.1), ContainsSubstring("probability is invalid") );
194 REQUIRE_THROWS_WITH( mixDephasing(qureg, 0, .6), ContainsSubstring("probability") && ContainsSubstring("1/2") );
195 }
196 SECTION( "density-matrix" ) {
197
198 Qureg vec = createQureg(NUM_QUBITS);
199 REQUIRE_NOTHROW( mixDephasing(vec, 0, 0) ); // zero prob ok in v4
200 REQUIRE_THROWS_WITH( mixDephasing(vec, 0, 0.1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
202 }
203 }
204 destroyQureg(qureg, getQuESTEnv());
205}
206
207
208
209/** @sa mixDepolarising
210 * @ingroup deprecatedtests
211 * @author Tyson Jones
212 */
213TEST_CASE( "mixDepolarising", "[decoherence]" ) {
214
215 PREPARE_TEST(qureg, ref);
216
217 SECTION( "correctness " ) {
218
219 int target = GENERATE( range(0,NUM_QUBITS) );
220 qreal prob = getRandomReal(0, 3/4.);
221 mixDepolarising(qureg, target, prob);
222
223 QMatrix xRef = ref;
224 applyReferenceOp(xRef, target, QMatrix{{0,1},{1,0}}); // X ref X
225 QMatrix yRef = ref;
226 applyReferenceOp(yRef, target, QMatrix{{0,-qcomp(0,1)},{qcomp(0,1),0}}); // Y ref Y
227 QMatrix zRef = ref;
228 applyReferenceOp(zRef, target, QMatrix{{1,0},{0,-1}}); // Z ref Z
229 ref = ((1 - prob) * ref) + ((prob/3.) * ( xRef + yRef + zRef));
230
231 REQUIRE( areEqual(qureg, ref) );
232 }
233 SECTION( "validation ") {
234
235 SECTION( "qubit index" ) {
236
237 int target = GENERATE( -1, NUM_QUBITS );
238 REQUIRE_THROWS_WITH( mixDepolarising(qureg, target, 0), ContainsSubstring("Invalid target") );
239
240 }
241 SECTION( "probability" ) {
242
243 REQUIRE_THROWS_WITH( mixDepolarising(qureg, 0, -.1), ContainsSubstring("probability is invalid") );
244 REQUIRE_THROWS_WITH( mixDepolarising(qureg, 0, .76), ContainsSubstring("probability") && ContainsSubstring("3/4") );
245 }
246 SECTION( "density-matrix" ) {
247
248 Qureg vec = createQureg(NUM_QUBITS);
249 REQUIRE_NOTHROW( mixDepolarising(vec, 0, 0) ); // zero-prob ok in v4
250 REQUIRE_THROWS_WITH( mixDepolarising(vec, 0, 0.1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
252 }
253 }
254 destroyQureg(qureg, getQuESTEnv());
255}
256
257
258
259/** @sa mixMultiQubitKrausMap
260 * @ingroup deprecatedtests
261 * @author Tyson Jones
262 */
263TEST_CASE( "mixMultiQubitKrausMap", "[decoherence]" ) {
264
265 PREPARE_TEST(qureg, ref);
266
267 // figure out max-num (inclusive) targs allowed by hardware backend
268 // (each node must contain as 2^(2*numTargs) amps)
269 int maxNumTargs = qureg.logNumAmpsPerNode / 2;
270
271 SECTION( "correctness" ) {
272
273 /* note that this function incurs a stack overhead when numTargs < 4,
274 * and a heap overhead when numTargs >= 4
275 */
276
277 // try every size (qubit wise) map
278 int numTargs = GENERATE_COPY( range(1,maxNumTargs+1) ); // inclusive upper bound
279
280 // previously, we tried every unique set of targets, via:
281 // int* targs = GENERATE_COPY( sublists(range(0,NUM_QUBITS), numTargs) );
282 // alas, this is too slow for CI, so we instead try a fixed number of random sets:
283 GENERATE( range(0, 10) );
284 vector<int> targs(numTargs);
285 setRandomTargets(targs, NUM_QUBITS);
286
287 // try the min and max number of operators, and 2 random numbers
288 // (there are way too many to try all!)
289 int maxNumOps = (2*numTargs)*(2*numTargs);
290 int numOps = GENERATE_COPY( 1, maxNumOps, take(2,random(1,maxNumOps)) );
291
292 // use a new random map
293 vector<QMatrix> matrs = getRandomKrausMap(numTargs, numOps);
294
295 // create map in QuEST datatypes
296 vector<ComplexMatrixN> ops(numOps);
297 for (int i=0; i<numOps; i++) {
298 ops[i] = createCompMatr(numTargs);
299 setCompMatr(ops[i], matrs[i]);
300 }
301
302 mixMultiQubitKrausMap(qureg, targs.data(), numTargs, ops.data(), numOps);
303
304 // set ref -> K_i ref K_i^dagger
305 vector<QMatrix> matrRefs(numOps);
306 for (int i=0; i<numOps; i++) {
307 matrRefs[i] = ref;
308 applyReferenceOp(matrRefs[i], targs.data(), numTargs, matrs[i]);
309 }
310 ref = getZeroMatrix(ref.size());
311 for (int i=0; i<numOps; i++)
312 ref += matrRefs[i];
313
314 REQUIRE( areEqual(qureg, ref, 1E2*REAL_EPS) );
315
316 // cleanup QuEST datatypes
317 for (int i=0; i<numOps; i++)
318 destroyCompMatr(ops[i]);
319 }
320 SECTION( "input validation" ) {
321
322 // spoof a list of properly-initialised CompMatr to avoid seg-fault in deprecation API.
323 // note some functions will need smaller matrices which is fine; a subset of each
324 // spoof'd matrix will be copied over by the deprecation layer
325 ComplexMatrixN spoofOps[NUM_QUBITS];
326 for (int i=0; i<NUM_QUBITS; i++) {
327 spoofOps[i] = createComplexMatrixN(NUM_QUBITS);
328 syncCompMatr(spoofOps[i]);
329 }
330
331 SECTION( "repetition of target" ) {
332
333 // make valid targets
334 int targs[NUM_QUBITS];
335 for (int i=0; i<NUM_QUBITS; i++)
336 targs[i] = i;
337
338 // duplicate one
339 int badInd = GENERATE( range(0,NUM_QUBITS) );
340 int copyInd = GENERATE_COPY( filter([=](int i){ return i!=badInd; }, range(0,NUM_QUBITS)) );
341 targs[badInd] = targs[copyInd];
342
343 REQUIRE_THROWS_WITH( mixMultiQubitKrausMap(qureg, targs, NUM_QUBITS, spoofOps, 1), ContainsSubstring("target qubits") && ContainsSubstring("unique") );
344 }
345 SECTION( "qubit indices" ) {
346
347 // make valid targets
348 int targs[NUM_QUBITS];
349 for (int i=0; i<NUM_QUBITS; i++)
350 targs[i] = i;
351
352 // make one invalid
353 targs[GENERATE( range(0,NUM_QUBITS) )] = GENERATE( -1, NUM_QUBITS );
354
355 REQUIRE_THROWS_WITH( mixMultiQubitKrausMap(qureg, targs, NUM_QUBITS, spoofOps, 1), ContainsSubstring("Invalid target qubit") );
356 }
357
358 // cannot test this with deprecated API, since 'numOps' informs a copy before validation
359
360 // SECTION( "number of operators" ) {
361
362 // int numTargs = GENERATE_COPY( range(1,maxNumTargs+1) );
363 // int numOps = GENERATE_REF( -1, 0 );
364
365 // // make valid targets to avoid triggering target validation
366 // vector<int> targs(numTargs);
367 // for (int i=0; i<numTargs; i++)
368 // targs[i] = i;
369
370 // REQUIRE_THROWS_WITH( mixMultiQubitKrausMap(qureg, targs.data(), numTargs, spoofOps, numOps), ContainsSubstring("must be given a strictly positive number of matrices") );
371 // }
372
373 // this validation cannot be performed by the deprecation API which copies
374 // over all ops arguments without prior checking them as NULL
375
376 // SECTION( "initialisation of operators" ) {
377
378 // /* compilers don't auto-initialise to NULL; the below circumstance
379 // * only really occurs when 'malloc' returns NULL in createCompMatr,
380 // * which actually triggers its own validation. Hence this test is useless
381 // * currently.
382 // */
383
384 // int numTargs = NUM_QUBITS;
385 // int numOps = (2*numTargs)*(2*numTargs);
386
387 // vector<ComplexMatrixN> ops(numOps);
388 // for (int i=0; i<numOps; i++)
389 // ops[i] = createComplexMatrixN(numTargs);
390
391 // // make one of the max-ops explicitly NULL
392 // ops[GENERATE_COPY( range(0,numTargs) )].cpuElems = NULL;
393
394 // // make valid targets to avoid triggering target validation
395 // vector<int> targs(numTargs);
396 // for (int i=0; i<numTargs; i++)
397 // targs[i] = i;
398
399 // REQUIRE_THROWS_WITH( mixMultiQubitKrausMap(qureg, targs.data(), numTargs, ops.data(), numOps), ContainsSubstring("ComplexMatrixN") && ContainsSubstring("created") );
400
401 // for (int i=0; i<numOps; i++)
402 // destroyComplexMatrixN(ops[i]);
403 // }
404
405 // this validation cannot be performed by the deprecation API which copies
406 // over all ops arguments without prior checking them as matching in dimension
407
408 // SECTION( "dimension of operators" ) {
409
410 // // make valid (dimension-wise) max-qubits Kraus map
411 // int numTargs = NUM_QUBITS;
412 // int numOps = (2*numTargs)*(2*numTargs);
413 // vector<ComplexMatrixN> ops(numOps);
414 // for (int i=0; i<numOps; i++)
415 // ops[i] = createComplexMatrixN(numTargs);
416
417 // // make one have wrong-dimensions
418 // int badInd = GENERATE_COPY( range(0,numTargs) );
419 // destroyComplexMatrixN(ops[badInd]);
420 // ops[badInd] = createComplexMatrixN(numTargs - 1);
421
422 // // make valid targets to avoid triggering target validation
423 // vector<int> targs(numTargs);
424 // for (int i=0; i<numTargs; i++)
425 // targs[i] = i;
426
427 // REQUIRE_THROWS_WITH( mixMultiQubitKrausMap(qureg, targs.data(), numTargs, ops.data(), numOps), ContainsSubstring("same number of qubits") );
428
429 // for (int i=0; i<numOps; i++)
430 // destroyComplexMatrixN(ops[i]);
431 // }
432
433 SECTION( "trace preserving" ) {
434
435 int numTargs = GENERATE_COPY( range(1,maxNumTargs+1) );
436 int maxNumOps = (2*numTargs) * (2*numTargs);
437 int numOps = GENERATE_COPY( 1, 2, maxNumOps );
438
439 // generate a valid map
440 vector<QMatrix> matrs = getRandomKrausMap(numTargs, numOps);
441 vector<ComplexMatrixN> ops(numOps);
442 for (int i=0; i<numOps; i++) {
443 ops[i] = createComplexMatrixN(numTargs);
444 toComplexMatrixN(matrs[i], ops[i]);
445 }
446
447 // make only one invalid
448 ops[GENERATE_COPY( 0, numOps - 1)].cpuElems[0][0] = -123456789;
449
450 // make valid targets to avoid triggering target validation
451 vector<int> targs(numTargs);
452 for (int i=0; i<numTargs; i++)
453 targs[i] = i;
454
455 REQUIRE_THROWS_WITH( mixMultiQubitKrausMap(qureg, targs.data(), numTargs, ops.data(), numOps), ContainsSubstring("trace preserving") );
456
457 for (int i=0; i<numOps; i++)
458 destroyComplexMatrixN(ops[i]);
459 }
460 SECTION( "density-matrix" ) {
461
462 Qureg statevec = createQureg(NUM_QUBITS);
463
464 // make valid targets to avoid triggering target validation
465 int targs[NUM_QUBITS];
466 for (int i=0; i<NUM_QUBITS; i++)
467 targs[i] = i;
468
469 REQUIRE_THROWS_WITH( mixMultiQubitKrausMap(statevec, targs, NUM_QUBITS, spoofOps, 1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
470 destroyQureg(statevec, getQuESTEnv());
471
472 }
473 SECTION( "operator fits in node" ) {
474
475 // each node requires (2 numTargs)^2 amplitudes
476 int minAmps = (2*NUM_QUBITS) * (2*NUM_QUBITS);
477
478 // make valid targets to avoid triggering target validation
479 int targs[NUM_QUBITS];
480 for (int i=0; i<NUM_QUBITS; i++)
481 targs[i] = i;
482
483 // make a simple Identity map
484 ComplexMatrixN ops[] = {createComplexMatrixN(NUM_QUBITS)};
485 for (int i=0; i<(1<<NUM_QUBITS); i++)
486 ops[0].cpuElems[i][i] = 1;
487 syncCompMatr(ops[0]);
488
489 // fake a smaller qureg
490 qureg.isDistributed = 1;
491 qureg.numAmpsPerNode = minAmps - 1;
492 qureg.logNumAmpsPerNode = log2(minAmps) - 1;
493 REQUIRE_THROWS_WITH( mixMultiQubitKrausMap(qureg, targs, NUM_QUBITS, ops, 1), ContainsSubstring("each node's communication buffer") && ContainsSubstring("cannot simultaneously store") );
494
495 destroyComplexMatrixN(ops[0]);
496 }
497
498 for (int i=0; i<NUM_QUBITS; i++)
499 destroyCompMatr(spoofOps[i]);
500 }
501 destroyQureg(qureg, getQuESTEnv());
502}
503
504
505
506/** @sa mixPauli
507 * @ingroup deprecatedtests
508 * @author Tyson Jones
509 */
510TEST_CASE( "mixPauli", "[decoherence]" ) {
511
512 PREPARE_TEST(qureg, ref);
513
514 SECTION( "correctness" ) {
515
516 int target = GENERATE( range(0,NUM_QUBITS) );
517
518 // randomly generate valid pauli-error probabilities
519 qreal probs[3];
520 qreal max0 = 1/2.; // satisfies p1 < 1 - py
521 probs[0] = getRandomReal(0, max0);
522 qreal max1 = (max0 - probs[0])/2.; // p2 can use half of p1's "unused space"
523 probs[1] = getRandomReal(0, max1);
524 qreal max2 = (max1 - probs[1])/2.; // p3 can use half of p2's "unused space"
525 probs[2] = getRandomReal(0, max2);
526
527 // uniformly randomly assign probs (bound to target)
528 int inds[3] = {0,1,2};
529 std::shuffle(inds,inds+3, std::default_random_engine(1E5 * target));
530 qreal probX = probs[inds[0]]; // seed:target shows no variation
531 qreal probY = probs[inds[1]];
532 qreal probZ = probs[inds[2]];
533
534 mixPauli(qureg, target, probX, probY, probZ);
535
536 QMatrix xRef = ref;
537 applyReferenceOp(xRef, target, QMatrix{{0,1},{1,0}}); // X ref X
538 QMatrix yRef = ref;
539 applyReferenceOp(yRef, target, QMatrix{{0,-qcomp(0,1)},{qcomp(0,1),0}}); // Y ref Y
540 QMatrix zRef = ref;
541 applyReferenceOp(zRef, target, QMatrix{{1,0},{0,-1}}); // Z ref Z
542 ref = ((1 - probX - probY - probZ) * ref) +
543 (probX * xRef) + (probY * yRef) + (probZ * zRef);
544
545 REQUIRE( areEqual(qureg, ref) );
546 }
547 SECTION( "input validation" ) {
548
549 SECTION( "qubit index" ) {
550
551 int target = GENERATE( -1, NUM_QUBITS );
552 REQUIRE_THROWS_WITH( mixPauli(qureg, target, 0, 0, 0), ContainsSubstring("Invalid target") );
553
554 }
555 SECTION( "probability" ) {
556
557 int target = 0;
558
559 // probs clearly must be in [0, 1]
560 REQUIRE_THROWS_WITH( mixPauli(qureg, target, -.1, 0, 0), ContainsSubstring("probability is invalid") );
561 REQUIRE_THROWS_WITH( mixPauli(qureg, target, 0, -.1, 0), ContainsSubstring("probability is invalid") );
562 REQUIRE_THROWS_WITH( mixPauli(qureg, target, 0, 0, -.1), ContainsSubstring("probability is invalid") );
563
564 // max single-non-zero-prob is 0.5
565 REQUIRE_THROWS_WITH( mixPauli(qureg, target, .6, 0, 0), ContainsSubstring("probabilities exceed that which induce maximal mixing") );
566 REQUIRE_THROWS_WITH( mixPauli(qureg, target, 0, .6, 0), ContainsSubstring("probabilities exceed that which induce maximal mixing") );
567 REQUIRE_THROWS_WITH( mixPauli(qureg, target, 0, 0, .6), ContainsSubstring("probabilities exceed that which induce maximal mixing") );
568
569 // must satisfy px, py, pz < 1 - px - py - pz
570 REQUIRE_THROWS_WITH( mixPauli(qureg, target, .3, .3, .3), ContainsSubstring("probabilities exceed that which induce maximal mixing") );
571 }
572 SECTION( "density-matrix" ) {
573
574 Qureg vec = createQureg(NUM_QUBITS);
575 REQUIRE_NOTHROW( mixPauli(vec, 0, 0, 0, 0) ); // zero-prob ok in v4
576 REQUIRE_THROWS_WITH( mixPauli(vec, 0, 0.1, 0, 0), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
577 REQUIRE_THROWS_WITH( mixPauli(vec, 0, 0, 0.1, 0), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
578 REQUIRE_THROWS_WITH( mixPauli(vec, 0, 0, 0, 0.1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
580 }
581 }
582 destroyQureg(qureg, getQuESTEnv());
583}
584
585
586
587/** @sa mixKrausMap
588 * @ingroup deprecatedtests
589 * @author Tyson Jones
590 */
591TEST_CASE( "mixKrausMap", "[decoherence]" ) {
592
593 PREPARE_TEST(qureg, ref);
594
595 SECTION( "correctness" ) {
596
597 int target = GENERATE( range(0,NUM_QUBITS) );
598 int numOps = GENERATE( range(1,5) ); // max 4 inclusive
599 vector<QMatrix> matrs = getRandomKrausMap(1, numOps);
600
601 vector<ComplexMatrix2> ops(numOps);
602 for (int i=0; i<numOps; i++)
603 ops[i] = toComplexMatrix2(matrs[i]);
604
605 v3_mixKrausMap(qureg, target, ops.data(), numOps);
606
607 // set ref -> K_i ref K_i^dagger
608 vector<QMatrix> matrRefs(numOps);
609 for (int i=0; i<numOps; i++) {
610 matrRefs[i] = ref;
611 applyReferenceOp(matrRefs[i], target, matrs[i]);
612 }
613 ref = getZeroMatrix(ref.size());
614 for (int i=0; i<numOps; i++)
615 ref += matrRefs[i];
616
617 REQUIRE( areEqual(qureg, ref, 10*REAL_EPS) );
618 }
619 SECTION( "input validation" ) {
620
621 // cannot use NULL because v3 deprecation API copies ops arg before v4 validation
622 ComplexMatrix2 spoofOps[1];
623
624 SECTION( "number of operators" ) {
625
626 int numOps = 0;
627
628 REQUIRE_THROWS_WITH( v3_mixKrausMap(qureg, 0, spoofOps, numOps), ContainsSubstring("must be given a strictly positive number of matrices") );
629 }
630 SECTION( "trace preserving" ) {
631
632 // valid Kraus map
633 int numOps = GENERATE( range(1,5) ); // max 4 inclusive
634 vector<QMatrix> matrs = getRandomKrausMap(1, numOps);
635 vector<ComplexMatrix2> ops(numOps);
636 for (int i=0; i<numOps; i++)
637 ops[i] = toComplexMatrix2(matrs[i]);
638
639 // make invalid
640 ops[GENERATE_REF( range(0,numOps) )].real[0][0] = 9999;
641 REQUIRE_THROWS_WITH( v3_mixKrausMap(qureg, 0, ops.data(), numOps), ContainsSubstring("trace preserving") );
642 }
643 SECTION( "qubit index" ) {
644
645 int target = GENERATE( -1, NUM_QUBITS );
646 REQUIRE_THROWS_WITH( v3_mixKrausMap(qureg, target, spoofOps, 1), ContainsSubstring("Invalid target qubit") );
647 }
648 SECTION( "density-matrix" ) {
649
650 Qureg vec = createQureg(NUM_QUBITS);
651 REQUIRE_THROWS_WITH( v3_mixKrausMap(vec, 0, spoofOps, 1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
653 }
654 SECTION( "operators fit in node" ) {
655
656 qureg.isDistributed = 1;
657 qureg.numAmpsPerNode = 3; // min 4
658 qureg.logNumAmpsPerNode = 1; // min 2
659
660 REQUIRE_THROWS_WITH( v3_mixKrausMap(qureg, 0, spoofOps, 1), ContainsSubstring("each node's communication buffer") && ContainsSubstring("cannot simultaneously store") );
661 }
662 }
663 destroyQureg(qureg, getQuESTEnv());
664}
665
666
667
668/** @sa mixNonTPKrausMap
669 * @ingroup deprecatedtests
670 * @author Tyson Jones
671 */
672TEST_CASE( "mixNonTPKrausMap", "[decoherence]" ) {
673
674 PREPARE_TEST(qureg, ref);
675
676 SECTION( "correctness" ) {
677
678 int target = GENERATE( range(0,NUM_QUBITS) );
679 int numOps = GENERATE( range(1,5) ); // max 4 inclusive
680
681 // map consists of unconstrained 2x2 random matrices
682 vector<QMatrix> matrs;
683 for (int i=0; i<numOps; i++)
684 matrs.push_back(getRandomQMatrix(2));
685
686 vector<ComplexMatrix2> ops(numOps);
687 for (int i=0; i<numOps; i++)
688 ops[i] = toComplexMatrix2(matrs[i]);
689 mixNonTPKrausMap(qureg, target, ops.data(), numOps);
690
691 // set ref -> K_i ref K_i^dagger
692 vector<QMatrix> matrRefs(numOps);
693 for (int i=0; i<numOps; i++) {
694 matrRefs[i] = ref;
695 applyReferenceOp(matrRefs[i], target, matrs[i]);
696 }
697 ref = getZeroMatrix(ref.size());
698 for (int i=0; i<numOps; i++)
699 ref += matrRefs[i];
700
701 REQUIRE( areEqual(qureg, ref, 1E2*REAL_EPS) );
702 }
703 SECTION( "input validation" ) {
704
705 // v3 deprecation API copies ops list before v4 validation, so we cannot pass ops=NULL
706 ComplexMatrix2 spoofOps[1];
707
708 SECTION( "number of operators" ) {
709
710 int numOps = 0;
711 REQUIRE_THROWS_WITH( mixNonTPKrausMap(qureg, 0, spoofOps, numOps), ContainsSubstring("must be given a strictly positive number of matrices") );
712 }
713 SECTION( "qubit index" ) {
714
715 int target = GENERATE( -1, NUM_QUBITS );
716 REQUIRE_THROWS_WITH( mixNonTPKrausMap(qureg, target, spoofOps, 1), ContainsSubstring("Invalid target qubit") );
717 }
718 SECTION( "density-matrix" ) {
719
720 Qureg vec = createQureg(NUM_QUBITS);
721 REQUIRE_THROWS_WITH( mixNonTPKrausMap(vec, 0, spoofOps, 1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
723 }
724 SECTION( "operators fit in node" ) {
725
726 qureg.isDistributed = 1;
727 qureg.numAmpsPerNode = 3; // min 4
728 qureg.logNumAmpsPerNode = 1; // min 2
729 REQUIRE_THROWS_WITH( mixNonTPKrausMap(qureg, 0, spoofOps, 1), ContainsSubstring("each node's communication buffer") && ContainsSubstring("cannot simultaneously store") );
730 }
731 }
732 destroyQureg(qureg, getQuESTEnv());
733}
734
735
736
737/** @sa mixNonTPMultiQubitKrausMap
738 * @ingroup deprecatedtests
739 * @author Tyson Jones
740 */
741TEST_CASE( "mixNonTPMultiQubitKrausMap", "[decoherence]" ) {
742
743 PREPARE_TEST(qureg, ref);
744
745 // figure out max-num (inclusive) targs allowed by hardware backend
746 // (each node must contain as 2^(2*numTargs) amps)
747 int maxNumTargs = calcLog2(qureg.numAmpsPerNode) / 2;
748
749 SECTION( "correctness" ) {
750
751 /* note that this function incurs a stack overhead when numTargs < 4,
752 * and a heap overhead when numTargs >= 4
753 */
754
755 int numTargs = GENERATE_COPY( range(1,maxNumTargs+1) ); // inclusive upper bound
756
757 // previously, we tried every unique set of targets, via:
758 // int* targs = GENERATE_COPY( sublists(range(0,NUM_QUBITS), numTargs) );
759 // alas, this is too slow for CI, so we instead try a fixed number of random sets:
760 GENERATE( range(0, 10) );
761 vector<int> targs(numTargs);
762 setRandomTargets(targs, NUM_QUBITS);
763
764 // try the min and max number of operators, and 2 random numbers
765 // (there are way too many to try all!)
766 int maxNumOps = (2*numTargs)*(2*numTargs);
767 int numOps = GENERATE_COPY( 1, maxNumOps, take(2,random(1,maxNumOps)) );
768
769 // use a new random map, of unconstrained random (2^numTargs x 2^numTargs) matrices
770 vector<QMatrix> matrs;
771 for (int i=0; i<numOps; i++)
772 matrs.push_back(getRandomQMatrix(1 << numTargs));
773
774 // create map in QuEST datatypes
775 vector<ComplexMatrixN> ops(numOps);
776 for (int i=0; i<numOps; i++) {
777 ops[i] = createComplexMatrixN(numTargs);
778 toComplexMatrixN(matrs[i], ops[i]);
779 }
780
781 mixNonTPMultiQubitKrausMap(qureg, targs.data(), numTargs, ops.data(), numOps);
782
783 // set ref -> K_i ref K_i^dagger
784 vector<QMatrix> matrRefs(numOps);
785 for (int i=0; i<numOps; i++) {
786 matrRefs[i] = ref;
787 applyReferenceOp(matrRefs[i], targs.data(), numTargs, matrs[i]);
788 }
789 ref = getZeroMatrix(ref.size());
790 for (int i=0; i<numOps; i++)
791 ref += matrRefs[i];
792
793 REQUIRE( areEqual(qureg, ref, 1E5*REAL_EPS) );
794
795 // cleanup QuEST datatypes
796 for (int i=0; i<numOps; i++)
797 destroyComplexMatrixN(ops[i]);
798 }
799 SECTION( "input validation" ) {
800
801 // spoof a list of properly-initialised CompMatr to avoid seg-fault in deprecation API.
802 // note some functions will need smaller matrices which is fine; a subset of each
803 // spoof'd matrix will be copied over by the deprecation layer
804 ComplexMatrixN spoofOps[NUM_QUBITS];
805 for (int i=0; i<NUM_QUBITS; i++) {
806 spoofOps[i] = createComplexMatrixN(NUM_QUBITS);
807 syncCompMatr(spoofOps[i]);
808 }
809
810 SECTION( "repetition of target" ) {
811
812 // make valid targets
813 int targs[NUM_QUBITS];
814 for (int i=0; i<NUM_QUBITS; i++)
815 targs[i] = i;
816
817 // duplicate one
818 int badInd = GENERATE( range(0,NUM_QUBITS) );
819 int copyInd = GENERATE_COPY( filter([=](int i){ return i!=badInd; }, range(0,NUM_QUBITS)) );
820 targs[badInd] = targs[copyInd];
821
822 REQUIRE_THROWS_WITH( mixNonTPMultiQubitKrausMap(qureg, targs, NUM_QUBITS, spoofOps, 1), ContainsSubstring("target qubits") && ContainsSubstring("unique") );
823 }
824 SECTION( "qubit indices" ) {
825
826 // make valid targets
827 int targs[NUM_QUBITS];
828 for (int i=0; i<NUM_QUBITS; i++)
829 targs[i] = i;
830
831 // make one invalid
832 targs[GENERATE( range(0,NUM_QUBITS) )] = GENERATE( -1, NUM_QUBITS );
833
834 REQUIRE_THROWS_WITH( mixNonTPMultiQubitKrausMap(qureg, targs, NUM_QUBITS, spoofOps, 1), ContainsSubstring("Invalid target qubit") );
835 }
836
837 // cannot test this with deprecated API, since 'numOps' informs a copy before validation
838
839 // SECTION( "number of operators" ) {
840
841 // int numTargs = GENERATE_COPY( range(1,maxNumTargs+1) );
842 // int numOps = GENERATE_REF( -1, 0 );
843
844 // // make valid targets to avoid triggering target validation
845 // vector<int> targs(numTargs);
846 // for (int i=0; i<numTargs; i++)
847 // targs[i] = i;
848 // REQUIRE_THROWS_WITH( mixNonTPMultiQubitKrausMap(qureg, targs.data(), numTargs, spoofOps, numOps), ContainsSubstring("must be given a strictly positive number of matrices") );
849 // }
850
851 // this validation cannot be performed by the deprecation API which copies
852 // over all ops arguments without prior checking them as NULL
853
854 // SECTION( "initialisation of operators" ) {
855
856 // /* compilers don't auto-initialise to NULL; the below circumstance
857 // * only really occurs when 'malloc' returns NULL in createComplexMatrixN,
858 // * which actually triggers its own validation. Hence this test is useless
859 // * currently.
860 // */
861
862 // int numTargs = NUM_QUBITS;
863 // int numOps = (2*numTargs)*(2*numTargs);
864
865 // // no need to initialise ops, but set their attribs correct to avoid triggering other validation
866 // vector<ComplexMatrixN> ops(numOps);
867 // for (int i=0; i<numOps; i++)
868 // ops[i].numQubits = numTargs;
869
870 // // make one of the max-ops explicitly NULL
871 // ops[GENERATE_COPY( range(0,numTargs) )].cpuElems = NULL;
872
873 // // make valid targets to avoid triggering target validation
874 // vector<int> targs(numTargs);
875 // for (int i=0; i<numTargs; i++)
876 // targs[i] = i;
877
878 // REQUIRE_THROWS_WITH( mixNonTPMultiQubitKrausMap(qureg, targs.data(), numTargs, ops.data(), numOps), ContainsSubstring("ComplexMatrixN") && ContainsSubstring("created") );
879 // }
880
881 // this validation cannot be performed by the deprecation API which copies
882 // over all ops arguments without prior checking them as matching in dimension
883
884 // SECTION( "dimension of operators" ) {
885
886 // // make valid (dimension-wise) max-qubits Kraus map
887 // int numTargs = NUM_QUBITS;
888 // int numOps = (2*numTargs)*(2*numTargs);
889 // vector<ComplexMatrixN> ops(numOps);
890 // for (int i=0; i<numOps; i++)
891 // ops[i] = createComplexMatrixN(numTargs);
892
893 // // make one have wrong-dimensions
894 // int badInd = GENERATE_COPY( range(0,numTargs) );
895 // destroyComplexMatrixN(ops[badInd]);
896 // ops[badInd] = createComplexMatrixN(numTargs - 1);
897
898 // // make valid targets to avoid triggering target validation
899 // vector<int> targs(numTargs);
900 // for (int i=0; i<numTargs; i++)
901 // targs[i] = i;
902
903 // REQUIRE_THROWS_WITH( mixNonTPMultiQubitKrausMap(qureg, targs.data(), numTargs, ops.data(), numOps), ContainsSubstring("same number of qubits") );
904
905 // for (int i=0; i<numOps; i++)
906 // destroyComplexMatrixN(ops[i]);
907 // }
908
909 SECTION( "density-matrix" ) {
910
911 Qureg statevec = createQureg(NUM_QUBITS);
912
913 // make valid targets to avoid triggering target validation
914 int targs[NUM_QUBITS];
915 for (int i=0; i<NUM_QUBITS; i++)
916 targs[i] = i;
917
918 REQUIRE_THROWS_WITH( mixNonTPMultiQubitKrausMap(statevec, targs, NUM_QUBITS, spoofOps, 1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
919 destroyQureg(statevec, getQuESTEnv());
920
921 }
922 SECTION( "operator fits in node" ) {
923
924 // each node requires (2 numTargs)^2 amplitudes
925 int minAmps = (2*NUM_QUBITS) * (2*NUM_QUBITS);
926
927 // make valid targets to avoid triggering target validation
928 int targs[NUM_QUBITS];
929 for (int i=0; i<NUM_QUBITS; i++)
930 targs[i] = i;
931
932 // make a simple Identity map
933 ComplexMatrixN ops[] = {createComplexMatrixN(NUM_QUBITS)};
934 for (int i=0; i<(1<<NUM_QUBITS); i++)
935 ops[0].cpuElems[i][i] = 1;
936 syncCompMatr(ops[0]);
937
938 // fake a smaller qureg
939 qureg.isDistributed = 1;
940 qureg.numAmpsPerNode = minAmps - 1;
941 qureg.logNumAmpsPerNode = log2(minAmps) - 1;
942 REQUIRE_THROWS_WITH( mixNonTPMultiQubitKrausMap(qureg, targs, NUM_QUBITS, ops, 1), ContainsSubstring("each node's communication buffer") && ContainsSubstring("cannot simultaneously store") );
943
944 destroyComplexMatrixN(ops[0]);
945 }
946
947 for (int i=0; i<NUM_QUBITS; i++)
948 destroyComplexMatrixN(spoofOps[i]);
949 }
950 destroyQureg(qureg, getQuESTEnv());
951}
952
953
954
955/** @sa mixNonTPTwoQubitKrausMap
956 * @ingroup deprecatedtests
957 * @author Tyson Jones
958 */
959TEST_CASE( "mixNonTPTwoQubitKrausMap", "[decoherence]" ) {
960
961 PREPARE_TEST(qureg, ref);
962
963 SECTION( "correctness" ) {
964
965 int targ1 = GENERATE( range(0,NUM_QUBITS) );
966 int targ2 = GENERATE_COPY( filter([=](int t){ return t!=targ1; }, range(0,NUM_QUBITS)) );
967 int numOps = GENERATE( range(1,17) ); // max 16 inclusive
968
969 // map consists of unconstrained 4x4 random matrices
970 vector<QMatrix> matrs;
971 for (int i=0; i<numOps; i++)
972 matrs.push_back(getRandomQMatrix(4));
973
974 vector<ComplexMatrix4> ops(numOps);
975 for (int i=0; i<numOps; i++)
976 ops[i] = toComplexMatrix4(matrs[i]);
977 mixNonTPTwoQubitKrausMap(qureg, targ1, targ2, ops.data(), numOps);
978
979 // set ref -> K_i ref K_i^dagger
980 int targs[2] = {targ1, targ2};
981 vector<QMatrix> matrRefs(numOps);
982 for (int i=0; i<numOps; i++) {
983 matrRefs[i] = ref;
984 applyReferenceOp(matrRefs[i], targs, 2, matrs[i]);
985 }
986 ref = getZeroMatrix(ref.size());
987 for (int i=0; i<numOps; i++)
988 ref += matrRefs[i];
989
990 REQUIRE( areEqual(qureg, ref, 1E4*REAL_EPS) );
991 }
992 SECTION( "input validation" ) {
993
994 // deprecated v3 API copies ops before v4 validation, so we cannot pass ops=NULL
995 ComplexMatrix4 spoofOps[1];
996
997 // cannot test this with deprecated API, since 'numOps' informs a copy before validation
998
999 // SECTION( "number of operators" ) {
1000
1001 // int numOps = GENERATE( -1, 0 );
1002 // REQUIRE_THROWS_WITH( mixNonTPTwoQubitKrausMap(qureg, 0,1, spoofOps, numOps), ContainsSubstring("must be given a strictly positive number of matrices") );
1003 // }
1004
1005 SECTION( "target collision" ) {
1006
1007 int target = GENERATE( range(0,NUM_QUBITS) );
1008 REQUIRE_THROWS_WITH( mixNonTPTwoQubitKrausMap(qureg, target, target, spoofOps, 1), ContainsSubstring("target qubits") && ContainsSubstring("unique") );
1009 }
1010 SECTION( "qubit index" ) {
1011
1012 int target = GENERATE( -1, NUM_QUBITS );
1013 REQUIRE_THROWS_WITH( mixNonTPTwoQubitKrausMap(qureg, 0,target, spoofOps, 1), ContainsSubstring("Invalid target qubit") );
1014 REQUIRE_THROWS_WITH( mixNonTPTwoQubitKrausMap(qureg, target,0, spoofOps, 1), ContainsSubstring("Invalid target qubit") );
1015 }
1016 SECTION( "density-matrix" ) {
1017
1018 Qureg vec = createQureg(NUM_QUBITS);
1019 REQUIRE_THROWS_WITH( mixNonTPTwoQubitKrausMap(vec, 0,1, spoofOps, 1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
1020 destroyQureg(vec, getQuESTEnv());
1021 }
1022 SECTION( "operators fit in node" ) {
1023
1024 qureg.isDistributed = 1;
1025 qureg.numAmpsPerNode = 15; // min 16
1026 qureg.logNumAmpsPerNode = 3; // min 4
1027 REQUIRE_THROWS_WITH( mixNonTPTwoQubitKrausMap(qureg, 0,1, spoofOps, 1), ContainsSubstring("each node's communication buffer") && ContainsSubstring("cannot simultaneously store") );
1028 }
1029 }
1030 destroyQureg(qureg, getQuESTEnv());
1031}
1032
1033
1034
1035/** @sa mixTwoQubitDephasing
1036 * @ingroup deprecatedtests
1037 * @author Tyson Jones
1038 */
1039TEST_CASE( "mixTwoQubitDephasing", "[decoherence]" ) {
1040
1041 PREPARE_TEST(qureg, ref);
1042
1043 SECTION( "correctness" ) {
1044
1045 int targ1 = GENERATE( range(0,NUM_QUBITS) );
1046 int targ2 = GENERATE_COPY( filter([=](int t){ return t!=targ1; }, range(0,NUM_QUBITS)) );
1047 qreal prob = getRandomReal(0, 3/4.);
1048
1049 mixTwoQubitDephasing(qureg, targ1, targ2, prob);
1050
1051 // ref -> (1 - prob) ref + prob/3 (Z1 ref Z1 + Z2 ref Z2 + Z1 Z2 ref Z1 Z2)
1052 QMatrix zMatr{{1,0},{0,-1}};
1053 QMatrix z1Ref = ref;
1054 applyReferenceOp(z1Ref, targ1, zMatr); // Z1 ref Z1
1055 QMatrix z2Ref = ref;
1056 applyReferenceOp(z2Ref, targ2, zMatr); // Z2 ref Z2
1057 QMatrix z1z2Ref = ref;
1058 applyReferenceOp(z1z2Ref, targ1, zMatr);
1059 applyReferenceOp(z1z2Ref, targ2, zMatr); // Z1 Z2 ref Z1 Z2
1060 ref = ((1 - prob) * ref) + (prob/3.) * (z1Ref + z2Ref + z1z2Ref);
1061
1062 REQUIRE( areEqual(qureg, ref) );
1063 }
1064 SECTION( "input validation" ) {
1065
1066 SECTION( "qubit indices" ) {
1067
1068 int targ = GENERATE( -1, NUM_QUBITS );
1069 REQUIRE_THROWS_WITH( mixTwoQubitDephasing(qureg, 0, targ, 0), ContainsSubstring("Invalid target") );
1070 REQUIRE_THROWS_WITH( mixTwoQubitDephasing(qureg, targ, 0, 0), ContainsSubstring("Invalid target") );
1071 }
1072 SECTION( "target collision" ) {
1073
1074 int targ = GENERATE( range(0,NUM_QUBITS) );
1075 REQUIRE_THROWS_WITH( mixTwoQubitDephasing(qureg, targ, targ, 0), ContainsSubstring("target") && ContainsSubstring("unique") );
1076 }
1077 SECTION( "probability" ) {
1078
1079 REQUIRE_THROWS_WITH( mixTwoQubitDephasing(qureg, 0, 1, -.1), ContainsSubstring("probability is invalid") );
1080 REQUIRE_THROWS_WITH( mixTwoQubitDephasing(qureg, 0, 1, 3/4. + .01), ContainsSubstring("probability") && ContainsSubstring("3/4") );
1081 }
1082 SECTION( "density-matrix" ) {
1083
1084 Qureg vec = createQureg(NUM_QUBITS);
1085 REQUIRE_NOTHROW( mixTwoQubitDephasing(vec, 0, 1, 0) ); // zero-prob ok in v4
1086 REQUIRE_THROWS_WITH( mixTwoQubitDephasing(vec, 0, 1, 0.1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
1087 destroyQureg(vec, getQuESTEnv());
1088 }
1089 }
1090 destroyQureg(qureg, getQuESTEnv());
1091}
1092
1093
1094
1095/** @sa mixTwoQubitDepolarising
1096 * @ingroup deprecatedtests
1097 * @author Tyson Jones
1098 */
1099TEST_CASE( "mixTwoQubitDepolarising", "[decoherence]" ) {
1100
1101 PREPARE_TEST(qureg, ref);
1102
1103 SECTION( "correctness" ) {
1104
1105 int targ1 = GENERATE( range(0,NUM_QUBITS) );
1106 int targ2 = GENERATE_COPY( filter([=](int t){ return t!=targ1; }, range(0,NUM_QUBITS)) );
1107 qreal prob = getRandomReal(0, 15/16.);
1108
1109 mixTwoQubitDepolarising(qureg, targ1, targ2, prob);
1110
1111 QMatrix paulis[4] = {
1112 QMatrix{{1,0},{0,1}}, // I
1113 QMatrix{{0,1},{1,0}}, // X
1114 QMatrix{{0,-qcomp(0,1)},{qcomp(0,1),0}}, // Y
1115 QMatrix{{1,0},{0,-1}} // Z
1116 };
1117
1118 int targs[2] = {targ1, targ2};
1119 QMatrix refInit = ref;
1120 ref = (1 - (16/15.)*prob) * ref;
1121 for (int i=0; i<4; i++) {
1122 for (int j=0; j<4; j++) {
1123 QMatrix term = refInit;
1124 QMatrix op = getKroneckerProduct(paulis[i], paulis[j]);
1125 applyReferenceOp(term, targs, 2, op);
1126 ref += (prob/15.) * term;
1127 }
1128 }
1129
1130 REQUIRE( areEqual(qureg, ref, 1E4*REAL_EPS) );
1131 }
1132 SECTION( "input validation" ) {
1133
1134 SECTION( "qubit indices" ) {
1135
1136 int targ = GENERATE( -1, NUM_QUBITS );
1137 REQUIRE_THROWS_WITH( mixTwoQubitDepolarising(qureg, 0, targ, 0), ContainsSubstring("Invalid target") );
1138 REQUIRE_THROWS_WITH( mixTwoQubitDepolarising(qureg, targ, 0, 0), ContainsSubstring("Invalid target") );
1139 }
1140 SECTION( "target collision" ) {
1141
1142 int targ = GENERATE( range(0,NUM_QUBITS) );
1143 REQUIRE_THROWS_WITH( mixTwoQubitDepolarising(qureg, targ, targ, 0), ContainsSubstring("target") && ContainsSubstring("unique") );
1144 }
1145 SECTION( "probability" ) {
1146
1147 REQUIRE_THROWS_WITH( mixTwoQubitDepolarising(qureg, 0, 1, -.1), ContainsSubstring("probability is invalid") );
1148 REQUIRE_THROWS_WITH( mixTwoQubitDepolarising(qureg, 0, 1, 15/16. + .01), ContainsSubstring("probability") && ContainsSubstring("15/16") );
1149 }
1150 SECTION( "density-matrix" ) {
1151
1152 Qureg vec = createQureg(NUM_QUBITS);
1153 REQUIRE_NOTHROW( mixTwoQubitDepolarising(vec, 0, 1, 0) ); // zero prob ok in v4
1154 REQUIRE_THROWS_WITH( mixTwoQubitDepolarising(vec, 0, 1, 0.1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
1155 destroyQureg(vec, getQuESTEnv());
1156 }
1157 }
1158 destroyQureg(qureg, getQuESTEnv());
1159}
1160
1161
1162
1163/** @sa mixTwoQubitKrausMap
1164 * @ingroup deprecatedtests
1165 * @author Tyson Jones
1166 */
1167TEST_CASE( "mixTwoQubitKrausMap", "[decoherence]" ) {
1168
1169 PREPARE_TEST(qureg, ref);
1170
1171 SECTION( "correctness" ) {
1172
1173 int targ1 = GENERATE( range(0,NUM_QUBITS) );
1174 int targ2 = GENERATE_COPY( filter([=](int t){ return t!=targ1; }, range(0,NUM_QUBITS)) );
1175 int numOps = GENERATE( range(1,17) ); // max 16 inclusive
1176 vector<QMatrix> matrs = getRandomKrausMap(2, numOps);
1177
1178 vector<ComplexMatrix4> ops(numOps);
1179 for (int i=0; i<numOps; i++)
1180 ops[i] = toComplexMatrix4(matrs[i]);
1181 mixTwoQubitKrausMap(qureg, targ1, targ2, ops.data(), numOps);
1182
1183 // set ref -> K_i ref K_i^dagger
1184 int targs[2] = {targ1, targ2};
1185 vector<QMatrix> matrRefs(numOps);
1186 for (int i=0; i<numOps; i++) {
1187 matrRefs[i] = ref;
1188 applyReferenceOp(matrRefs[i], targs, 2, matrs[i]);
1189 }
1190 ref = getZeroMatrix(ref.size());
1191 for (int i=0; i<numOps; i++)
1192 ref += matrRefs[i];
1193
1194 REQUIRE( areEqual(qureg, ref, 10*REAL_EPS) );
1195 }
1196 SECTION( "input validation" ) {
1197
1198 ComplexMatrix4 emptyOps[1];
1199
1200 SECTION( "number of operators" ) {
1201
1202 // in v4, a separate createKausMap call is made which throws the below error
1203 int numOps = 0;
1204 REQUIRE_THROWS_WITH( mixTwoQubitKrausMap(qureg, 0,1, emptyOps, numOps), ContainsSubstring("must be given a strictly positive number of matrices") );
1205 }
1206 SECTION( "trace preserving" ) {
1207
1208 // valid Kraus map
1209 int numOps = GENERATE( range(1,16) );
1210 vector<QMatrix> matrs = getRandomKrausMap(2, numOps);
1211 vector<ComplexMatrix4> ops(numOps);
1212 for (int i=0; i<numOps; i++)
1213 ops[i] = toComplexMatrix4(matrs[i]);
1214
1215 // make only one of the ops at a time invalid
1216 ops[GENERATE_REF( range(0,numOps) )].real[0][0] = 999;
1217
1218 REQUIRE_THROWS_WITH( mixTwoQubitKrausMap(qureg, 0,1, ops.data(), numOps), ContainsSubstring("trace preserving") );
1219 }
1220 SECTION( "target collision" ) {
1221
1222 int target = GENERATE( range(0,NUM_QUBITS) );
1223 REQUIRE_THROWS_WITH( mixTwoQubitKrausMap(qureg, target, target, emptyOps, 1), ContainsSubstring("target qubits") && ContainsSubstring("unique") );
1224 }
1225 SECTION( "qubit index" ) {
1226
1227 int target = GENERATE( -1, NUM_QUBITS );
1228 REQUIRE_THROWS_WITH( mixTwoQubitKrausMap(qureg, 0,target, emptyOps, 1), ContainsSubstring("Invalid target qubit") );
1229 REQUIRE_THROWS_WITH( mixTwoQubitKrausMap(qureg, target,0, emptyOps, 1), ContainsSubstring("Invalid target qubit") );
1230 }
1231 SECTION( "density-matrix" ) {
1232
1233 Qureg vec = createQureg(NUM_QUBITS);
1234 REQUIRE_THROWS_WITH( mixTwoQubitKrausMap(vec, 0,1, emptyOps, 1), ContainsSubstring("Expected a density matrix Qureg but received a statevector") );
1235 destroyQureg(vec, getQuESTEnv());
1236 }
1237 SECTION( "operators fit in node" ) {
1238
1239 qureg.isDistributed = 1;
1240 qureg.numAmpsPerNode = 15; // min 16
1241 qureg.logNumAmpsPerNode = 3; // min 4
1242 REQUIRE_THROWS_WITH( mixTwoQubitKrausMap(qureg, 0,1, emptyOps, 1), ContainsSubstring("each node's communication buffer") && ContainsSubstring("cannot simultaneously store") );
1243 }
1244 }
1245 destroyQureg(qureg, getQuESTEnv());
1246}
void mixDepolarising(Qureg qureg, int qubit, qreal prob)
void mixTwoQubitDephasing(Qureg qureg, int qubit1, int qubit2, qreal prob)
void mixDamping(Qureg qureg, int qubit, qreal prob)
void mixTwoQubitDepolarising(Qureg qureg, int qubit1, int qubit2, qreal prob)
void mixDephasing(Qureg qureg, int qubit, qreal prob)
TEST_CASE("mixDamping", "[decoherence]")
ComplexMatrix4 toComplexMatrix4(QMatrix qm)
unsigned int calcLog2(long unsigned int res)
bool areEqual(QVector a, QVector b)
vector< vector< qcomp > > QMatrix
ComplexMatrix2 toComplexMatrix2(QMatrix qm)
void applyReferenceOp(QVector &state, int *ctrls, int numCtrls, int *targs, int numTargs, QMatrix op)
void toComplexMatrixN(QMatrix qm, ComplexMatrixN cm)
QMatrix getRandomQMatrix(int dim)
void setRandomTargets(int *targs, int numTargs, int numQb)
QMatrix toQMatrix(CompMatr1 src)
QuESTEnv getQuESTEnv()
void initDebugState(Qureg qureg)
CompMatr createCompMatr(int numQubits)
Definition matrices.cpp:211
void destroyCompMatr(CompMatr matrix)
Definition matrices.cpp:402
void setCompMatr(CompMatr matr, qcomp **vals)
Definition matrices.cpp:427
void syncCompMatr(CompMatr matr)
Definition matrices.cpp:377
Qureg createDensityQureg(int numQubits)
Definition qureg.cpp:285
Qureg createForcedDensityQureg(int numQubits)
Definition qureg.cpp:303
Qureg createQureg(int numQubits)
Definition qureg.cpp:277
void destroyQureg(Qureg qureg)
Definition qureg.cpp:328
qmatrix getKroneckerProduct(qmatrix a, qmatrix b)
Definition linalg.cpp:523
qmatrix getZeroMatrix(size_t dim)
Definition qmatrix.cpp:18
qreal getRandomReal(qreal min, qreal maxIncl)
Definition random.cpp:60
vector< qmatrix > getRandomKrausMap(int numQb, int numOps)
Definition random.cpp:383
Definition qureg.h:42