The Quantum Exact Simulation Toolkit v4.0.0
Loading...
Searching...
No Matches
test_data_structures.cpp
1/** @file
2 * Ported tests of the deprecated QuEST v3 interface,
3 * unit testing the "data structures" 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 * @author Richard Meister (QuEST v3 patches)
12 */
13
14#include <catch2/catch_test_macros.hpp>
15#include <catch2/matchers/catch_matchers_string.hpp>
16#include <catch2/generators/catch_generators_range.hpp>
17
18// must define preprocessors to enable quest's
19// deprecated v3 API, and disable the numerous
20// warnings issued by its compilation
21#define INCLUDE_DEPRECATED_FUNCTIONS 1
22#define DISABLE_DEPRECATION_WARNINGS 1
23#include "quest/include/quest.h"
24
25#include "test_utilities.hpp"
26
27/* allows concise use of ContainsSubstring in catch's REQUIRE_THROWS_WITH */
28using Catch::Matchers::ContainsSubstring;
29
30
31
32// fromComplex removed because Complex is completely deprecated
33
34 // /** @sa fromComplex
35 // * @ingroup deprecatedtests
36 // * @author Tyson Jones
37 // */
38 // TEST_CASE( "fromComplex", "[data_structures]" ) {
39
40 // Complex a;
41 // a.real= .5;
42 // a.imag= -.2;
43 // qcomp b = fromComplex(a);
44
45 // REQUIRE( a.real == real(b) );
46 // REQUIRE( a.imag == imag(b) );
47 // }
48
49
50
51// getStaticComplexMatrixN is completely deprecated
52
53 // /** @sa getStaticComplexMatrixN
54 // * @ingroup deprecatedtests
55 // * @author Tyson Jones
56 // */
57 // TEST_CASE( "getStaticComplexMatrixN", "[data_structures]" ) {
58
59 // /* use of this function is illegal in C++ */
60 // SUCCEED( );
61 // }
62
63
64
65// toComplex removed because Complex is completely deprecated
66
67 // /** @sa toComplex
68 // * @ingroup deprecatedtests
69 // * @author Tyson Jones
70 // */
71 // TEST_CASE( "toComplex", "[data_structures]" ) {
72
73 // qcomp a = qcomp(.5,-.2);
74 // #if (!defined(_WIN32)) && (!defined(_WIN64))
75 // Complex b = toComplex(a);
76 // #else
77 // // MSVC profanely forbids in-line struct initialisation
78 // Complex b; b.real = real(a); b.imag = imag(a);
79 // #endif
80
81 // REQUIRE( real(a) == b.real );
82 // REQUIRE( imag(a) == b.imag );
83 // }
84
85
86
87/** @sa createCloneQureg
88 * @ingroup deprecatedtests
89 * @author Tyson Jones
90 */
91TEST_CASE( "createCloneQureg", "[data_structures]" ) {
92
93 SECTION( "state-vector" ) {
94
95 Qureg a = createForcedQureg(NUM_QUBITS);
97
98 // check properties are the same
99 REQUIRE( b.isMultithreaded == a.isMultithreaded );
100 REQUIRE( b.isGpuAccelerated == a.isGpuAccelerated );
101 REQUIRE( b.isDistributed == a.isDistributed );
102
103 REQUIRE( b.rank == a.rank );
104 REQUIRE( b.numNodes == a.numNodes );
105 REQUIRE( b.logNumNodes == a.logNumNodes );
106
107 REQUIRE( b.isDensityMatrix == a.isDensityMatrix );
108 REQUIRE( b.numQubits == a.numQubits );
109 REQUIRE( b.numAmps == a.numAmps );
110 REQUIRE( b.logNumAmps == a.logNumAmps );
111
112 REQUIRE( b.numAmpsPerNode == a.numAmpsPerNode );
113 REQUIRE( b.logNumAmpsPerNode == a.logNumAmpsPerNode );
114 REQUIRE( b.logNumColsPerNode == a.logNumColsPerNode );
115
116 // check amps the same (works for GPU and distributed)
117 REQUIRE( areEqual(a, b) );
118
119 destroyQureg(a);
120 destroyQureg(b);
121 }
122 SECTION( "density-matrix" ) {
123
124 Qureg a = createForcedDensityQureg(NUM_QUBITS);
125 Qureg b = createCloneQureg(a);
126
127 // check properties are the same
128 REQUIRE( b.isMultithreaded == a.isMultithreaded );
129 REQUIRE( b.isGpuAccelerated == a.isGpuAccelerated );
130 REQUIRE( b.isDistributed == a.isDistributed );
131
132 REQUIRE( b.rank == a.rank );
133 REQUIRE( b.numNodes == a.numNodes );
134 REQUIRE( b.logNumNodes == a.logNumNodes );
135
136 REQUIRE( b.isDensityMatrix == a.isDensityMatrix );
137 REQUIRE( b.numQubits == a.numQubits );
138 REQUIRE( b.numAmps == a.numAmps );
139 REQUIRE( b.logNumAmps == a.logNumAmps );
140
141 REQUIRE( b.numAmpsPerNode == a.numAmpsPerNode );
142 REQUIRE( b.logNumAmpsPerNode == a.logNumAmpsPerNode );
143 REQUIRE( b.logNumColsPerNode == a.logNumColsPerNode );
144
145 // check amps are the same (works for GPU and distributed)
146 REQUIRE( areEqual(a, b) );
147
148 destroyQureg(a);
149 destroyQureg(b);
150 }
151}
152
153
154
155/** @sa createComplexMatrixN
156 * @ingroup deprecatedtests
157 * @author Tyson Jones
158 */
159TEST_CASE( "createComplexMatrixN", "[data_structures]" ) {
160
161 SECTION( "correctness" ) {
162
163 int numQb = GENERATE( range(1,10+1) );
164 ComplexMatrixN m = createComplexMatrixN(numQb);
165
166 // ensure elems are created and initialised to 0
167 REQUIRE( areEqual(toQMatrix(m), getZeroMatrix(1<<numQb)) );
168
169 destroyComplexMatrixN(m);
170 }
171 SECTION( "input validation" ) {
172
173 SECTION( "number of qubits" ) {
174
175 int numQb = GENERATE( -1, 0 );
176 REQUIRE_THROWS_WITH( createComplexMatrixN(numQb), ContainsSubstring("must target one or more qubits") );
177 }
178 }
179}
180
181
182
183/** @sa createDensityQureg
184 * @ingroup deprecatedtests
185 * @author Tyson Jones
186 */
187TEST_CASE( "createDensityQureg", "[data_structures]" ) {
188
189 // must be at least one column per node
190 int minNumQb = calcLog2(getQuESTEnv().numNodes);
191 if (minNumQb <= 0)
192 minNumQb = 1;
193
194 SECTION( "correctness" ) {
195
196 // try 10 valid number of qubits
197 int numQb = GENERATE_COPY( range(minNumQb, minNumQb+10) );
198 Qureg reg = createForcedDensityQureg(numQb);
199
200 // ensure elems (CPU and/or GPU) are created, and reg begins in |0><0|
201 QMatrix ref = getZeroMatrix(1<<numQb);
202 ref[0][0] = 1; // |0><0|
203 REQUIRE( areEqual(reg, ref) );
204
205 destroyQureg(reg);
206 }
207 SECTION( "input validation") {
208
209 SECTION( "number of qubits" ) {
210
211 int numQb = GENERATE( -1, 0 );
212 REQUIRE_THROWS_WITH( createForcedDensityQureg(numQb), ContainsSubstring("must contain one or more qubits") );
213 }
214 SECTION( "number of amplitudes" ) {
215
216 // v4 does not consult QuESTEnv
217
218 // // use local QuESTEnv to safely modify
219 // QuESTEnv env = getQuESTEnv();
220
221 // too many amplitudes to store in type
222 int maxQb = (int) calcLog2(SIZE_MAX) / 2;
223 REQUIRE_THROWS_WITH( createForcedDensityQureg(maxQb+1),
224 ContainsSubstring("the density matrix would contain more amplitudes") && ContainsSubstring("than can be addressed by the qindex type") );
225
226 // it is non-trivial to force an invalid distribution
227 // in v4, since QuESTEnv arg is no longer consulted; we
228 // will reserve this input validation for the full tests
229
230 // /* n-qubit density matrix contains 2^(2n) amplitudes
231 // * so can be spread between at most 2^(2n) ranks
232 // */
233 // /* env.numNodes is an int, so maxQb must be capped at 16 for this
234 // * test to avoid an integer overflow when storing 2**(2*minQb) in it
235 // */
236 // maxQb = maxQb > 16 ? 16 : maxQb;
237 // int minQb = GENERATE_COPY( range(3,maxQb) );
238 // env.numNodes = (int) pow(2, 2*minQb);
239 // int numQb = GENERATE_COPY( range(1,minQb) );
240 // REQUIRE_THROWS_WITH( createForcedDensityQureg(numQb, env), ContainsSubstring("Too few qubits") );
241 }
242 SECTION( "available memory" ) {
243
244 /* there is no reliable way to force the malloc statements to
245 * fail, and hence trigger the matrixInit validation */
246 SUCCEED( );
247 }
248 }
249}
250
251
252
253/** @sa createDiagonalOp
254 * @ingroup deprecatedtests
255 * @author Tyson Jones
256 */
257TEST_CASE( "createDiagonalOp", "[data_structures]" ) {
258
259 QuESTEnv env = getQuESTEnv();
260
261 // must be at least one amplitude per node
262 int minNumQb = calcLog2(env.numNodes);
263 if (minNumQb == 0)
264 minNumQb = 1;
265
266 SECTION( "correctness" ) {
267
268 // try 10 valid number of qubits
269 int numQb = GENERATE_COPY( range(minNumQb, minNumQb+10) );
270 DiagonalOp op = createDiagonalOp(numQb, env);
271
272 // check properties are correct
273 REQUIRE( op.numQubits == numQb );
274 REQUIRE( op.numElemsPerNode == (1LL << numQb) / (op.isDistributed? env.numNodes : 1) );
275 REQUIRE( op.cpuElems != NULL );
276
277 // check all elements in CPU are zero
278 REQUIRE( areEqual(toQVector(op), QVector(1LL << numQb)) );
279
280 // (no concise way to check this for GPU)
281
282 destroyDiagonalOp(op, env);
283 }
284 SECTION( "input validation" ) {
285
286 SECTION( "number of qubits" ) {
287
288 int numQb = GENERATE( -1, 0 );
289 REQUIRE_THROWS_WITH( createDiagonalOp(numQb, env), ContainsSubstring("must target one or more qubits") );
290 }
291 SECTION( "number of elements" ) {
292
293 // too many amplitudes to store in type
294 int maxQb = (int) calcLog2(SIZE_MAX);
295 REQUIRE_THROWS_WITH( createDiagonalOp(maxQb+1, env),
296 ContainsSubstring("the matrix would contain more elements") && ContainsSubstring("than the maximum which can be addressed by the qindex type") );
297
298 // cannot test when there are too few elements, since the deprecated
299 // interface is redirecting to createFullStateDiagMatr which auto-deploys,
300 // and so simply automatically disables distribution
301
302 // /* env.numNodes is an int, so maxQb must be capped at 32 for this
303 // * test to avoid an integer overflow when storing 2**minQb in it
304 // */
305 // maxQb = maxQb > 32 ? 32 : maxQb;
306 // // too few amplitudes to distribute
307 // int minQb = GENERATE_COPY( range(2,maxQb) );
308 // env.numNodes = (int) pow(2, minQb);
309 // int numQb = GENERATE_COPY( range(1,minQb) );
310 // REQUIRE_THROWS_WITH( createDiagonalOp(numQb, env), ContainsSubstring("Too few qubits") && ContainsSubstring("distributed"));
311 }
312 SECTION( "available memory" ) {
313
314 /* there is no reliable way to force the malloc statements to
315 * fail, and hence trigger the diagonalOpInit validation */
316 SUCCEED( );
317 }
318 }
319}
320
321
322
323// createDiagonalOpFromPauliHamilFile removed because PauliHamil
324// is deprecated, and replacement PauliStrSum (with analogous function
325// createFullStateDiagMatrFromPauliStrSumFile) is incompatible
326
327 // /** @sa createDiagonalOpFromPauliHamilFile
328 // * @ingroup deprecatedtests
329 // * @author Tyson Jones
330 // */
331 // TEST_CASE( "createDiagonalOpFromPauliHamilFile", "[data_structures]" ) {
332
333 // // files created & populated during the test, and deleted afterward
334 // char fnPrefix[] = "temp_createDiagonalOpFromPauliHamilFile";
335 // char fn[100];
336 // int fnMaxLen = 100;
337
338 // // each test uses a unique filename (managed by master node), to avoid file IO locks
339 // // (it's safe for sub-test to overwrite this, since after each test, all files
340 // // with prefix fnPrefix are deleted)
341 // setUniqueFilename(fn, fnMaxLen, fnPrefix);
342
343 // // diagonal op must have at least one amplitude per node
344 // int minNumQb = calcLog2(getQuESTEnv().numNodes);
345 // if (minNumQb == 0)
346 // minNumQb = 1;
347
348 // SECTION( "correctness" ) {
349
350 // SECTION( "general" ) {
351
352 // // try several Pauli Hamiltonian sizes
353 // int numQb = GENERATE_COPY( range(minNumQb, 6+minNumQb) );
354 // int numTerms = GENERATE_COPY( 1, minNumQb, 10*minNumQb );
355
356 // // create a PauliHamil with random elements
357
358 // PauliHamil hamil = createPauliHamil(numQb, numTerms);
359 // createPauliStrSum()
360 // setRandomDiagPauliHamil(hamil, numQb);
361
362 // // write the Hamiltonian to file (with trailing whitespace, and trailing newline)
363 // if (getQuESTEnv().rank == 0) {
364 // FILE* file = fopen(fn, "w");
365 // int i=0;
366 // for (int n=0; n<numTerms; n++) {
367 // fprintf(file, REAL_STRING_FORMAT, hamil.termCoeffs[n]);
368 // fprintf(file, " ");
369 // for (int q=0; q<numQb; q++)
370 // fprintf(file, "%d ", (int) hamil.pauliCodes[i++]);
371 // fprintf(file, "\n");
372 // }
373 // fprintf(file, "\n");
374 // fclose(file);
375 // }
376 // syncQuESTEnv(getQuESTEnv());
377
378 // // load the file as a diagonal operator, and compare
379 // DiagonalOp op = createDiagonalOpFromPauliHamilFile(fn, env);
380 // REQUIRE( areEqual(toQMatrix(op), toQMatrix(hamil)) );
381
382 // destroyPauliHamil(hamil);
383 // destroyDiagonalOp(op, env);
384 // }
385 // SECTION( "edge cases" ) {
386
387 // // prepare a valid single-term diagonal Pauli Hamiltonian
388 // qreal coeffs[] = {.1};
389 // VLA(pauliOpType, codes, minNumQb);
390 // for (int q=0; q<minNumQb; q++)
391 // codes[q] = (q%2)? PAULI_I : PAULI_Z;
392
393 // QMatrix ref = toQMatrix(coeffs, codes, minNumQb, 1);
394
395 // // prepare basic encoding string
396 // string line = to_string(coeffs[0]) + " ";
397 // for (int q=0; q<minNumQb; q++)
398 // line += to_string(codes[q]) + ((q<minNumQb-1)? " ":"");
399
400 // SECTION( "no trailing newline or space" ) {
401
402 // writeToFileSynch(fn, line);
403
404 // DiagonalOp op = createDiagonalOpFromPauliHamilFile(fn, env);
405 // REQUIRE( areEqual(ref, toQMatrix(op)) );
406
407 // destroyDiagonalOp(op, env);
408 // }
409 // SECTION( "trailing newlines" ) {
410
411 // writeToFileSynch(fn, line + "\n\n\n");
412
413 // DiagonalOp op = createDiagonalOpFromPauliHamilFile(fn, env);
414 // REQUIRE( areEqual(ref, toQMatrix(op)) );
415
416 // destroyDiagonalOp(op, env);
417 // }
418 // SECTION( "trailing spaces" ) {
419
420 // writeToFileSynch(fn, line + " ");
421
422 // DiagonalOp op = createDiagonalOpFromPauliHamilFile(fn, env);
423 // REQUIRE( areEqual(ref, toQMatrix(op)) );
424
425 // destroyDiagonalOp(op, env);
426 // }
427 // }
428 // }
429 // SECTION( "input validation") {
430
431 // SECTION( "number of qubits" ) {
432
433 // writeToFileSynch(fn, ".1 "); // 0 qubits
434 // REQUIRE_THROWS_WITH( createDiagonalOpFromPauliHamilFile(fn, env), ContainsSubstring("The number of qubits") && ContainsSubstring("strictly positive"));
435 // }
436 // SECTION( "number of elements" ) {
437
438 // // too many amplitudes to store in type
439 // int maxQb = (int) calcLog2(SIZE_MAX);
440
441 // // encode one more qubit than legal to file
442 // string line = ".1 ";
443 // for (int q=0; q<(maxQb+1); q++)
444 // line += "3 "; // trailing space ok
445 // writeToFileSynch(fn, line);
446
447 // REQUIRE_THROWS_WITH( createDiagonalOpFromPauliHamilFile(fn, env), ContainsSubstring("Too many qubits") && ContainsSubstring("size_t type") );
448
449 // // use local QuESTEnv to safely modify
450 // QuESTEnv env = getQuESTEnv();
451
452 // // too few elements to distribute
453 // /* env.numNodes is an int, so maxQb must be capped at 32 for this
454 // * test to avoid an integer overflow when storing 2**minQb in it
455 // */
456 // maxQb = maxQb > 32 ? 32 : maxQb;
457 // int minQb = GENERATE_COPY( range(2,maxQb) );
458 // env.numNodes = (int) pow(2, minQb);
459 // int numQb = GENERATE_COPY( range(1,minQb) );
460
461 // line = ".1 ";
462 // for (int q=0; q<numQb; q++)
463 // line += "3 "; // trailing space ok
464 // setUniqueFilename(fn, fnMaxLen, fnPrefix);
465 // writeToFileSynch(fn, line);
466
467 // REQUIRE_THROWS_WITH( createDiagonalOpFromPauliHamilFile(fn, env), ContainsSubstring("Too few qubits") && ContainsSubstring("distributed") );
468 // }
469 // SECTION( "coefficient type" ) {
470
471 // writeToFileSynch(fn, "notanumber 1 2 3");
472 // REQUIRE_THROWS_WITH( createDiagonalOpFromPauliHamilFile(fn, env), ContainsSubstring("Failed to parse") && ContainsSubstring("coefficient"));
473 // }
474 // SECTION( "pauli code" ) {
475
476 // writeToFileSynch(fn, ".1 0 3 2"); // final is invalid Y
477 // REQUIRE_THROWS_WITH( createDiagonalOpFromPauliHamilFile(fn, env), ContainsSubstring("contained operators other than PAULI_Z and PAULI_I"));
478
479 // setUniqueFilename(fn, fnMaxLen, fnPrefix);
480 // writeToFileSynch(fn, ".1 0 1 3"); // second is invalid X
481 // REQUIRE_THROWS_WITH( createDiagonalOpFromPauliHamilFile(fn, env), ContainsSubstring("contained operators other than PAULI_Z and PAULI_I"));
482
483 // setUniqueFilename(fn, fnMaxLen, fnPrefix);
484 // writeToFileSynch(fn, ".1 0 1 4"); // final is invalid Pauli code
485 // REQUIRE_THROWS_WITH( createDiagonalOpFromPauliHamilFile(fn, env), ContainsSubstring("invalid pauli code"));
486
487 // setUniqueFilename(fn, fnMaxLen, fnPrefix);
488 // writeToFileSynch(fn, ".1 3 0 notanumber"); // final is invalid type
489 // REQUIRE_THROWS_WITH( createDiagonalOpFromPauliHamilFile(fn, env), ContainsSubstring("Failed to parse the next expected Pauli code"));
490 // }
491 // }
492
493 // // delete all files created above
494 // deleteFilesWithPrefixSynch(fnPrefix);
495 // }
496
497
498
499// createPauliHamil removed because PauliHamil is deprecated, and
500// replacement PauliStrSum has an incompatible constructor
501
502 // /** @sa createPauliHamil
503 // * @ingroup deprecatedtests
504 // * @author Tyson Jones
505 // */
506 // TEST_CASE( "createPauliHamil", "[data_structures]" ) {
507
508 // SECTION( "correctness" ) {
509
510 // int numQb = GENERATE( range(1,5) );
511 // int numTerms = GENERATE( range(1,5) );
512 // PauliHamil hamil = createPauliHamil(numQb, numTerms);
513
514 // // check fields are correct
515 // REQUIRE( hamil.numQubits == numQb );
516 // REQUIRE( hamil.numSumTerms == numTerms );
517
518 // // check all Pauli codes are identity
519 // int numPaulis = numQb * numTerms;
520 // for (int i=0; i<numPaulis; i++) {
521 // REQUIRE( hamil.pauliCodes[i] == PAULI_I );
522 // }
523
524 // // check all term coefficients can be written to (no seg fault)
525 // for (int j=0; j<numTerms; j++) {
526 // hamil.termCoeffs[j] = 1;
527 // REQUIRE( hamil.termCoeffs[j] == 1 );
528 // }
529
530 // destroyPauliHamil(hamil);
531 // }
532 // SECTION( "input validation") {
533
534 // SECTION( "number of qubits" ) {
535
536 // int numQb = GENERATE( -1, 0 );
537 // REQUIRE_THROWS_WITH( createPauliHamil(numQb, 1), ContainsSubstring("The number of qubits and terms in the PauliHamil must be strictly positive.") );
538 // }
539 // SECTION( "number of terms" ) {
540
541 // int numTerms = GENERATE( -1, 0 );
542 // REQUIRE_THROWS_WITH( createPauliHamil(1, numTerms), ContainsSubstring("The number of qubits and terms in the PauliHamil must be strictly positive.") );
543 // }
544 // }
545 // }
546
547
548
549// createPauliHamilFromFile removed because PauliHamil is deprecated, and
550// replacement PauliStrSum (with analogous function createPauliStrSumFromFile
551// has incompatible struct fields
552
553 // /** @sa createPauliHamilFromFile
554 // * @ingroup deprecatedtests
555 // * @author Tyson Jones
556 // */
557 // TEST_CASE( "createPauliHamilFromFile", "[data_structures]" ) {
558
559 // // files created & populated during the test, and deleted afterward
560 // char fnPrefix[] = "temp_createPauliHamilFromFile";
561 // char fn[100];
562 // int fnMaxLen = 100;
563
564 // // each test uses a unique filename (managed by master node), to avoid file IO locks
565 // // (it's safe for sub-test to overwrite this, since after each test, all files
566 // // with prefix fnPrefix are deleted)
567 // setUniqueFilename(fn, fnMaxLen, fnPrefix);
568
569 // SECTION( "correctness" ) {
570
571 // SECTION( "general" ) {
572
573 // // for several sizes...
574 // int numQb = GENERATE( 1, 5, 10, 15 );
575 // int numTerms = GENERATE( 1, 10, 30 );
576 // int numPaulis = numQb*numTerms;
577
578 // // create a PauliHamil with random elements
579 // VLA(qreal, coeffs, numTerms);
580 // VLA(pauliOpType, paulis, numPaulis);
581 // setRandomPauliSum(coeffs, paulis, numQb, numTerms);
582
583 // // write the Hamiltonian to file (with trailing whitespace, and trailing newline)
584 // if (getQuESTEnv().rank == 0) {
585 // FILE* file = fopen(fn, "w");
586 // int i=0;
587 // for (int n=0; n<numTerms; n++) {
588 // fprintf(file, REAL_STRING_FORMAT, coeffs[n]);
589 // fprintf(file, " ");
590 // for (int q=0; q<numQb; q++)
591 // fprintf(file, "%d ", (int) paulis[i++]);
592 // fprintf(file, "\n");
593 // }
594 // fprintf(file, "\n");
595 // fclose(file);
596 // }
597 // syncQuESTEnv(getQuESTEnv());
598
599 // // load the file as a PauliHamil
600 // PauliHamil hamil = createPauliHamilFromFile(fn);
601
602 // // check fields agree
603 // REQUIRE( hamil.numQubits == numQb );
604 // REQUIRE( hamil.numSumTerms == numTerms );
605
606 // // check elements agree
607 // int j=0;
608 // for (int n=0; n<numTerms; n++) {
609 // REQUIRE( absReal(hamil.termCoeffs[n] - coeffs[n]) <= REAL_EPS );
610 // for (int q=0; q<numQb; q++) {
611 // REQUIRE( hamil.pauliCodes[j] == paulis[j] );
612 // j++;
613 // }
614 // }
615
616 // destroyPauliHamil(hamil);
617 // }
618 // SECTION( "edge cases" ) {
619
620 // SECTION( "no trailing newline or space" ) {
621
622 // writeToFileSynch(fn, ".1 1 0 1");
623
624 // PauliHamil hamil = createPauliHamilFromFile(fn);
625 // REQUIRE( hamil.numSumTerms == 1 );
626 // REQUIRE( hamil.numQubits == 3 );
627
628 // destroyPauliHamil(hamil);
629 // }
630 // SECTION( "trailing newlines" ) {
631
632 // writeToFileSynch(fn, ".1 1 0 1\n\n\n");
633
634 // PauliHamil hamil = createPauliHamilFromFile(fn);
635 // REQUIRE( hamil.numSumTerms == 1 );
636 // REQUIRE( hamil.numQubits == 3 );
637
638 // destroyPauliHamil(hamil);
639 // }
640 // SECTION( "trailing spaces" ) {
641
642 // writeToFileSynch(fn, ".1 1 0 1 ");
643
644 // PauliHamil hamil = createPauliHamilFromFile(fn);
645 // REQUIRE( hamil.numSumTerms == 1 );
646 // REQUIRE( hamil.numQubits == 3 );
647
648 // destroyPauliHamil(hamil);
649 // }
650 // }
651 // }
652 // SECTION( "input validation") {
653
654 // SECTION( "number of qubits" ) {
655
656 // writeToFileSynch(fn, ".1 ");
657
658 // REQUIRE_THROWS_WITH( createPauliHamilFromFile(fn), ContainsSubstring("The number of qubits") && ContainsSubstring("strictly positive"));
659 // }
660 // SECTION( "coefficient type" ) {
661
662 // writeToFileSynch(fn, "notanumber 1 2 3");
663
664 // REQUIRE_THROWS_WITH( createPauliHamilFromFile(fn), ContainsSubstring("Failed to parse") && ContainsSubstring("coefficient"));
665 // }
666 // SECTION( "pauli code" ) {
667
668 // writeToFileSynch(fn, ".1 1 2 4"); // invalid int
669
670 // REQUIRE_THROWS_WITH( createPauliHamilFromFile(fn), ContainsSubstring("invalid pauli code"));
671
672 // setUniqueFilename(fn, fnMaxLen, fnPrefix);
673 // writeToFileSynch(fn, ".1 1 2 notanumber"); // invalid type
674
675 // REQUIRE_THROWS_WITH( createPauliHamilFromFile(fn), ContainsSubstring("Failed to parse the next expected Pauli code"));
676 // }
677 // }
678
679 // // cleanup temp files
680 // deleteFilesWithPrefixSynch(fn);
681 // }
682
683
684
685/** @sa createQuESTEnv
686 * @ingroup deprecatedtests
687 * @author Tyson Jones
688 */
689TEST_CASE( "createQuESTEnv", "[data_structures]" ) {
690
691 /* there is no meaningful way to test this */
692 SUCCEED( );
693}
694
695
696
697/** @sa createQureg
698 * @ingroup deprecatedtests
699 * @author Tyson Jones
700 */
701TEST_CASE( "createQureg", "[data_structures]" ) {
702
703 // we use createForcedQureg() in lieu of createQureg()
704 // to override the auto-deployer and force Qureg to use
705 // every available parallellisation method
706
707 // must be at least one amplitude per node
708 int minNumQb = calcLog2(getQuESTEnv().numNodes);
709 if (minNumQb == 0)
710 minNumQb = 1;
711
712 SECTION( "correctness" ) {
713
714 // try 10 valid number of qubits
715 int numQb = GENERATE_COPY( range(minNumQb, minNumQb+10) );
716 Qureg reg = createForcedQureg(numQb);
717
718 // ensure elems (CPU and/or GPU) are created, and reg begins in |0>
719 QVector ref = QVector(1<<numQb);
720 ref[0] = 1; // |0>
721 REQUIRE( areEqual(reg, ref) );
722
723 destroyQureg(reg);
724 }
725 SECTION( "input validation") {
726
727 SECTION( "number of qubits" ) {
728
729 int numQb = GENERATE( -1, 0 );
730 REQUIRE_THROWS_WITH( createForcedQureg(numQb), ContainsSubstring("must contain one or more qubits") );
731 }
732 SECTION( "number of amplitudes" ) {
733
734 // too many amplitudes to store in type
735 int maxQb = (int) calcLog2(SIZE_MAX);
736 REQUIRE_THROWS_WITH( createForcedQureg(maxQb+1),
737 ContainsSubstring("the statevector would contain more amplitudes") &&
738 ContainsSubstring("than the maximum which can be addressed by the qindex type") );
739
740 // it is non-trivial to force an invalid distribution
741 // in v4, since QuESTEnv arg is no longer consulted; we
742 // will reserve this input validation for the full tests
743
744 // // too few amplitudes to distribute
745 // /* env.numNodes is an int, so maxQb must be capped at 32 for this
746 // * test to avoid an integer overflow when storing 2**minQb in it
747 // */
748 // maxQb = maxQb > 32 ? 32 : maxQb;
749 // int minQb = GENERATE_COPY( range(2,maxQb) );
750 // env.numNodes = (int) pow(2, minQb);
751 // int numQb = GENERATE_COPY( range(1,minQb) );
752 // REQUIRE_THROWS_WITH( createForcedQureg(numQb, env), ContainsSubstring("Too few qubits") );
753 }
754 SECTION( "available memory" ) {
755
756 /* there is no reliable way to force the malloc statements to
757 * fail, and hence trigger the matrixInit validation */
758 SUCCEED( );
759 }
760 }
761}
762
763
764
765/** @sa createSubDiagonalOp
766 * @ingroup deprecatedtests
767 * @author Tyson Jones
768 */
769TEST_CASE( "createSubDiagonalOp", "[data_structures]" ) {
770
771 SECTION( "correctness" ) {
772
773 int numQb = GENERATE( range(1,10+1) );
774 SubDiagonalOp op = createSubDiagonalOp(numQb);
775
776 // ensure elems are created and initialised to 0
777 REQUIRE( areEqual(toQMatrix(op), getZeroMatrix(1<<numQb)) );
778
779 destroySubDiagonalOp(op);
780 }
781 SECTION( "input validation" ) {
782
783 SECTION( "number of qubits" ) {
784
785 int numQb = GENERATE( -1, 0 );
786 REQUIRE_THROWS_WITH( createSubDiagonalOp(numQb), ContainsSubstring("must target one or more qubits") );
787
788 numQb = 100;
789 REQUIRE_THROWS_WITH( createSubDiagonalOp(numQb),
790 ContainsSubstring("the matrix would contain more elements") &&
791 ContainsSubstring("than the maximum which can be addressed by the qindex type") );
792 }
793 }
794}
795
796
797
798/** @sa destroyComplexMatrixN
799 * @ingroup deprecatedtests
800 * @author Tyson Jones
801 */
802TEST_CASE( "destroyComplexMatrixN", "[data_structures]" ) {
803
804 SECTION( "correctness" ) {
805
806 /* there is no meaningful way to test this */
807 SUCCEED( );
808 }
809 SECTION( "input validation" ) {
810
811 SECTION( "matrix not created" ) {
812
813 /* this is an artificial test case since nothing in the QuEST API
814 * automatically sets un-initialised ComplexMatrixN fields to
815 * the NULL pointer.
816 */
817 ComplexMatrixN m;
818 m.cpuElems = NULL;
819
820 /* the error message is also somewhat unrelated, but oh well
821 */
822 REQUIRE_THROWS_WITH( destroyComplexMatrixN(m), ContainsSubstring("Invalid CompMatr") );
823 }
824 }
825}
826
827
828
829/** @sa destroyDiagonalOp
830 * @ingroup deprecatedtests
831 * @author Tyson Jones
832 */
833TEST_CASE( "destroyDiagonalOp", "[data_structures]" ) {
834
835 /* there is no meaningful way to test this */
836 SUCCEED( );
837}
838
839
840
841/** @sa destroyPauliHamil
842 * @ingroup deprecatedtests
843 * @author Tyson Jones
844 */
845TEST_CASE( "destroyPauliHamil", "[data_structures]" ) {
846
847 /* there is no meaningful way to test this.
848 * We e.g. cannot check that the pointers are NULL because
849 * they are not updated; this function passes the struct by value,
850 * not by reference. We also cannot reliably monitor the
851 * memory used in the heap at runtime.
852 */
853 SUCCEED( );
854}
855
856
857
858/** @sa destroyQuESTEnv
859 * @ingroup deprecatedtests
860 * @author Tyson Jones
861 */
862TEST_CASE( "destroyQuESTEnv", "[data_structures]" ) {
863
864 /* there is no meaningful way to test this */
865 SUCCEED( );
866}
867
868
869
870/** @sa destroyQureg
871 * @ingroup deprecatedtests
872 * @author Tyson Jones
873 */
874TEST_CASE( "destroyQureg", "[data_structures]" ) {
875
876 /* there is no meaningful way to test this.
877 * We e.g. cannot check that the pointers are NULL because
878 * they are not updated; this function passes the struct by value,
879 * not by reference. We also cannot reliably monitor the
880 * memory used in the heap at runtime.
881 */
882 SUCCEED( );
883}
884
885
886
887/** @sa destroySubDiagonalOp
888 * @ingroup deprecatedtests
889 * @author Tyson Jones
890 */
891TEST_CASE( "destroySubDiagonalOp", "[data_structures]" ) {
892
893 /* there is no meaningful way to test this */
894 SUCCEED( );
895}
896
897
898
899// initComplexMatrixN removed because ComplexMatrixN is deprecated,
900// and replacement CompMatr has analogous function setCompMatr with
901// an incompatible signature
902
903 // /** @sa initComplexMatrixN
904 // * @ingroup deprecatedtests
905 // * @author Tyson Jones
906 // */
907 // TEST_CASE( "initComplexMatrixN", "[data_structures]" ) {
908
909 // /* use of this function is illegal in C++ */
910 // SUCCEED( );
911 // }
912
913
914
915// initDiagonalOp removed because DiagonalOp is deprecated, and
916// analogous function setFullStateDiagMatr of replacement
917// FullStateDiagMatr has an incompatible signature
918
919 /** @sa initDiagonalOp
920 * @ingroup deprecatedtests
921 * @author Tyson Jones
922 */
923 // TEST_CASE( "initDiagonalOp", "[data_structures]" ) {
924
925 // // must be at least one amplitude per node
926 // int minNumQb = calcLog2(getQuESTEnv().numNodes);
927 // if (minNumQb == 0)
928 // minNumQb = 1;
929
930 // SECTION( "correctness" ) {
931
932 // // try 10 valid number of qubits
933 // int numQb = GENERATE_COPY( range(minNumQb, minNumQb+10) );
934 // DiagonalOp op = createDiagonalOp(numQb, getQuESTEnv());
935
936 // long long int len = (1LL << numQb);
937 // vector<qcomp> elems(len);
938
939 // VLA(qreal, reals, len);
940 // VLA(qreal, imags, len);
941 // long long int n;
942 // for (n=0; n<len; n++) {
943 // reals[n] = (qreal) n;
944 // imags[n] = (qreal) -2*n; // (n - 2n i)
945 // }
946 // initDiagonalOp(op, reals, imags);
947
948 // // check that op.real and op.imag were modified...
949 // REQUIRE( areEqual(toQVector(op), reals, imags) );
950
951 // // and also that GPU real and imag were modified
952 // // via if it modifies an all-unity state-vec correctly
953 // Qureg qureg = createForcedQureg(numQb);
954 // for (long long int i=0; i<qureg.numAmpsPerChunk; i++) {
955 // qureg.stateVec.real[i] = 1;
956 // qureg.stateVec.imag[i] = 1;
957 // }
958 // copyStateToGPU(qureg);
959
960 // QVector prodRef = toQMatrix(op) * toQVector(qureg);
961
962 // // (n - 2n i) * (1 + 1i) = 3n - n*i
963 // applyDiagonalOp(qureg, op);
964 // copyStateFromGPU(qureg);
965 // QVector result = toQVector(qureg);
966 // REQUIRE( areEqual(prodRef, result) );
967
968 // destroyQureg(qureg, getQuESTEnv());
969 // destroyDiagonalOp(op, getQuESTEnv());
970 // }
971// }
972
973
974
975// initDiagonalOpFromPauliHamil removed because DiagonalOp is
976// deprecated in favour of FullStateDiagMatr, which indeed has
977// analogous function setFullStateDiagMatrFromPauliStrSum, but
978// the involved PauliStrSum is incompatible with the deprecated
979// PauliHamil
980
981 /** @sa initDiagonalOpFromPauliHamil
982 * @ingroup deprecatedtests
983 * @author Tyson Jones
984 */
985 // TEST_CASE( "initDiagonalOpFromPauliHamil", "[data_structures]" ) {
986
987 // // distributed diagonal op must contain at least one amplitude per node
988 // int minNumQb = calcLog2(getQuESTEnv().numNodes);
989 // if (minNumQb == 0)
990 // minNumQb = 1;
991
992 // SECTION( "correctness" ) {
993
994 // // try (at most) 10 valid number of qubits (even for validation)
995 // int numQb = GENERATE_COPY( range(minNumQb, std::min(10,minNumQb+10)) );
996 // DiagonalOp op = createDiagonalOp(numQb, getQuESTEnv());
997
998 // // try several sized random all-Z Hamiltonians
999 // int numTerms = GENERATE_COPY( 1, numQb, 5*numQb );
1000 // PauliHamil hamil = createPauliHamil(numQb, numTerms);
1001 // setRandomDiagPauliHamil(hamil);
1002
1003 // initDiagonalOpFromPauliHamil(op, hamil);
1004 // REQUIRE( areEqual(toQMatrix(op), toQMatrix(hamil)) );
1005
1006 // destroyPauliHamil(hamil);
1007 // destroyDiagonalOp(op, getQuESTEnv());
1008 // }
1009 // SECTION( "input validation" ) {
1010
1011 // SECTION( "hamiltonian parameters" ) {
1012
1013 // DiagonalOp op = createDiagonalOp(minNumQb, getQuESTEnv());
1014 // PauliHamil hamil;
1015
1016 // hamil.numQubits = GENERATE( -1, 0 );
1017 // hamil.numSumTerms = 1;
1018 // REQUIRE_THROWS_WITH( initDiagonalOpFromPauliHamil(op, hamil), ContainsSubstring("number of qubits") && ContainsSubstring("strictly positive") );
1019
1020 // hamil.numQubits = minNumQb;
1021 // hamil.numSumTerms = GENERATE( -1, 0 );
1022 // REQUIRE_THROWS_WITH( initDiagonalOpFromPauliHamil(op, hamil), ContainsSubstring("terms") && ContainsSubstring("strictly positive") );
1023
1024 // destroyDiagonalOp(op, getQuESTEnv());
1025 // }
1026 // SECTION( "mismatching dimensions" ) {
1027
1028 // int numQbA = minNumQb+1;
1029 // int numQbB = GENERATE_COPY( numQbA-1, numQbA+1 );
1030
1031 // DiagonalOp op = createDiagonalOp(numQbA, getQuESTEnv());
1032 // PauliHamil hamil = createPauliHamil(numQbB, 1);
1033
1034 // REQUIRE_THROWS_WITH( initDiagonalOpFromPauliHamil(op, hamil), ContainsSubstring("Pauli Hamiltonian and diagonal operator have different, incompatible dimensions") );
1035
1036 // destroyDiagonalOp(op, getQuESTEnv());
1037 // destroyPauliHamil(hamil);
1038 // }
1039 // SECTION( "pauli codes" ) {
1040
1041 // DiagonalOp op = createDiagonalOp(minNumQb, getQuESTEnv());
1042 // PauliHamil hamil = createPauliHamil(minNumQb, 5); // all I
1043
1044 // // make only one code invalid
1045 // int numCodes = minNumQb * hamil.numSumTerms;
1046 // int ind = GENERATE_COPY( range(0,numCodes) );
1047 // hamil.pauliCodes[ind] = GENERATE( PAULI_X, PAULI_Y );
1048
1049 // REQUIRE_THROWS_WITH( initDiagonalOpFromPauliHamil(op, hamil), ContainsSubstring("contained operators other than PAULI_Z and PAULI_I") );
1050
1051 // destroyDiagonalOp(op, getQuESTEnv());
1052 // destroyPauliHamil(hamil);
1053 // }
1054 // }
1055 // }
1056
1057
1058
1059// initPauliHamil removed because PauliHamil is deprecated in
1060// favour of PauliStrSum with (so far) an incompatible initialiser
1061
1062 /** @sa initPauliHamil
1063 * @ingroup deprecatedtests
1064 * @author Tyson Jones
1065 */
1066 // TEST_CASE( "initPauliHamil", "[data_structures]" ) {
1067
1068 // SECTION( "correctness" ) {
1069
1070 // PauliHamil hamil = createPauliHamil(3, 2);
1071
1072 // qreal coeffs[] = {-5, 5};
1073 // enum pauliOpType codes[] = {
1074 // PAULI_X, PAULI_Y, PAULI_Z,
1075 // PAULI_Z, PAULI_Y, PAULI_X};
1076 // initPauliHamil(hamil, coeffs, codes);
1077
1078 // // check everything written correctly
1079 // for (int t=0; t<2; t++) {
1080 // REQUIRE( coeffs[t] == hamil.termCoeffs[t] );
1081 // for (int q=0; q<3; q++) {
1082 // int ind = 3*t+q;
1083 // REQUIRE( codes[ind] == hamil.pauliCodes[ind] );
1084 // }
1085 // }
1086
1087 // destroyPauliHamil(hamil);
1088 // }
1089 // SECTION( "input validation" ) {
1090
1091 // SECTION( "parameters" ) {
1092
1093 // // parameters checked before codes, so safe to leave un-initialised
1094 // qreal coeffs[1];
1095 // enum pauliOpType codes[1];
1096 // PauliHamil hamil;
1097
1098 // hamil.numQubits = GENERATE( -1, 0 );
1099 // hamil.numSumTerms = 1;
1100 // REQUIRE_THROWS_WITH( initPauliHamil(hamil, coeffs, codes), ContainsSubstring("number of qubits") && ContainsSubstring("strictly positive") );
1101
1102 // hamil.numQubits = 1;
1103 // hamil.numSumTerms = GENERATE( -1, 0 );
1104 // REQUIRE_THROWS_WITH( initPauliHamil(hamil, coeffs, codes), ContainsSubstring("terms") && ContainsSubstring("strictly positive") );
1105 // }
1106 // SECTION( "Pauli codes" ) {
1107
1108 // int numQb = 3;
1109 // int numTerms = 2;
1110 // int numCodes = numQb * numTerms;
1111 // VLA(qreal, coeffs, numTerms);
1112 // VLA(pauliOpType, codes, numCodes);
1113
1114 // // make only one code invalid
1115 // for (int i=0; i<numCodes; i++)
1116 // codes[i] = PAULI_I;
1117 // codes[GENERATE_COPY( range(0,numCodes) )] = (pauliOpType) GENERATE( -1, 4 );
1118
1119 // PauliHamil hamil = createPauliHamil(numQb, numTerms);
1120 // REQUIRE_THROWS_WITH( initPauliHamil(hamil, coeffs, codes), ContainsSubstring("Invalid Pauli code") );
1121 // destroyPauliHamil(hamil);
1122 // }
1123 // }
1124 // }
1125
1126
1127
1128// setDiagonalOpElems removed because DiagonalOp is deprecated
1129// in favour of FullStateDiagMatr, which has an initialiser
1130// setFullStateDiagMatr() with an incompatible signature
1131
1132 // /** @sa setDiagonalOpElems
1133 // * @ingroup deprecatedtests
1134 // * @author Tyson Jones
1135 // */
1136 // TEST_CASE( "setDiagonalOpElems", "[data_structures]" ) {
1137
1138 // // must be at least one amplitude per node
1139 // int minNumQb = calcLog2(getQuESTEnv().numNodes);
1140 // if (minNumQb == 0)
1141 // minNumQb = 1;
1142
1143 // // try 10 valid number of qubits (even for validation)
1144 // int numQb = GENERATE_COPY( range(minNumQb, minNumQb+10) );
1145 // DiagonalOp op = createDiagonalOp(numQb, getQuESTEnv());
1146
1147 // SECTION( "correctness" ) {
1148
1149 // // make entire array on every node
1150 // long long int len = (1LL << numQb);
1151 // VLA(qreal, reals, len);
1152 // VLA(qreal, imags, len);
1153 // long long int n;
1154 // for (n=0; n<len; n++) {
1155 // reals[n] = (qreal) n;
1156 // imags[n] = (qreal) -2*n; // (n - 2n i)
1157 // }
1158
1159 // // set one value at a time (only relevant nodes will update)
1160 // for (n=0; n<len; n++)
1161 // setDiagonalOpElems(op, n, &reals[n], &imags[n], 1);
1162
1163 // // check op.real and op.imag updated correctly
1164 // REQUIRE( areEqual(toQVector(op), reals, imags) );
1165
1166 // // no check that GPU values updated (occurs in initDiagonalOp)
1167 // }
1168 // SECTION( "input validation" ) {
1169
1170 // long long int maxInd = (1LL << numQb);
1171 // qreal *reals = NULL;
1172 // qreal *imags = NULL;
1173
1174 // SECTION( "start index" ) {
1175
1176 // int startInd = GENERATE_COPY( -1, maxInd );
1177 // int numAmps = 1;
1178 // REQUIRE_THROWS_WITH( setDiagonalOpElems(op, startInd, reals, imags, numAmps), ContainsSubstring("Invalid element index") );
1179 // }
1180
1181 // SECTION( "number of elements" ) {
1182
1183 // // independent
1184 // int startInd = 0;
1185 // int numAmps = GENERATE_COPY( -1, maxInd+1 );
1186 // REQUIRE_THROWS_WITH( setDiagonalOpElems(op, startInd, reals, imags, numAmps), ContainsSubstring("Invalid number of elements") );
1187
1188 // // invalid considering start-index
1189 // startInd = maxInd - 1;
1190 // numAmps = 2;
1191 // REQUIRE_THROWS_WITH( setDiagonalOpElems(op, startInd, reals, imags, numAmps), ContainsSubstring("More elements given than exist") );
1192 // }
1193 // }
1194
1195 // destroyDiagonalOp(op);
1196 // }
1197
1198
1199
1200/** @sa syncDiagonalOp
1201 * @ingroup deprecatedtests
1202 * @author Tyson Jones
1203 */
1204TEST_CASE( "syncDiagonalOp", "[data_structures]" ) {
1205
1206 // must be at least one amplitude per node
1207 int minNumQb = calcLog2(getQuESTEnv().numNodes);
1208 if (minNumQb == 0)
1209 minNumQb = 1;
1210
1211 SECTION( "correctness" ) {
1212
1213 // try 10 valid number of qubits
1214 int numQb = GENERATE_COPY( range(minNumQb, minNumQb+10) );
1215
1216 // createDiagonalOp() redirects to createFullStateDiagMatr()
1217 // which auto-deploys and will auto disable distribution,
1218 // so we resort to custom invocation
1219
1220 QuESTEnv env = getQuESTEnv();
1221 DiagonalOp op = createCustomFullStateDiagMatr(numQb, env.isDistributed, env.isGpuAccelerated, env.isMultithreaded);
1222
1223 // check that changes get sync'd to the GPU...
1224 long long int n;
1225 for (n=0; n<op.numElemsPerNode; n++)
1226 op.cpuElems[n] = qcomp(n, -2*n);
1227 syncDiagonalOp(op);
1228
1229 // via if it modifies an all-unity state-vec correctly
1230 Qureg qureg = createForcedQureg(numQb);
1231 for (long long int i=0; i<qureg.numAmpsPerNode; i++)
1232 qureg.cpuAmps[i] = qcomp(1,1);
1233 copyStateToGPU(qureg);
1234
1235 // (n - 2n i) * (1 + 1i) = 3n - n*i
1236 applyDiagonalOp(qureg, op);
1237 copyStateFromGPU(qureg);
1238 for (n=0; n<qureg.numAmpsPerNode; n++) {
1239 REQUIRE( real(qureg.cpuAmps[n]) == 3*n );
1240 REQUIRE( imag(qureg.cpuAmps[n]) == -n );
1241 }
1242
1243 destroyQureg(qureg, getQuESTEnv());
1244 destroyDiagonalOp(op, getQuESTEnv());
1245 }
1246}
TEST_CASE("createCloneQureg", "[data_structures]")
unsigned int calcLog2(long unsigned int res)
bool areEqual(QVector a, QVector b)
vector< vector< qcomp > > QMatrix
QMatrix toQMatrix(CompMatr1 src)
vector< qcomp > QVector
QVector toQVector(Qureg qureg)
QuESTEnv getQuESTEnv()
FullStateDiagMatr createCustomFullStateDiagMatr(int numQubits, int useDistrib, int useGpuAccel, int useMultithread)
Definition matrices.cpp:322
Qureg createForcedQureg(int numQubits)
Definition qureg.cpp:293
Qureg createForcedDensityQureg(int numQubits)
Definition qureg.cpp:303
Qureg createCloneQureg(Qureg qureg)
Definition qureg.cpp:313
void destroyQureg(Qureg qureg)
Definition qureg.cpp:328
qmatrix getZeroMatrix(size_t dim)
Definition qmatrix.cpp:18
Definition qureg.h:42