The Quantum Exact Simulation Toolkit v4.0.0
Loading...
Searching...
No Matches
debug.cpp
1/** @file
2 * Unit tests of the debug module.
3 *
4 * @author Tyson Jones
5 *
6 * @defgroup unitdebug Debug
7 * @ingroup unittests
8 */
9
10#include "quest/include/quest.h"
11
12#include <catch2/catch_test_macros.hpp>
13#include <catch2/matchers/catch_matchers_string.hpp>
14#include <catch2/generators/catch_generators_range.hpp>
15
16#include "tests/utils/macros.hpp"
17#include "tests/utils/cache.hpp"
18#include "tests/utils/convert.hpp"
19#include "tests/utils/compare.hpp"
20#include "tests/utils/random.hpp"
21
22#include <string>
23#include <vector>
24#include <sstream>
25#include <iostream>
26
27using Catch::Matchers::ContainsSubstring;
28using std::vector;
29
30
31
32/*
33 * UTILITIES
34 */
35
36#define TEST_CATEGORY \
37 LABEL_UNIT_TAG "[debug]"
38
39
40
41/**
42 * TESTS
43 *
44 * @ingroup unitdebug
45 * @{
46 */
47
48
49TEST_CASE( "setMaxNumReportedSigFigs", TEST_CATEGORY ) {
50
51 SECTION( LABEL_CORRECTNESS ) {
52
53 qcomp scalar = getQcomp(0.12345, 0.12345);
54
55 vector<std::string> refs = {
56 "0.1+0.1i",
57 "0.12+0.12i",
58 "0.123+0.123i",
59 "0.1235+0.1235i", // rounded
60 "0.12345+0.12345i"
61 };
62
63 // disable auto \n after lines
65
66 for (size_t numSigFigs=1; numSigFigs<=refs.size(); numSigFigs++) {
67
68 setMaxNumReportedSigFigs(numSigFigs);
69
70 // redirect stdout to buffer
71 std::stringstream buffer;
72 std::streambuf * old = std::cout.rdbuf(buffer.rdbuf());
73 reportScalar("", scalar);
74 std::cout.rdbuf(old);
75 std::string out = buffer.str();
76
77 std::string ref = refs[numSigFigs-1];
78
79 CAPTURE( numSigFigs, ref );
80 REQUIRE( out == ref );
81 }
82 }
83
84 SECTION( LABEL_VALIDATION ) {
85
86 SECTION( "number" ) {
87
88 int num = GENERATE( -1, 0 );
89
90 REQUIRE_THROWS_WITH( setMaxNumReportedSigFigs(num), ContainsSubstring("Cannot be less than one") );
91 }
92 }
93
94 // restore to QuEST default for future tests
96}
97
98
99TEST_CASE( "setNumReportedNewlines", TEST_CATEGORY ) {
100
101 SECTION( LABEL_CORRECTNESS ) {
102
103 for (int numNewlines=0; numNewlines<3; numNewlines++) {
104
105 setNumReportedNewlines(numNewlines);
106
107 // redirect stdout to buffer
108 std::stringstream buffer;
109 std::streambuf * old = std::cout.rdbuf(buffer.rdbuf());
110 reportStr("x");
111 std::cout.rdbuf(old);
112 std::string out = buffer.str();
113
114 std::string ref = "x" + std::string(numNewlines, '\n');
115
116 CAPTURE( numNewlines, ref );
117 REQUIRE( out == ref );
118 }
119 }
120
121 SECTION( LABEL_VALIDATION ) {
122
123 SECTION( "number" ) {
124
125 REQUIRE_THROWS_WITH( setNumReportedNewlines(-1), ContainsSubstring("Cannot generally be less than zero") );
126 }
127
128 SECTION( "multine number" ) {
129
131
132 REQUIRE_THROWS_WITH( reportQuESTEnv(), ContainsSubstring("zero") && ContainsSubstring("not permitted when calling multi-line") );
133 }
134 }
135
136 // restore to QuEST default for future tests
138}
139
140
141TEST_CASE( "setSeeds", TEST_CATEGORY ) {
142
143 SECTION( LABEL_CORRECTNESS ) {
144
145 // perform the below test for every combination
146 // of deployments, since their RNGs differ
147
148 for (auto& [label, qureg]: getCachedDensmatrs()) {
149
150 DYNAMIC_SECTION( label ) {
151
152 SECTION( "same seed consistency" ) {
153
154 unsigned seeds[] = {123, 543, 755};
155 const int numSeeds = 3;
156 const int numMixedStates = 10;
157 const int numReps = 5;
158
159 // set an arbitrary fixed seed...
160 setSeeds(seeds, numSeeds);
161
162 // generate and remember a random state
163 initRandomMixedState(qureg, numMixedStates);
164 qmatrix ref = getMatrix(qureg);
165
166 // get a set of random measurement outcomes
167 vector<int> outcomes(qureg.numQubits);
168 for (int i=0; i<qureg.numQubits; i++)
169 outcomes[i] = applyQubitMeasurement(qureg, i);
170
171 // repeatedly...
172 for (int r=0; r<numReps; r++) {
173
174 // reset the seed
175 setSeeds(seeds, numSeeds);
176
177 // and confirm all random states are re-produced
178 initRandomMixedState(qureg, numMixedStates);
179 REQUIRE_AGREE( qureg, ref);
180
181 // as are all measurement outcomes
182 for (int i=0; i<qureg.numQubits; i++)
183 REQUIRE( outcomes[i] == applyQubitMeasurement(qureg,i) );
184 }
185 }
186
187 SECTION( "different key inconsistency" ) {
188
189 unsigned seeds[] = {123, 543, 755};
190 const int numSeeds = 3;
191 const int ampInd = 0;
192
193 // set arbitrary seed and collect random-state amp
194 setSeeds(seeds, numSeeds);
195 initRandomPureState(qureg);
196 qcomp amp1 = getDensityQuregAmp(qureg, ampInd, ampInd);
197
198 // change one passed seed and re-collect random-state amp
199 int i = GENERATE_COPY( range(0,numSeeds) );
200 seeds[i] = 987654321;
201 setSeeds(seeds, numSeeds);
202 initRandomPureState(qureg);
203 qcomp amp2 = getDensityQuregAmp(qureg, ampInd, ampInd);
204
205 // amps differ if seed successfully impacts outcomes
206 REQUIRE( amp1 != amp2 );
207 }
208 }
209 }
210 }
211
212 SECTION( LABEL_VALIDATION ) {
213
214 SECTION( "env not initialised" ) {
215
216 // no way to test this
217 SUCCEED( );
218 }
219
220 SECTION( "number of seeds" ) {
221
222 int numSeeds = GENERATE( -1, 0 );
223
224 REQUIRE_THROWS_WITH( setSeeds(nullptr, numSeeds), ContainsSubstring("Invalid number of random seeds") );
225 }
226
227 // inconsistency between nodes is permitted
228 }
229
230 // re-randomise seeds for remaining tests
232}
233
234
235TEST_CASE( "setSeedsToDefault", TEST_CATEGORY ) {
236
237 SECTION( LABEL_CORRECTNESS ) {
238
239 // perform the below test for every combination
240 // of deployments, since their RNGs differ
241
242 for (auto& [label, qureg]: getCachedDensmatrs()) {
243
244 DYNAMIC_SECTION( label ) {
245
246 SECTION( "different key inconsistency" ) {
247
248 const int ampInd = 0;
249
250 // randomise seed and collect random-state amp
252 initRandomPureState(qureg);
253 qcomp amp1 = getDensityQuregAmp(qureg, ampInd, ampInd);
254
255 // re-randomise seed and collect new random-state amp
257 initRandomPureState(qureg);
258 qcomp amp2 = getDensityQuregAmp(qureg, ampInd, ampInd);
259
260 // amps differ if seed re-randomisation impacts outcomes
261 REQUIRE( amp1 != amp2 );
262 }
263 }
264 }
265 }
266
267 SECTION( LABEL_VALIDATION ) {
268
269 SECTION( "env not initialised" ) {
270
271 // no way to test this
272 SUCCEED( );
273 }
274 }
275
276 // re-randomise seeds for remaining tests
278}
279
280
281TEST_CASE( "getSeeds", TEST_CATEGORY ) {
282
283 SECTION( LABEL_CORRECTNESS ) {
284
285 SECTION( "can be called immediately" ) {
286
287 REQUIRE_NOTHROW( getNumSeeds() );
288
289 int numSeeds = getNumSeeds();
290 vector<unsigned> out(numSeeds);
291
292 REQUIRE_NOTHROW( getSeeds(out.data()) );
293 }
294
295 SECTION( "correct output" ) {
296
297 GENERATE( range(0,10) );
298
299 // prepare random seeds (using test utils RNG, not QuEST's)
300 int numSeeds = getRandomInt(1, 10);
301 vector<unsigned> in(numSeeds);
302 for (int i=0; i<numSeeds; i++)
303 in[i] = static_cast<unsigned>(getRandomInt(0, 99999));
304
305 // pass seeds to QuEST
306 setSeeds(in.data(), numSeeds);
307
308 // check we get them back
309 vector<unsigned> out(numSeeds);
310 getSeeds(out.data());
311 for (int i=0; i<numSeeds; i++)
312 REQUIRE( in[i] == out[i] );
313 }
314 }
315
316 SECTION( LABEL_VALIDATION ) {
317
318 SECTION( "env not initialised" ) {
319
320 // no way to test this
321 SUCCEED( );
322 }
323 }
324
325 // re-randomise seeds for remaining tests
327}
328
329
330TEST_CASE( "getNumSeeds", TEST_CATEGORY ) {
331
332 SECTION( LABEL_CORRECTNESS ) {
333
334 SECTION( "can be called immediately" ) {
335
336 REQUIRE_NOTHROW( getNumSeeds() );
337 }
338
339 SECTION( "correct output" ) {
340
341 GENERATE( range(0,10) );
342
343 // prepare random seeds (using test utils RNG, not QuEST's)
344 int numSeeds = getRandomInt(1, 10);
345 vector<unsigned> in(numSeeds);
346 for (int i=0; i<numSeeds; i++)
347 in[i] = static_cast<unsigned>(getRandomInt(0, 99999));
348
349 // pass seeds to QuEST
350 setSeeds(in.data(), numSeeds);
351
352 // confirm we get out correct number
353 REQUIRE( getNumSeeds() == numSeeds );
354 }
355 }
356
357 SECTION( LABEL_VALIDATION ) {
358
359 SECTION( "env not initialised" ) {
360
361 // no way to test this
362 SUCCEED( );
363 }
364 }
365
366 // re-randomise seeds for remaining tests
368}
369
370
371TEST_CASE( "invalidQuESTInputError", TEST_CATEGORY ) {
372
373 SECTION( LABEL_CORRECTNESS ) {
374
375 REQUIRE_THROWS_WITH( invalidQuESTInputError("msg", "func"), ContainsSubstring("msg") && ContainsSubstring("func") );
376 }
377
378 SECTION( LABEL_VALIDATION ) {
379
380 // has no validation
381 SUCCEED( );
382 }
383}
384
385
386TEST_CASE( "setValidationOn", TEST_CATEGORY ) {
387
388 SECTION( LABEL_CORRECTNESS ) {
389
390 // always safe to call
391 for (int i=0; i<3; i++)
392 REQUIRE_NOTHROW( setValidationOn() );
393
394 // illegal and caught
395 REQUIRE_THROWS( setSeeds(nullptr, -99) );
396 }
397
398 SECTION( LABEL_VALIDATION ) {
399
400 // has no validation
401 SUCCEED( );
402 }
403}
404
405
406TEST_CASE( "setValidationOff", TEST_CATEGORY ) {
407
408 SECTION( LABEL_CORRECTNESS ) {
409
410 // confirm always safe to call
411 for (int i=0; i<3; i++)
412 REQUIRE_NOTHROW( setValidationOff() );
413
414 // prepare non-unitary matrix
415 CompMatr1 m = getCompMatr1({{1,2},{3,4}});
416 Qureg qureg = createQureg(1);
417
418 // confirm not-unitary error suppressed...
419 REQUIRE_NOTHROW( applyCompMatr1(qureg, 0, m) );
420
421 // which otherwise triggers
423 REQUIRE_THROWS( applyCompMatr1(qureg, 0, m) );
424
425 destroyQureg(qureg);
426 }
427
428 SECTION( LABEL_VALIDATION ) {
429
430 // has no validation
431 SUCCEED( );
432 }
433
434 // ensure validation is on for remaining tests
436}
437
438
439TEST_CASE( "setValidationEpsilon", TEST_CATEGORY ) {
440
441 SECTION( LABEL_CORRECTNESS ) {
442
443 SECTION( "affects validation" ) {
444
445 Qureg qureg = createQureg(5);
446
447 SECTION( "unitarity" ) {
448
449 // prepare non-unitary matrix
450 CompMatr1 m = getCompMatr1({{1,2},{3,4}});
451
452 // confirm it throws non-unitary error
453 REQUIRE_THROWS( applyCompMatr1(qureg, 0, m) );
454
455 // confirm setting = 0 disables epsilon errors...
457 REQUIRE_NOTHROW( applyCompMatr1(qureg, 0, m) );
458
459 // but does not disable absolute errors
460 REQUIRE_THROWS( applyCompMatr1(qureg, -1, m) );
461
462 // confirm non-zero (forgive all) works
463 setValidationEpsilon(9999); // bigger than dist of m*conj(m) from identity squared
464 REQUIRE_NOTHROW( applyCompMatr1(qureg, 0, m) );
465 }
466
467 /// @todo
468 /// to be completely rigorous, we should test
469 /// that unitarity, hermiticity, CPTPness and
470 /// non-zero-ness of all of CompMatr, DiagMatr
471 /// and FullStateDiagMatr are affected! This
472 /// is quite a chore of course
473
474 destroyQureg(qureg);
475 }
476
477 SECTION( "affects struct fields" ) {
478
479 SECTION( "CompMatr" ) {
480
482 *(m.isApproxUnitary) = 1;
483 *(m.isApproxHermitian) = 1;
484
486 REQUIRE( *(m.isApproxUnitary) == -1 );
487 REQUIRE( *(m.isApproxHermitian) == -1 );
488
490 }
491
492 SECTION( "DiagMatr" ) {
493
495 *(m.isApproxUnitary) = 1;
496 *(m.isApproxHermitian) = 0;
497 *(m.isApproxNonZero) = 1;
498
500 REQUIRE( *(m.isApproxUnitary) == -1 );
501 REQUIRE( *(m.isApproxHermitian) == -1 );
502 REQUIRE( *(m.isApproxNonZero) == -1 );
503
505 }
506
507 SECTION( "FullStateDiagMatr" ) {
508
510 *(m.isApproxUnitary) = 1;
511 *(m.isApproxHermitian) = 0;
512 *(m.isApproxNonZero) = 1;
513
515 REQUIRE( *(m.isApproxUnitary) == -1 );
516 REQUIRE( *(m.isApproxHermitian) == -1 );
517 REQUIRE( *(m.isApproxNonZero) == -1 );
518
520 }
521
522 SECTION( "KrausMap" ) {
523
524 KrausMap k = createKrausMap(1, 3);
525 *(k.isApproxCPTP) = 1;
526
528 REQUIRE( *(k.isApproxCPTP) == -1 );
529
531 }
532 }
533 }
534
535 SECTION( LABEL_VALIDATION ) {
536
537 SECTION( "negative epsilon" ) {
538
539 qreal eps = GENERATE( -0.5, -1, -100 );
540
541 REQUIRE_THROWS_WITH( setValidationEpsilon(eps), ContainsSubstring("positive number") );
542 }
543 }
544
545 // ensure validation epsilon is default for remaining tests
547}
548
549
550TEST_CASE( "getValidationEpsilon", TEST_CATEGORY ) {
551
552 SECTION( LABEL_CORRECTNESS ) {
553
554 // confirm always safe to call
555 for (int i=0; i<3; i++)
556 REQUIRE_NOTHROW( getValidationEpsilon() ); // ignores output
557
558 GENERATE( range(0,10) );
559
560 // confirm set correctly
561 qreal eps = getRandomReal(0, 99999);
563
564 REQUIRE( getValidationEpsilon() == eps );
565 }
566
567 SECTION( LABEL_VALIDATION ) {
568
569 // has no validation
570 SUCCEED( );
571 }
572
573 // ensure validation epsilon is default for remaining tests
575}
576
577
578TEST_CASE( "setValidationEpsilonToDefault", TEST_CATEGORY ) {
579
580 SECTION( LABEL_CORRECTNESS ) {
581
582 SECTION( "always safe to call" ) {
583
584 for (int i=0; i<3; i++)
585 REQUIRE_NOTHROW( setValidationEpsilonToDefault() );
586 }
587
588 SECTION( "affects validation" ) {
589
590 // prepare non-unitary matrix
591 CompMatr1 m = getCompMatr1({{1,2},{3,4}});
592 Qureg qureg = createQureg(1);
593
594 // confirm it throws non-unitary error
595 REQUIRE_THROWS( applyCompMatr1(qureg, 0, m) );
596
597 // confirm setting = 0 disables epsilon errors...
599 REQUIRE_NOTHROW( applyCompMatr1(qureg, 0, m) );
600
601 // which returns when stored to default
603 REQUIRE_THROWS( applyCompMatr1(qureg, 0, m) );
604
605 destroyQureg(qureg);
606 }
607
608 SECTION( "affects struct fields" ) {
609
610 SECTION( "CompMatr" ) {
611
613 *(m.isApproxUnitary) = 1;
614 *(m.isApproxHermitian) = 1;
615
617 REQUIRE( *(m.isApproxUnitary) == -1 );
618 REQUIRE( *(m.isApproxHermitian) == -1 );
619
621 }
622
623 SECTION( "DiagMatr" ) {
624
626 *(m.isApproxUnitary) = 1;
627 *(m.isApproxHermitian) = 0;
628 *(m.isApproxNonZero) = 1;
629
631 REQUIRE( *(m.isApproxUnitary) == -1 );
632 REQUIRE( *(m.isApproxHermitian) == -1 );
633 REQUIRE( *(m.isApproxNonZero) == -1 );
634
636 }
637
638 SECTION( "FullStateDiagMatr" ) {
639
641 *(m.isApproxUnitary) = 1;
642 *(m.isApproxHermitian) = 0;
643 *(m.isApproxNonZero) = 1;
644
646 REQUIRE( *(m.isApproxUnitary) == -1 );
647 REQUIRE( *(m.isApproxHermitian) == -1 );
648 REQUIRE( *(m.isApproxNonZero) == -1 );
649
651 }
652
653 SECTION( "KrausMap" ) {
654
655 KrausMap k = createKrausMap(1, 3);
656 *(k.isApproxCPTP) = 1;
657
659 REQUIRE( *(k.isApproxCPTP) == -1 );
660
662 }
663 }
664 }
665
666 SECTION( LABEL_VALIDATION ) {
667
668 SECTION( LABEL_VALIDATION ) {
669
670 // performs no validation
671 SUCCEED( );
672 }
673 }
674
675 // ensure validation epsilon is default for remaining tests
677}
678
679
680TEST_CASE( "getGpuCacheSize", TEST_CATEGORY ) {
681
682 SECTION( LABEL_CORRECTNESS ) {
683
684 // confirm cache begins empty
686 REQUIRE( getGpuCacheSize() == 0 );
687
688 // hackily detect cuQuantum
689 char envStr[200];
690 getEnvironmentString(envStr);
691 bool usingCuQuantum = std::string(envStr).find("cuQuantum=0") == std::string::npos;
692
693 // proceed only if we're ever using our own GPU cache
694 if (getQuESTEnv().isGpuAccelerated && !usingCuQuantum) {
695
696 // GPU cache created for >= 6 qubit matrices
697 Qureg qureg = createCustomQureg(10, 0, 0,1,0); // gpu only
698 CompMatr matr = createCompMatr(6); // always in gpu
699 for (qindex i=0; i<matr.numRows; i++)
700 matr.cpuElems[i][i] = 1;
701 syncCompMatr(matr);
702
703 // each control qubit halves the necessary cache size;
704 // so we start with many controls and remove each in-turn,
705 // expanding the GPU cache in every call
706 int targs[] = {0,1,2,3,4,5};
707 int ctrls[] = {6,7,8,9};
708 qindex cacheSize = 0;
709
710 for (int numCtrls=4; numCtrls>=0; numCtrls--) {
711
712 // expand the cache
713 applyMultiControlledCompMatr(qureg, ctrls, numCtrls, targs, 6, matr);
714
715 // confirm it expanded, OR stayed the same, which happens when
716 // the total number of simultaneous threads needed hits/exceeds
717 // the number available in the hardware
718 qindex newSize = getGpuCacheSize();
719 CAPTURE( cacheSize, newSize );
720 REQUIRE( newSize >= cacheSize );
721
722 cacheSize = newSize;
723 }
724
725 destroyQureg(qureg);
726 destroyCompMatr(matr);
727 }
728 }
729
730 SECTION( LABEL_VALIDATION ) {
731
732 // performs no validation
733 SUCCEED( );
734 }
735}
736
737
738/** @} (end defgroup) */
739
740
741
742/**
743 * @todo
744 * UNTESTED FUNCTIONS
745 */
746
747
748void setMaxNumReportedItems(qindex numRows, qindex numCols);
749
750void getEnvironmentString(char str[200]);
751
KrausMap createKrausMap(int numQubits, int numOperators)
Definition channels.cpp:172
void destroyKrausMap(KrausMap map)
Definition channels.cpp:208
qindex getGpuCacheSize()
Definition debug.cpp:152
void clearGpuCache()
Definition debug.cpp:163
void getEnvironmentString(char str[200])
void setMaxNumReportedItems(qindex numRows, qindex numCols)
Definition debug.cpp:116
void setMaxNumReportedSigFigs(int numSigFigs)
Definition debug.cpp:130
void setNumReportedNewlines(int numNewlines)
Definition debug.cpp:138
void setSeedsToDefault()
Definition debug.cpp:38
int getNumSeeds()
Definition debug.cpp:45
void setSeeds(unsigned *seeds, int numSeeds)
Definition debug.cpp:30
void getSeeds(unsigned *seeds)
Definition debug.cpp:51
void setValidationOff()
Definition debug.cpp:74
void setValidationEpsilonToDefault()
Definition debug.cpp:96
void invalidQuESTInputError(const char *msg, const char *func)
Definition main.cpp:62
void setValidationOn()
Definition debug.cpp:68
qreal getValidationEpsilon()
Definition debug.cpp:103
void setValidationEpsilon(qreal eps)
Definition debug.cpp:88
void reportQuESTEnv()
QuESTEnv getQuESTEnv()
void initRandomPureState(Qureg qureg)
void initRandomMixedState(Qureg qureg, qindex numPureStates)
FullStateDiagMatr createFullStateDiagMatr(int numQubits)
Definition matrices.cpp:327
CompMatr createCompMatr(int numQubits)
Definition matrices.cpp:211
DiagMatr createDiagMatr(int numQubits)
Definition matrices.cpp:246
void destroyDiagMatr(DiagMatr matrix)
Definition matrices.cpp:403
void destroyFullStateDiagMatr(FullStateDiagMatr matrix)
Definition matrices.cpp:404
void destroyCompMatr(CompMatr matrix)
Definition matrices.cpp:402
static CompMatr1 getCompMatr1(qcomp **in)
Definition matrices.h:304
void syncCompMatr(CompMatr matr)
Definition matrices.cpp:377
void applyCompMatr1(Qureg qureg, int target, CompMatr1 matrix)
void applyMultiControlledCompMatr(Qureg qureg, int *controls, int numControls, int *targets, int numTargets, CompMatr matr)
int applyQubitMeasurement(Qureg qureg, int target)
Qureg createCustomQureg(int numQubits, int isDensMatr, int useDistrib, int useGpuAccel, int useMultithread)
Definition qureg.cpp:271
Qureg createQureg(int numQubits)
Definition qureg.cpp:277
void destroyQureg(Qureg qureg)
Definition qureg.cpp:328
qcomp getDensityQuregAmp(Qureg qureg, qindex row, qindex column)
Definition qureg.cpp:496
qreal getRandomReal(qreal min, qreal maxIncl)
Definition random.cpp:60
int getRandomInt(int min, int maxExcl)
Definition random.cpp:76
void reportStr(const char *str)
Definition types.cpp:29
static qcomp getQcomp(qreal re, qreal im)
Definition types.h:91
void reportScalar(const char *label, qcomp num)
Definition types.cpp:51
TEST_CASE("setMaxNumReportedSigFigs", TEST_CATEGORY)
Definition debug.cpp:49
int * isApproxNonZero
Definition matrices.h:159
Definition qureg.h:42