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/linalg.hpp"
18#include "tests/utils/convert.hpp"
19#include "tests/utils/compare.hpp"
20#include "tests/utils/random.hpp"
24using Catch::Matchers::ContainsSubstring;
33#define TEST_CATEGORY \
34 LABEL_UNIT_TAG "[channels]"
48 SECTION( LABEL_CORRECTNESS ) {
50 int numQubits = GENERATE( range(1,4) );
51 int numMatrices = GENERATE( 1, 2, 10 );
52 CAPTURE( numQubits, numMatrices );
57 REQUIRE( map.numQubits == numQubits );
58 REQUIRE( map.numMatrices == numMatrices );
59 REQUIRE( map.numRows == getPow2(numQubits) );
62 REQUIRE( map.superop.numQubits == numQubits );
63 REQUIRE( map.superop.numRows == getPow2(2 * numQubits) );
66 REQUIRE( *(map.isApproxCPTP) == -1 );
67 REQUIRE( *(map.superop.wasGpuSynced) == 0 );
70 REQUIRE( map.matrices !=
nullptr );
71 REQUIRE( map.superop.cpuElems !=
nullptr );
72 REQUIRE( map.superop.cpuElemsFlat !=
nullptr );
74 REQUIRE( map.superop.gpuElemsFlat !=
nullptr );
76 REQUIRE( map.superop.gpuElemsFlat ==
nullptr );
80 for (qindex i=0; i<map.numMatrices && isZero; i++)
81 for (qindex r=0; r<map.numRows && isZero; r++)
82 for (qindex c=0; c<map.numRows && isZero; c++)
83 isZero = (map.matrices[i][r][c] == qcomp(0,0));
88 for (qindex r=0; r<map.superop.numRows && isZero; r++)
89 for (qindex c=0; c<map.superop.numRows && isZero; c++)
90 isZero = (map.superop.cpuElems[r][c] == qcomp(0,0));
96 SECTION( LABEL_VALIDATION ) {
98 SECTION(
"env is init" ) {
104 SECTION(
"too few qubits" ) {
106 int numQubits = GENERATE( -1, 0 );
108 REQUIRE_THROWS_WITH(
createKrausMap(numQubits,1), ContainsSubstring(
"one or more qubits") );
111 SECTION(
"too few operators" ) {
113 int numOpers = GENERATE( -1, 0 );
115 REQUIRE_THROWS_WITH(
createKrausMap(1,numOpers), ContainsSubstring(
"strictly positive number of matrices") );
118 SECTION(
"too many qubits" ) {
120 REQUIRE_THROWS_WITH(
createKrausMap(100,1), ContainsSubstring(
"can be addressed by the qindex type") );
122 REQUIRE_THROWS_WITH(
createKrausMap(15,1), ContainsSubstring(
"necessary memory") && ContainsSubstring(
"would overflow") );
128 #ifndef SANITIZER_IS_ACTIVE
129 REQUIRE_THROWS_WITH(
createKrausMap(12,1), ContainsSubstring(
"failed") || ContainsSubstring(
"insufficient available memory") || ContainsSubstring(
"available GPU memory") );
133 SECTION(
"too many operators" ) {
146TEST_CASE(
"destroyKrausMap", TEST_CATEGORY ) {
148 SECTION( LABEL_CORRECTNESS ) {
154 SECTION( LABEL_VALIDATION ) {
157 #ifndef SANITIZER_IS_ACTIVE
158 SECTION(
"not created" ) {
161 REQUIRE_THROWS_WITH(
destroyKrausMap(m), ContainsSubstring(
"Invalid KrausMap") && ContainsSubstring(
"not created") );
170 SECTION( LABEL_CORRECTNESS ) {
172 int numTargs = GENERATE( range(1,4) );
173 int numMatrs = GENERATE( 1, 2, 10 );
174 CAPTURE( numTargs, numMatrs );
177 REQUIRE( *(map.superop.wasGpuSynced) == 0 );
179 *(map.isApproxCPTP) = 0;
183 REQUIRE( *(map.superop.wasGpuSynced) == 1 );
184 REQUIRE( *(map.isApproxCPTP) == -1 );
190 REQUIRE_AGREE( map.superop, getSuperOperator(matrices) );
200 SECTION( LABEL_VALIDATION ) {
205 #ifndef SANITIZER_IS_ACTIVE
206 SECTION(
"not created" ) {
209 REQUIRE_THROWS_WITH(
syncKrausMap(m), ContainsSubstring(
"Invalid KrausMap") && ContainsSubstring(
"not created") );
219 SECTION( LABEL_CORRECTNESS ) {
221 int numQubits = GENERATE( range(1,4) );
222 int numOps = GENERATE( 1, 2, 10 );
223 CAPTURE( numQubits, numOps );
228 SECTION( LABEL_C_INTERFACE ) {
230 qcomp*** ptrs = (qcomp***) malloc(numOps *
sizeof *ptrs);
231 for (
int n=0; n<numOps; n++) {
232 ptrs[n] = (qcomp**) malloc(map.numRows *
sizeof **ptrs);
233 for (
int r=0; r<map.numRows; r++)
234 ptrs[n][r] = matrices[n][r].data();
238 REQUIRE( *(map.superop.wasGpuSynced) == 1 );
239 REQUIRE_AGREE( map.superop, getSuperOperator(matrices) );
241 for (
int n=0; n<numOps; n++)
246 SECTION( LABEL_CPP_INTERFACE ) {
248 *(map.superop.wasGpuSynced) = 0;
251 REQUIRE( *(map.superop.wasGpuSynced) == 1 );
252 REQUIRE_AGREE( map.superop, getSuperOperator(matrices) );
263 SECTION( LABEL_VALIDATION ) {
271 int err = GENERATE( -1, +1 );
276 #ifndef SANITIZER_IS_ACTIVE
277 SECTION(
"not created" ) {
285 SECTION(
"inconsistent dimensions" ) {
290 SECTION(
"inconsistent number of matrices" ) {
302 SECTION( LABEL_CORRECTNESS ) {
304 int numQubits = GENERATE( range(1,4) );
305 int numOps = GENERATE( 1, 2, 10 );
306 CAPTURE( numQubits, numOps );
313 SECTION( LABEL_CPP_INTERFACE ) {
315 *(map.superop.wasGpuSynced) = 0;
318 REQUIRE( *(map.superop.wasGpuSynced) == 1 );
319 REQUIRE_AGREE( map.superop, getSuperOperator(matrices) );
327 SECTION( LABEL_VALIDATION ) {
335 int err = GENERATE( -1, +1 );
340 #ifndef SANITIZER_IS_ACTIVE
341 SECTION(
"not created" ) {
349 SECTION(
"macro parameters" ) {
352 REQUIRE_THROWS_WITH(
setInlineKrausMap(map, numQubits+err, numOps,
getRandomKrausMap(numQubits, numOps)), ContainsSubstring(
"number of Kraus operators") && ContainsSubstring(
"qubits") );
353 REQUIRE_THROWS_WITH(
setInlineKrausMap(map, numQubits, numOps+err,
getRandomKrausMap(numQubits, numOps)), ContainsSubstring(
"number of Kraus operators") && ContainsSubstring(
"qubits") );
356 SECTION(
"dimensions" ) {
370 SECTION( LABEL_CORRECTNESS ) {
372 int numQubits = GENERATE( range(1,4) );
373 int numOps = GENERATE( 1, 2, 10 );
374 CAPTURE( numQubits, numOps );
380 SECTION( LABEL_CPP_INTERFACE ) {
384 REQUIRE( *(map.superop.wasGpuSynced) == 1 );
385 REQUIRE_AGREE( map.superop, getSuperOperator(matrices) );
393 SECTION( LABEL_VALIDATION ) {
397 SECTION(
"env not created" ) {
403 SECTION(
"incomaptible number of matrices" ) {
405 int err = GENERATE( -1, +1 );
410 SECTION(
"incompatible dimensions" ) {
412 int err = GENERATE( -1, +1 );
417 SECTION(
"too few qubits" ) {
419 int numQubits = GENERATE( -1, 0 );
421 REQUIRE_THROWS_WITH(
createInlineKrausMap(numQubits,1,{{{}}}), ContainsSubstring(
"one or more qubits") );
424 SECTION(
"too few operators" ) {
426 int numOpers = GENERATE( -1, 0 );
428 REQUIRE_THROWS_WITH(
createInlineKrausMap(1,numOpers,{{{}}}), ContainsSubstring(
"strictly positive number of matrices") );
431 SECTION(
"too many qubits" ) {
433 REQUIRE_THROWS_WITH(
createInlineKrausMap(100,1,{{{}}}), ContainsSubstring(
"can be addressed by the qindex type") );
435 REQUIRE_THROWS_WITH(
createInlineKrausMap(15,1,{{{}}}), ContainsSubstring(
"necessary memory") && ContainsSubstring(
"would overflow") );
441 SECTION(
"too many operators" ) {
455TEST_CASE(
"createSuperOp", TEST_CATEGORY ) {
457 SECTION( LABEL_CORRECTNESS ) {
459 int numQubits = GENERATE( range(1,4) );
460 CAPTURE( numQubits );
465 REQUIRE( op.numQubits == numQubits );
466 REQUIRE( op.numRows == getPow2(2*numQubits) );
469 REQUIRE( *(op.wasGpuSynced) == 0 );
472 REQUIRE( op.cpuElems !=
nullptr );
473 REQUIRE( op.cpuElemsFlat !=
nullptr );
475 REQUIRE( op.gpuElemsFlat !=
nullptr );
477 REQUIRE( op.gpuElemsFlat ==
nullptr );
481 for (qindex r=0; r<op.numRows && isZero; r++)
482 for (qindex c=0; c<op.numRows && isZero; c++)
483 isZero = (op.cpuElems[r][c] == qcomp(0,0));
489 SECTION( LABEL_VALIDATION ) {
491 SECTION(
"env is init" ) {
497 SECTION(
"too few qubits" ) {
499 int numQubits = GENERATE( -1, 0 );
501 REQUIRE_THROWS_WITH(
createSuperOp(numQubits), ContainsSubstring(
"one or more qubits") );
504 SECTION(
"too many qubits" ) {
506 REQUIRE_THROWS_WITH(
createSuperOp(100), ContainsSubstring(
"qindex") );
508 REQUIRE_THROWS_WITH(
createSuperOp(15), ContainsSubstring(
"size_t") );
514 #ifndef SANITIZER_IS_ACTIVE
515 REQUIRE_THROWS_WITH(
createSuperOp(12), ContainsSubstring(
"failed") || ContainsSubstring(
"insufficient available memory") || ContainsSubstring(
"available GPU memory") );
524 SECTION( LABEL_CORRECTNESS ) {
527 REQUIRE( *(op.wasGpuSynced) == 0 );
530 REQUIRE( *(op.wasGpuSynced) == 1 );
540 SECTION( LABEL_VALIDATION ) {
545 #ifndef SANITIZER_IS_ACTIVE
546 SECTION(
"not created" ) {
549 REQUIRE_THROWS_WITH(
syncSuperOp(m), ContainsSubstring(
"invalid fields") );
557TEST_CASE(
"destroySuperOp", TEST_CATEGORY ) {
559 SECTION( LABEL_CORRECTNESS ) {
565 SECTION( LABEL_VALIDATION ) {
568 #ifndef SANITIZER_IS_ACTIVE
569 SECTION(
"not created" ) {
572 REQUIRE_THROWS_WITH(
destroySuperOp(m), ContainsSubstring(
"invalid fields") );
581 SECTION( LABEL_CORRECTNESS ) {
583 int numQubits = GENERATE( range(1,4) );
584 CAPTURE( numQubits );
587 REQUIRE( *(op.wasGpuSynced) == 0 );
589 int dim = getPow2(2 * numQubits);
590 qmatrix ref = getRandomMatrix(dim);
592 SECTION( LABEL_C_INTERFACE ) {
595 qcomp** ptrs = (qcomp**) malloc(dim *
sizeof *ptrs);
596 for (
int i=0; i<dim; i++) {
597 ptrs[i] = (qcomp*) malloc(dim *
sizeof **ptrs);
598 for (
int j=0; j<dim; j++)
599 ptrs[i][j] = ref[i][j];
602 REQUIRE_AGREE( op, ref );
603 REQUIRE( *(op.wasGpuSynced) == 1 );
608 for (
int i=0; i<dim; i++)
613 SECTION( LABEL_CPP_INTERFACE ) {
618 REQUIRE_AGREE( op, ref );
619 REQUIRE( *(op.wasGpuSynced) == 1 );
625 SECTION( LABEL_VALIDATION ) {
632 #ifndef SANITIZER_IS_ACTIVE
633 SECTION(
"not created" ) {
637 REQUIRE_THROWS_WITH(
setSuperOp(bad, dummy), ContainsSubstring(
"invalid fields") );
642 SECTION(
"null pointer" ) {
644 qcomp** ptr =
nullptr;
645 REQUIRE_THROWS_WITH(
setSuperOp(op, ptr), ContainsSubstring(
"was a null pointer") );
647 qcomp* arr[1] = {
nullptr};
648 REQUIRE_THROWS_WITH(
setSuperOp(op, arr), ContainsSubstring(
"contained a null pointer") );
651 SECTION(
"invalid dimensions" ) {
655 REQUIRE_NOTHROW(
setSuperOp(op, qmatrix(4, qvector(4))) );
657 REQUIRE_THROWS_WITH(
setSuperOp(op, qmatrix(3, qvector(4))), ContainsSubstring(
"Incompatible number of rows") );
658 REQUIRE_THROWS_WITH(
setSuperOp(op, qmatrix(5, qvector(4))), ContainsSubstring(
"Incompatible number of rows") );
660 REQUIRE_THROWS_WITH(
setSuperOp(op, qmatrix(4, qvector(3))), ContainsSubstring(
"Incompatible number of columns") );
661 REQUIRE_THROWS_WITH(
setSuperOp(op, qmatrix(4, qvector(5))), ContainsSubstring(
"Incompatible number of columns") );
671 SECTION( LABEL_CORRECTNESS ) {
674 REQUIRE( *(op.wasGpuSynced) == 0 );
678 setInlineSuperOp( op, 1, {{1,2,3,4},{5,6,7,8},{-9,-8,-7,-6},{5_i,4_i,3_i,2_i}} );
679 REQUIRE_AGREE( op, {{1,2,3,4},{5,6,7,8},{-9,-8,-7,-6},{5_i,4_i,3_i,2_i}} );
680 REQUIRE( *(op.wasGpuSynced) == 1 );
685 SECTION( LABEL_VALIDATION ) {
692 #ifndef SANITIZER_IS_ACTIVE
693 SECTION(
"not created" ) {
696 REQUIRE_THROWS_WITH(
setInlineSuperOp(bad, 1, {{}}), ContainsSubstring(
"invalid fields") );
701 SECTION(
"mismatching dimension" ) {
703 REQUIRE_THROWS_WITH(
setInlineSuperOp(op, 2, {{}}), ContainsSubstring(
"specified number of qubits") && ContainsSubstring(
"differs") );
706 SECTION(
"invalid dimensions" ) {
712 REQUIRE_THROWS_WITH(
setInlineSuperOp(op, 1, qmatrix(3, qvector(4))), ContainsSubstring(
"Incompatible number of rows") );
713 REQUIRE_THROWS_WITH(
setInlineSuperOp(op, 1, qmatrix(5, qvector(4))), ContainsSubstring(
"Incompatible number of rows") );
715 REQUIRE_THROWS_WITH(
setInlineSuperOp(op, 1, qmatrix(4, qvector(3))), ContainsSubstring(
"Incompatible number of columns") );
716 REQUIRE_THROWS_WITH(
setInlineSuperOp(op, 1, qmatrix(4, qvector(5))), ContainsSubstring(
"Incompatible number of columns") );
724TEST_CASE(
"createInlineSuperOp", TEST_CATEGORY ) {
726 SECTION( LABEL_CORRECTNESS ) {
728 SuperOp op =
createInlineSuperOp(1, {{1,2,3,4}, {5,6,7,8}, {-9,-8,-7,-6}, {-5_i,-4_i,-3_i,-2_i}});
729 REQUIRE_AGREE( op, {{1,2,3,4}, {5,6,7,8}, {-9,-8,-7,-6}, {-5_i,-4_i,-3_i,-2_i}});
730 REQUIRE( *(op.wasGpuSynced) == 1 );
735 SECTION( LABEL_VALIDATION ) {
737 SECTION(
"env not initialised" ) {
743 SECTION(
"too few qubits" ) {
745 int numQubits = GENERATE( -1, 0 );
747 REQUIRE_THROWS_WITH(
createInlineSuperOp(numQubits, {{}}), ContainsSubstring(
"must act upon one or more qubits") );
750 SECTION(
"mismatching dimension" ) {
752 REQUIRE_THROWS_WITH(
createInlineSuperOp(1, {{1,2},{3,4}}), ContainsSubstring(
"inconsistent with the number of rows") );
KrausMap createInlineKrausMap(int numQubits, int numOperators, std::vector< std::vector< std::vector< qcomp > > > matrices)
SuperOp createInlineSuperOp(int numQubits, std::vector< std::vector< qcomp > > matrix)
KrausMap createKrausMap(int numQubits, int numOperators)
SuperOp createSuperOp(int numQubits)
void destroySuperOp(SuperOp op)
void destroyKrausMap(KrausMap map)
void reportKrausMap(KrausMap map)
void reportSuperOp(SuperOp op)
void setInlineSuperOp(SuperOp op, int numQb, std::vector< std::vector< qcomp > > matrix)
void setInlineKrausMap(KrausMap map, int numQb, int numOps, std::vector< std::vector< std::vector< qcomp > > > matrices)
void setSuperOp(SuperOp op, qcomp **matrix)
void setKrausMap(KrausMap map, qcomp ***matrices)
void syncSuperOp(SuperOp op)
void syncKrausMap(KrausMap map)
qmatrix getZeroMatrix(size_t dim)
vector< qmatrix > getRandomKrausMap(int numQb, int numOps)
TEST_CASE("createKrausMap", TEST_CATEGORY)