The Quantum Exact Simulation Toolkit v4.0.0
Loading...
Searching...
No Matches
channels.cpp
1/** @file
2 * Unit tests of the channels module.
3 *
4 * @author Tyson Jones
5 *
6 * @defgroup unitchannels Channels
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/linalg.hpp"
18#include "tests/utils/convert.hpp"
19#include "tests/utils/compare.hpp"
20#include "tests/utils/random.hpp"
21
22#include <vector>
23
24using Catch::Matchers::ContainsSubstring;
25using std::vector;
26
27
28
29/*
30 * UTILITIES
31 */
32
33#define TEST_CATEGORY \
34 LABEL_UNIT_TAG "[channels]"
35
36
37
38/**
39 * TESTS
40 *
41 * @ingroup unitchannels
42 * @{
43 */
44
45
46TEST_CASE( "createKrausMap", TEST_CATEGORY ) {
47
48 SECTION( LABEL_CORRECTNESS ) {
49
50 int numQubits = GENERATE( range(1,4) ); // 4qb superop = 8qb matrix = 16 statevec
51 int numMatrices = GENERATE( 1, 2, 10 );
52 CAPTURE( numQubits, numMatrices );
53
54 KrausMap map = createKrausMap(numQubits, numMatrices);
55
56 // verify dimensions
57 REQUIRE( map.numQubits == numQubits );
58 REQUIRE( map.numMatrices == numMatrices );
59 REQUIRE( map.numRows == getPow2(numQubits) );
60
61 // verify superoperator dimensions
62 REQUIRE( map.superop.numQubits == numQubits );
63 REQUIRE( map.superop.numRows == getPow2(2 * numQubits) );
64
65 // verify default fields
66 REQUIRE( *(map.isApproxCPTP) == -1 );
67 REQUIRE( *(map.superop.wasGpuSynced) == 0 );
68
69 // verify pointers
70 REQUIRE( map.matrices != nullptr );
71 REQUIRE( map.superop.cpuElems != nullptr );
72 REQUIRE( map.superop.cpuElemsFlat != nullptr );
73 if (getQuESTEnv().isGpuAccelerated)
74 REQUIRE( map.superop.gpuElemsFlat != nullptr );
75 else
76 REQUIRE( map.superop.gpuElemsFlat == nullptr );
77
78 // verify that all matrices default to zero
79 bool isZero = true;
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));
84 REQUIRE( isZero );
85
86 // verify superoperator defaults to zero
87 isZero = true;
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));
91 REQUIRE( isZero );
92
93 destroyKrausMap(map);
94 }
95
96 SECTION( LABEL_VALIDATION ) {
97
98 SECTION( "env is init" ) {
99
100 // no way to test this
101 SUCCEED( );
102 }
103
104 SECTION( "too few qubits" ) {
105
106 int numQubits = GENERATE( -1, 0 );
107
108 REQUIRE_THROWS_WITH( createKrausMap(numQubits,1), ContainsSubstring("one or more qubits") );
109 }
110
111 SECTION( "too few operators" ) {
112
113 int numOpers = GENERATE( -1, 0 );
114
115 REQUIRE_THROWS_WITH( createKrausMap(1,numOpers), ContainsSubstring("strictly positive number of matrices") );
116 }
117
118 SECTION( "too many qubits" ) {
119
120 REQUIRE_THROWS_WITH( createKrausMap(100,1), ContainsSubstring("can be addressed by the qindex type") );
121
122 REQUIRE_THROWS_WITH( createKrausMap(15,1), ContainsSubstring("necessary memory") && ContainsSubstring("would overflow") );
123
124 // no overflows, but definitely exceeds local RAM and fails to allocate; frightens address sanitizer!
125 // note the specific error message depends on the what backend the auto-deployer tried to use (e.g.
126 // GPU-accel or distributed) and whether memory-probers realised there was insufficient memory in
127 // advance or whether it proceeded to malloc() which subsequently failed
128 #ifndef SANITIZER_IS_ACTIVE
129 REQUIRE_THROWS_WITH( createKrausMap(12,1), ContainsSubstring("failed") || ContainsSubstring("insufficient available memory") || ContainsSubstring("available GPU memory") );
130 #endif
131 }
132
133 SECTION( "too many operators" ) {
134
135 /// @todo
136 /// there is currently no validation for that the operator
137 /// list was too big and would ergo overflow memory (since
138 /// it's just a ridiculous, annoying scenario)
139
140 SUCCEED( );
141 }
142 }
143}
144
145
146TEST_CASE( "destroyKrausMap", TEST_CATEGORY ) {
147
148 SECTION( LABEL_CORRECTNESS ) {
149
150 KrausMap map = createKrausMap(3, 3);
151 REQUIRE_NOTHROW( destroyKrausMap(map) );
152 }
153
154 SECTION( LABEL_VALIDATION ) {
155
156 // sanitizer interferes with un-initialised struct values
157 #ifndef SANITIZER_IS_ACTIVE
158 SECTION( "not created" ) {
159
160 KrausMap m;
161 REQUIRE_THROWS_WITH( destroyKrausMap(m), ContainsSubstring("Invalid KrausMap") && ContainsSubstring("not created") );
162 }
163 #endif
164 }
165}
166
167
168TEST_CASE( "syncKrausMap", TEST_CATEGORY ) {
169
170 SECTION( LABEL_CORRECTNESS ) {
171
172 int numTargs = GENERATE( range(1,4) ); // 4qb superop = 8qb matrix = 16qb statevec
173 int numMatrs = GENERATE( 1, 2, 10 );
174 CAPTURE( numTargs, numMatrs );
175
176 KrausMap map = createKrausMap(numTargs, numMatrs);
177 REQUIRE( *(map.superop.wasGpuSynced) == 0 );
178
179 *(map.isApproxCPTP) = 0;
180
181 // validate the fields are updated
182 syncKrausMap(map);
183 REQUIRE( *(map.superop.wasGpuSynced) == 1 );
184 REQUIRE( *(map.isApproxCPTP) == -1 );
185
186 // validate the superop gets inferred correctly from the kraus map
187 auto matrices = getRandomKrausMap(numTargs, numMatrs);
188 setKrausMap(map, matrices); // calls sync anyway
189 syncKrausMap(map);
190 REQUIRE_AGREE( map.superop, getSuperOperator(matrices) );
191
192 // to test that the GPU memory was actually overwritten,
193 // we would need a custom accessor of GPU memory, requiring
194 // the tests are CUDA-compiled - no thank you mam! It is
195 // certain this function works from the other GPU tests.
196
197 destroyKrausMap(map);
198 }
199
200 SECTION( LABEL_VALIDATION ) {
201
202 /// @todo fails in MSVC for unknown reason
203 #ifndef _MSC_VER
204 // sanitizer messes with default initialisation
205 #ifndef SANITIZER_IS_ACTIVE
206 SECTION( "not created" ) {
207
208 KrausMap m;
209 REQUIRE_THROWS_WITH( syncKrausMap(m), ContainsSubstring("Invalid KrausMap") && ContainsSubstring("not created") );
210 }
211 #endif
212 #endif
213 }
214}
215
216
217TEST_CASE( "setKrausMap", TEST_CATEGORY ) {
218
219 SECTION( LABEL_CORRECTNESS ) {
220
221 int numQubits = GENERATE( range(1,4) ); // 4qb superop = 8qb matrix = 16qb statevec
222 int numOps = GENERATE( 1, 2, 10 );
223 CAPTURE( numQubits, numOps );
224
225 KrausMap map = createKrausMap(numQubits, numOps);
226 auto matrices = getRandomKrausMap(numQubits, numOps);
227
228 SECTION( LABEL_C_INTERFACE ) {
229
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();
235 }
236
237 setKrausMap(map, ptrs);
238 REQUIRE( *(map.superop.wasGpuSynced) == 1 );
239 REQUIRE_AGREE( map.superop, getSuperOperator(matrices) );
240
241 for (int n=0; n<numOps; n++)
242 free(ptrs[n]);
243 free(ptrs);
244 }
245
246 SECTION( LABEL_CPP_INTERFACE ) {
247
248 *(map.superop.wasGpuSynced) = 0;
249
250 setKrausMap(map, matrices);
251 REQUIRE( *(map.superop.wasGpuSynced) == 1 );
252 REQUIRE_AGREE( map.superop, getSuperOperator(matrices) );
253 }
254
255 // to test that the GPU memory was actually overwritten,
256 // we would need a custom accessor of GPU memory, requiring
257 // the tests are CUDA-compiled - no thank you mam! It is
258 // certain this function works from the other GPU tests.
259
260 destroyKrausMap(map);
261 }
262
263 SECTION( LABEL_VALIDATION ) {
264
265 // only C++ interface is validated
266
267 int numQubits = 3;
268 int numOps = 3;
269 KrausMap map = createKrausMap(numQubits, numOps);
270
271 int err = GENERATE( -1, +1 );
272
273 /// @todo fails in MSVC for unknown reason
274 #ifndef _MSC_VER
275 // sanitizer messes with default initialisation
276 #ifndef SANITIZER_IS_ACTIVE
277 SECTION( "not created" ) {
278
279 KrausMap bad;
280 REQUIRE_THROWS_WITH( setKrausMap(bad, getRandomKrausMap(numQubits, numOps)), ContainsSubstring("invalid") );
281 }
282 #endif
283 #endif
284
285 SECTION( "inconsistent dimensions" ) {
286
287 REQUIRE_THROWS_WITH( setKrausMap(map, getRandomKrausMap(numQubits+err, numOps)), ContainsSubstring("dimension") );
288 }
289
290 SECTION( "inconsistent number of matrices" ) {
291
292 REQUIRE_THROWS_WITH( setKrausMap(map, getRandomKrausMap(numQubits, numOps-err)), ContainsSubstring("number of matrices") );
293 }
294
295 destroyKrausMap(map);
296 }
297}
298
299
300TEST_CASE( "setInlineKrausMap", TEST_CATEGORY ) {
301
302 SECTION( LABEL_CORRECTNESS ) {
303
304 int numQubits = GENERATE( range(1,4) ); // 4qb superop = 8qb matrix = 16qb statevec
305 int numOps = GENERATE( 1, 2, 10 );
306 CAPTURE( numQubits, numOps );
307
308 KrausMap map = createKrausMap(numQubits, numOps);
309 auto matrices = getRandomKrausMap(numQubits, numOps);
310
311 // only the C++ interface can be tested
312
313 SECTION( LABEL_CPP_INTERFACE ) {
314
315 *(map.superop.wasGpuSynced) = 0;
316
317 setInlineKrausMap(map, numQubits, numOps, matrices);
318 REQUIRE( *(map.superop.wasGpuSynced) == 1 );
319 REQUIRE_AGREE( map.superop, getSuperOperator(matrices) );
320
321 /// @todo test GPU memory is overwritten
322 }
323
324 destroyKrausMap(map);
325 }
326
327 SECTION( LABEL_VALIDATION ) {
328
329 // only C++ interface can be validated
330
331 int numQubits = 3;
332 int numOps = 3;
333 KrausMap map = createKrausMap(numQubits, numOps);
334
335 int err = GENERATE( -1, +1 );
336
337 /// @todo fails in MSVC for unknown reason
338 #ifndef _MSC_VER
339 // sanitizer messes with default initialisation
340 #ifndef SANITIZER_IS_ACTIVE
341 SECTION( "not created" ) {
342
343 KrausMap bad;
344 REQUIRE_THROWS_WITH( setInlineKrausMap(bad, numQubits, numOps, getRandomKrausMap(numQubits, numOps)), ContainsSubstring("invalid") );
345 }
346 #endif
347 #endif
348
349 SECTION( "macro parameters" ) {
350
351 // check macro parameters are consistent
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") );
354 }
355
356 SECTION( "dimensions" ) {
357
358 // check lists are correctly sized
359 REQUIRE_THROWS_WITH( setInlineKrausMap(map, numQubits, numOps, getRandomKrausMap(numQubits+err, numOps )), ContainsSubstring("dimension") );
360 REQUIRE_THROWS_WITH( setInlineKrausMap(map, numQubits, numOps, getRandomKrausMap(numQubits, numOps+err)), ContainsSubstring("number of matrices") );
361 }
362
363 destroyKrausMap(map);
364 }
365}
366
367
368TEST_CASE( "createInlineKrausMap", TEST_CATEGORY ) {
369
370 SECTION( LABEL_CORRECTNESS ) {
371
372 int numQubits = GENERATE( range(1,4) ); // 4qb superop = 8qb matrix = 16qb statevec
373 int numOps = GENERATE( 1, 2, 10 );
374 CAPTURE( numQubits, numOps );
375
376 auto matrices = getRandomKrausMap(numQubits, numOps);
377
378 // only the C++ interface can be tested
379
380 SECTION( LABEL_CPP_INTERFACE ) {
381
382 KrausMap map = createInlineKrausMap(numQubits, numOps, matrices);
383
384 REQUIRE( *(map.superop.wasGpuSynced) == 1 );
385 REQUIRE_AGREE( map.superop, getSuperOperator(matrices) );
386
387 /// @todo test GPU memory is overwritten
388
389 destroyKrausMap(map);
390 }
391 }
392
393 SECTION( LABEL_VALIDATION ) {
394
395 // only C++ interface can be validated
396
397 SECTION( "env not created" ) {
398
399 // no way to test
400 SUCCEED( );
401 }
402
403 SECTION( "incomaptible number of matrices" ) {
404
405 int err = GENERATE( -1, +1 );
406
407 REQUIRE_THROWS_WITH( createInlineKrausMap(3, 3, getRandomKrausMap(3, 3+err)), ContainsSubstring("matrices") );
408 }
409
410 SECTION( "incompatible dimensions" ) {
411
412 int err = GENERATE( -1, +1 );
413
414 REQUIRE_THROWS_WITH( createInlineKrausMap(3, 3, getRandomKrausMap(3+err, 3)), ContainsSubstring("rows") );
415 }
416
417 SECTION( "too few qubits" ) {
418
419 int numQubits = GENERATE( -1, 0 );
420
421 REQUIRE_THROWS_WITH( createInlineKrausMap(numQubits,1,{{{}}}), ContainsSubstring("one or more qubits") );
422 }
423
424 SECTION( "too few operators" ) {
425
426 int numOpers = GENERATE( -1, 0 );
427
428 REQUIRE_THROWS_WITH( createInlineKrausMap(1,numOpers,{{{}}}), ContainsSubstring("strictly positive number of matrices") );
429 }
430
431 SECTION( "too many qubits" ) {
432
433 REQUIRE_THROWS_WITH( createInlineKrausMap(100,1,{{{}}}), ContainsSubstring("can be addressed by the qindex type") );
434
435 REQUIRE_THROWS_WITH( createInlineKrausMap(15,1,{{{}}}), ContainsSubstring("necessary memory") && ContainsSubstring("would overflow") );
436
437 // cannot check when massive alloc run-time fails since the passed
438 // vectors must be of the correct size - and ergo impossibly big!
439 }
440
441 SECTION( "too many operators" ) {
442
443 /// @todo
444 /// there is currently no validation for that the operator
445 /// list was too big and would ergo overflow memory (since
446 /// it's just a ridiculous, annoying scenario)
447
448 SUCCEED( );
449 }
450 }
451}
452
453
454
455TEST_CASE( "createSuperOp", TEST_CATEGORY ) {
456
457 SECTION( LABEL_CORRECTNESS ) {
458
459 int numQubits = GENERATE( range(1,4) ); // 4qb superop = 8qb matrix = 16 statevec
460 CAPTURE( numQubits );
461
462 SuperOp op = createSuperOp(numQubits);
463
464 // verify dimensions
465 REQUIRE( op.numQubits == numQubits );
466 REQUIRE( op.numRows == getPow2(2*numQubits) );
467
468 // verify default fields
469 REQUIRE( *(op.wasGpuSynced) == 0 );
470
471 // verify pointers
472 REQUIRE( op.cpuElems != nullptr );
473 REQUIRE( op.cpuElemsFlat != nullptr );
474 if (getQuESTEnv().isGpuAccelerated)
475 REQUIRE( op.gpuElemsFlat != nullptr );
476 else
477 REQUIRE( op.gpuElemsFlat == nullptr );
478
479 // verify superoperator defaults to zero
480 bool isZero = true;
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));
484 REQUIRE( isZero );
485
486 destroySuperOp(op);
487 }
488
489 SECTION( LABEL_VALIDATION ) {
490
491 SECTION( "env is init" ) {
492
493 // no way to test this
494 SUCCEED( );
495 }
496
497 SECTION( "too few qubits" ) {
498
499 int numQubits = GENERATE( -1, 0 );
500
501 REQUIRE_THROWS_WITH( createSuperOp(numQubits), ContainsSubstring("one or more qubits") );
502 }
503
504 SECTION( "too many qubits" ) {
505
506 REQUIRE_THROWS_WITH( createSuperOp(100), ContainsSubstring("qindex") );
507
508 REQUIRE_THROWS_WITH( createSuperOp(15), ContainsSubstring("size_t") );
509
510 // no overflows, but definitely exceeds local RAM and fails to allocate; frightens address sanitizer!
511 // note the specific error message depends on the what backend the auto-deployer tried to use (e.g.
512 // GPU-accel or distributed) and whether memory-probers realised there was insufficient memory in
513 // advance or whether it proceeded to malloc() which subsequently failed
514 #ifndef SANITIZER_IS_ACTIVE
515 REQUIRE_THROWS_WITH( createSuperOp(12), ContainsSubstring("failed") || ContainsSubstring("insufficient available memory") || ContainsSubstring("available GPU memory") );
516 #endif
517 }
518 }
519}
520
521
522TEST_CASE( "syncSuperOp", TEST_CATEGORY ) {
523
524 SECTION( LABEL_CORRECTNESS ) {
525
526 SuperOp op = createSuperOp(1);
527 REQUIRE( *(op.wasGpuSynced) == 0 );
528
529 syncSuperOp(op);
530 REQUIRE( *(op.wasGpuSynced) == 1 );
531
532 // to test that the GPU memory was actually overwritten,
533 // we would need a custom accessor of GPU memory, requiring
534 // the tests are CUDA-compiled - no thank you mam! It is
535 // certain this function works from the other GPU tests.
536
537 destroySuperOp(op);
538 }
539
540 SECTION( LABEL_VALIDATION ) {
541
542 /// @todo fails in MSVC for unknown reason
543 #ifndef _MSC_VER
544 // sanitizer messes with default initialisation
545 #ifndef SANITIZER_IS_ACTIVE
546 SECTION( "not created" ) {
547
548 SuperOp m;
549 REQUIRE_THROWS_WITH( syncSuperOp(m), ContainsSubstring("invalid fields") );
550 }
551 #endif
552 #endif
553 }
554}
555
556
557TEST_CASE( "destroySuperOp", TEST_CATEGORY ) {
558
559 SECTION( LABEL_CORRECTNESS ) {
560
561 SuperOp m = createSuperOp(5);
562 REQUIRE_NOTHROW( destroySuperOp(m) );
563 }
564
565 SECTION( LABEL_VALIDATION ) {
566
567 // sanitizer interferes with un-initialised struct values
568 #ifndef SANITIZER_IS_ACTIVE
569 SECTION( "not created" ) {
570
571 SuperOp m;
572 REQUIRE_THROWS_WITH( destroySuperOp(m), ContainsSubstring("invalid fields") );
573 }
574 #endif
575 }
576}
577
578
579TEST_CASE( "setSuperOp", TEST_CATEGORY ) {
580
581 SECTION( LABEL_CORRECTNESS ) {
582
583 int numQubits = GENERATE( range(1,4) ); // 3qb superop = 6qb matrix = 12qb statevec
584 CAPTURE( numQubits );
585
586 SuperOp op = createSuperOp(numQubits);
587 REQUIRE( *(op.wasGpuSynced) == 0 );
588
589 int dim = getPow2(2 * numQubits);
590 qmatrix ref = getRandomMatrix(dim);
591
592 SECTION( LABEL_C_INTERFACE ) {
593
594 // nested pointers
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];
600 }
601 setSuperOp(op, ptrs);
602 REQUIRE_AGREE( op, ref );
603 REQUIRE( *(op.wasGpuSynced) == 1 );
604
605 // cannot test 2D VLAs in this C++ file
606
607 // cleanup
608 for (int i=0; i<dim; i++)
609 free(ptrs[i]);
610 free(ptrs);
611 }
612
613 SECTION( LABEL_CPP_INTERFACE ) {
614
615 // nested vectors
616 setSuperOp( op, getZeroMatrix(dim) ); // clear
617 setSuperOp( op, ref );
618 REQUIRE_AGREE( op, ref );
619 REQUIRE( *(op.wasGpuSynced) == 1 );
620 }
621
622 destroySuperOp(op);
623 }
624
625 SECTION( LABEL_VALIDATION ) {
626
627 SuperOp op = createSuperOp(1);
628
629 /// @todo fails in MSVC for unknown reason
630 #ifndef _MSC_VER
631 // sanitizer messes with default initialisation
632 #ifndef SANITIZER_IS_ACTIVE
633 SECTION( "not created" ) {
634
635 SuperOp bad;
636 qcomp** dummy;
637 REQUIRE_THROWS_WITH( setSuperOp(bad, dummy), ContainsSubstring("invalid fields") );
638 }
639 #endif
640 #endif
641
642 SECTION( "null pointer" ) {
643
644 qcomp** ptr = nullptr;
645 REQUIRE_THROWS_WITH( setSuperOp(op, ptr), ContainsSubstring("was a null pointer") );
646
647 qcomp* arr[1] = {nullptr};
648 REQUIRE_THROWS_WITH( setSuperOp(op, arr), ContainsSubstring("contained a null pointer") );
649 }
650
651 SECTION( "invalid dimensions" ) {
652
653 // detectable only by the C++ interface
654
655 REQUIRE_NOTHROW( setSuperOp(op, qmatrix(4, qvector(4))) );
656
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") );
659
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") );
662 }
663
664 destroySuperOp(op);
665 }
666}
667
668
669TEST_CASE( "setInlineSuperOp", TEST_CATEGORY ) {
670
671 SECTION( LABEL_CORRECTNESS ) {
672
673 SuperOp op = createSuperOp(1);
674 REQUIRE( *(op.wasGpuSynced) == 0 );
675
676 // can only check C++ interface
677
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 );
681
682 destroySuperOp(op);
683 }
684
685 SECTION( LABEL_VALIDATION ) {
686
687 SuperOp op = createSuperOp(1);
688
689 /// @todo fails in MSVC for unknown reason
690 #ifndef _MSC_VER
691 // sanitizer messes with default initialisation
692 #ifndef SANITIZER_IS_ACTIVE
693 SECTION( "not created" ) {
694
695 SuperOp bad;
696 REQUIRE_THROWS_WITH( setInlineSuperOp(bad, 1, {{}}), ContainsSubstring("invalid fields") );
697 }
698 #endif
699 #endif
700
701 SECTION( "mismatching dimension" ) {
702
703 REQUIRE_THROWS_WITH( setInlineSuperOp(op, 2, {{}}), ContainsSubstring("specified number of qubits") && ContainsSubstring("differs") );
704 }
705
706 SECTION( "invalid dimensions" ) {
707
708 // detectable only by the C++ interface
709
710 REQUIRE_NOTHROW( setInlineSuperOp(op, 1, qmatrix(4, qvector(4))) );
711
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") );
714
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") );
717 }
718
719 destroySuperOp(op);
720 }
721}
722
723
724TEST_CASE( "createInlineSuperOp", TEST_CATEGORY ) {
725
726 SECTION( LABEL_CORRECTNESS ) {
727
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 );
731
732 destroySuperOp(op);
733 }
734
735 SECTION( LABEL_VALIDATION ) {
736
737 SECTION( "env not initialised" ) {
738
739 // impossible to test
740 SUCCEED( );
741 }
742
743 SECTION( "too few qubits" ) {
744
745 int numQubits = GENERATE( -1, 0 );
746
747 REQUIRE_THROWS_WITH( createInlineSuperOp(numQubits, {{}}), ContainsSubstring("must act upon one or more qubits") );
748 }
749
750 SECTION( "mismatching dimension" ) {
751
752 REQUIRE_THROWS_WITH( createInlineSuperOp(1, {{1,2},{3,4}}), ContainsSubstring("inconsistent with the number of rows") );
753 }
754 }
755}
756
757
758/** @} (end defgroup) */
759
760
761
762/**
763 * @todo
764 * UNTESTED FUNCTIONS
765 */
766
767void reportKrausMap(KrausMap map);
768
769void reportSuperOp(SuperOp op);
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)
Definition channels.cpp:172
SuperOp createSuperOp(int numQubits)
Definition channels.cpp:158
void destroySuperOp(SuperOp op)
Definition channels.cpp:201
void destroyKrausMap(KrausMap map)
Definition channels.cpp:208
void reportKrausMap(KrausMap map)
Definition channels.cpp:465
void reportSuperOp(SuperOp op)
Definition channels.cpp:445
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)
Definition channels.cpp:269
void setKrausMap(KrausMap map, qcomp ***matrices)
Definition channels.cpp:307
void syncSuperOp(SuperOp op)
Definition channels.cpp:221
void syncKrausMap(KrausMap map)
Definition channels.cpp:234
QuESTEnv getQuESTEnv()
qmatrix getZeroMatrix(size_t dim)
Definition qmatrix.cpp:18
vector< qmatrix > getRandomKrausMap(int numQb, int numOps)
Definition random.cpp:383
TEST_CASE("createKrausMap", TEST_CATEGORY)
Definition channels.cpp:46