10#include "quest/include/quest.h"
12#include <catch2/catch_test_macros.hpp>
13#include <catch2/matchers/catch_matchers_string.hpp>
14#include <catch2/generators/catch_generators_range.hpp>
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"
27using Catch::Matchers::ContainsSubstring;
36#define TEST_CATEGORY \
37 LABEL_UNIT_TAG "[debug]"
49TEST_CASE(
"setMaxNumReportedSigFigs", TEST_CATEGORY ) {
51 SECTION( LABEL_CORRECTNESS ) {
53 qcomp scalar =
getQcomp(0.12345, 0.12345);
55 vector<std::string> refs = {
66 for (
size_t numSigFigs=1; numSigFigs<=refs.size(); numSigFigs++) {
71 std::stringstream buffer;
72 std::streambuf * old = std::cout.rdbuf(buffer.rdbuf());
75 std::string out = buffer.str();
77 std::string ref = refs[numSigFigs-1];
79 CAPTURE( numSigFigs, ref );
80 REQUIRE( out == ref );
84 SECTION( LABEL_VALIDATION ) {
88 int num = GENERATE( -1, 0 );
99TEST_CASE(
"setNumReportedNewlines", TEST_CATEGORY ) {
101 SECTION( LABEL_CORRECTNESS ) {
103 for (
int numNewlines=0; numNewlines<3; numNewlines++) {
108 std::stringstream buffer;
109 std::streambuf * old = std::cout.rdbuf(buffer.rdbuf());
111 std::cout.rdbuf(old);
112 std::string out = buffer.str();
114 std::string ref =
"x" + std::string(numNewlines,
'\n');
116 CAPTURE( numNewlines, ref );
117 REQUIRE( out == ref );
121 SECTION( LABEL_VALIDATION ) {
123 SECTION(
"number" ) {
125 REQUIRE_THROWS_WITH(
setNumReportedNewlines(-1), ContainsSubstring(
"Cannot generally be less than zero") );
128 SECTION(
"multine number" ) {
132 REQUIRE_THROWS_WITH(
reportQuESTEnv(), ContainsSubstring(
"zero") && ContainsSubstring(
"not permitted when calling multi-line") );
143 SECTION( LABEL_CORRECTNESS ) {
148 for (
auto& [label, qureg]: getCachedDensmatrs()) {
150 DYNAMIC_SECTION( label ) {
152 SECTION(
"same seed consistency" ) {
154 unsigned seeds[] = {123, 543, 755};
155 const int numSeeds = 3;
156 const int numMixedStates = 10;
157 const int numReps = 5;
164 qmatrix ref = getMatrix(qureg);
167 vector<int> outcomes(qureg.numQubits);
168 for (
int i=0; i<qureg.numQubits; i++)
172 for (
int r=0; r<numReps; r++) {
179 REQUIRE_AGREE( qureg, ref);
182 for (
int i=0; i<qureg.numQubits; i++)
187 SECTION(
"different key inconsistency" ) {
189 unsigned seeds[] = {123, 543, 755};
190 const int numSeeds = 3;
191 const int ampInd = 0;
199 int i = GENERATE_COPY( range(0,numSeeds) );
200 seeds[i] = 987654321;
206 REQUIRE( amp1 != amp2 );
212 SECTION( LABEL_VALIDATION ) {
214 SECTION(
"env not initialised" ) {
220 SECTION(
"number of seeds" ) {
222 int numSeeds = GENERATE( -1, 0 );
224 REQUIRE_THROWS_WITH(
setSeeds(
nullptr, numSeeds), ContainsSubstring(
"Invalid number of random seeds") );
235TEST_CASE(
"setSeedsToDefault", TEST_CATEGORY ) {
237 SECTION( LABEL_CORRECTNESS ) {
242 for (
auto& [label, qureg]: getCachedDensmatrs()) {
244 DYNAMIC_SECTION( label ) {
246 SECTION(
"different key inconsistency" ) {
248 const int ampInd = 0;
261 REQUIRE( amp1 != amp2 );
267 SECTION( LABEL_VALIDATION ) {
269 SECTION(
"env not initialised" ) {
283 SECTION( LABEL_CORRECTNESS ) {
285 SECTION(
"can be called immediately" ) {
290 vector<unsigned> out(numSeeds);
292 REQUIRE_NOTHROW(
getSeeds(out.data()) );
295 SECTION(
"correct output" ) {
297 GENERATE( range(0,10) );
301 vector<unsigned> in(numSeeds);
302 for (
int i=0; i<numSeeds; i++)
309 vector<unsigned> out(numSeeds);
311 for (
int i=0; i<numSeeds; i++)
312 REQUIRE( in[i] == out[i] );
316 SECTION( LABEL_VALIDATION ) {
318 SECTION(
"env not initialised" ) {
330TEST_CASE(
"getNumSeeds", TEST_CATEGORY ) {
332 SECTION( LABEL_CORRECTNESS ) {
334 SECTION(
"can be called immediately" ) {
339 SECTION(
"correct output" ) {
341 GENERATE( range(0,10) );
345 vector<unsigned> in(numSeeds);
346 for (
int i=0; i<numSeeds; i++)
357 SECTION( LABEL_VALIDATION ) {
359 SECTION(
"env not initialised" ) {
371TEST_CASE(
"invalidQuESTInputError", TEST_CATEGORY ) {
373 SECTION( LABEL_CORRECTNESS ) {
375 REQUIRE_THROWS_WITH(
invalidQuESTInputError(
"msg",
"func"), ContainsSubstring(
"msg") && ContainsSubstring(
"func") );
378 SECTION( LABEL_VALIDATION ) {
386TEST_CASE(
"setValidationOn", TEST_CATEGORY ) {
388 SECTION( LABEL_CORRECTNESS ) {
391 for (
int i=0; i<3; i++)
395 REQUIRE_THROWS(
setSeeds(
nullptr, -99) );
398 SECTION( LABEL_VALIDATION ) {
406TEST_CASE(
"setValidationOff", TEST_CATEGORY ) {
408 SECTION( LABEL_CORRECTNESS ) {
411 for (
int i=0; i<3; i++)
428 SECTION( LABEL_VALIDATION ) {
441 SECTION( LABEL_CORRECTNESS ) {
443 SECTION(
"affects validation" ) {
447 SECTION(
"unitarity" ) {
477 SECTION(
"affects struct fields" ) {
479 SECTION(
"CompMatr" ) {
482 *(m.isApproxUnitary) = 1;
483 *(m.isApproxHermitian) = 1;
486 REQUIRE( *(m.isApproxUnitary) == -1 );
487 REQUIRE( *(m.isApproxHermitian) == -1 );
492 SECTION(
"DiagMatr" ) {
495 *(m.isApproxUnitary) = 1;
496 *(m.isApproxHermitian) = 0;
500 REQUIRE( *(m.isApproxUnitary) == -1 );
501 REQUIRE( *(m.isApproxHermitian) == -1 );
507 SECTION(
"FullStateDiagMatr" ) {
510 *(m.isApproxUnitary) = 1;
511 *(m.isApproxHermitian) = 0;
512 *(m.isApproxNonZero) = 1;
515 REQUIRE( *(m.isApproxUnitary) == -1 );
516 REQUIRE( *(m.isApproxHermitian) == -1 );
517 REQUIRE( *(m.isApproxNonZero) == -1 );
522 SECTION(
"KrausMap" ) {
525 *(k.isApproxCPTP) = 1;
528 REQUIRE( *(k.isApproxCPTP) == -1 );
535 SECTION( LABEL_VALIDATION ) {
537 SECTION(
"negative epsilon" ) {
539 qreal eps = GENERATE( -0.5, -1, -100 );
550TEST_CASE(
"getValidationEpsilon", TEST_CATEGORY ) {
552 SECTION( LABEL_CORRECTNESS ) {
555 for (
int i=0; i<3; i++)
558 GENERATE( range(0,10) );
567 SECTION( LABEL_VALIDATION ) {
578TEST_CASE(
"setValidationEpsilonToDefault", TEST_CATEGORY ) {
580 SECTION( LABEL_CORRECTNESS ) {
582 SECTION(
"always safe to call" ) {
584 for (
int i=0; i<3; i++)
588 SECTION(
"affects validation" ) {
608 SECTION(
"affects struct fields" ) {
610 SECTION(
"CompMatr" ) {
613 *(m.isApproxUnitary) = 1;
614 *(m.isApproxHermitian) = 1;
617 REQUIRE( *(m.isApproxUnitary) == -1 );
618 REQUIRE( *(m.isApproxHermitian) == -1 );
623 SECTION(
"DiagMatr" ) {
626 *(m.isApproxUnitary) = 1;
627 *(m.isApproxHermitian) = 0;
631 REQUIRE( *(m.isApproxUnitary) == -1 );
632 REQUIRE( *(m.isApproxHermitian) == -1 );
638 SECTION(
"FullStateDiagMatr" ) {
641 *(m.isApproxUnitary) = 1;
642 *(m.isApproxHermitian) = 0;
643 *(m.isApproxNonZero) = 1;
646 REQUIRE( *(m.isApproxUnitary) == -1 );
647 REQUIRE( *(m.isApproxHermitian) == -1 );
648 REQUIRE( *(m.isApproxNonZero) == -1 );
653 SECTION(
"KrausMap" ) {
656 *(k.isApproxCPTP) = 1;
659 REQUIRE( *(k.isApproxCPTP) == -1 );
666 SECTION( LABEL_VALIDATION ) {
668 SECTION( LABEL_VALIDATION ) {
680TEST_CASE(
"getGpuCacheSize", TEST_CATEGORY ) {
682 SECTION( LABEL_CORRECTNESS ) {
691 bool usingCuQuantum = std::string(envStr).find(
"cuQuantum=0") == std::string::npos;
694 if (
getQuESTEnv().isGpuAccelerated && !usingCuQuantum) {
699 for (qindex i=0; i<matr.numRows; i++)
700 matr.cpuElems[i][i] = 1;
706 int targs[] = {0,1,2,3,4,5};
707 int ctrls[] = {6,7,8,9};
708 qindex cacheSize = 0;
710 for (
int numCtrls=4; numCtrls>=0; numCtrls--) {
719 CAPTURE( cacheSize, newSize );
720 REQUIRE( newSize >= cacheSize );
730 SECTION( LABEL_VALIDATION ) {
KrausMap createKrausMap(int numQubits, int numOperators)
void destroyKrausMap(KrausMap map)
void getEnvironmentString(char str[200])
void setMaxNumReportedItems(qindex numRows, qindex numCols)
void setMaxNumReportedSigFigs(int numSigFigs)
void setNumReportedNewlines(int numNewlines)
void setSeeds(unsigned *seeds, int numSeeds)
void getSeeds(unsigned *seeds)
void setValidationEpsilonToDefault()
void invalidQuESTInputError(const char *msg, const char *func)
qreal getValidationEpsilon()
void setValidationEpsilon(qreal eps)
void initRandomPureState(Qureg qureg)
void initRandomMixedState(Qureg qureg, qindex numPureStates)
FullStateDiagMatr createFullStateDiagMatr(int numQubits)
CompMatr createCompMatr(int numQubits)
DiagMatr createDiagMatr(int numQubits)
void destroyDiagMatr(DiagMatr matrix)
void destroyFullStateDiagMatr(FullStateDiagMatr matrix)
void destroyCompMatr(CompMatr matrix)
static CompMatr1 getCompMatr1(qcomp **in)
void syncCompMatr(CompMatr matr)
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)
Qureg createQureg(int numQubits)
void destroyQureg(Qureg qureg)
qcomp getDensityQuregAmp(Qureg qureg, qindex row, qindex column)
qreal getRandomReal(qreal min, qreal maxIncl)
int getRandomInt(int min, int maxExcl)
void reportStr(const char *str)
static qcomp getQcomp(qreal re, qreal im)
void reportScalar(const char *label, qcomp num)
TEST_CASE("setMaxNumReportedSigFigs", TEST_CATEGORY)