The Quantum Exact Simulation Toolkit v4.0.0
Loading...
Searching...
No Matches
initialisations.cpp
1/** @file
2 * Unit tests of the initialisations module.
3 *
4 * @author Tyson Jones
5 *
6 * @defgroup unitinit Initialisation
7 * @ingroup unittests
8 */
9
10#include "quest/include/quest.h"
11
12#include <catch2/catch_test_macros.hpp>
13#include <catch2/generators/catch_generators_range.hpp>
14
15#include "tests/utils/qvector.hpp"
16#include "tests/utils/qmatrix.hpp"
17#include "tests/utils/cache.hpp"
18#include "tests/utils/compare.hpp"
19#include "tests/utils/convert.hpp"
20#include "tests/utils/evolve.hpp"
21#include "tests/utils/linalg.hpp"
22#include "tests/utils/lists.hpp"
23#include "tests/utils/macros.hpp"
24#include "tests/utils/measure.hpp"
25#include "tests/utils/random.hpp"
26
27
28
29/*
30 * UTILITIES
31 */
32
33
34#define TEST_CATEGORY \
35 LABEL_UNIT_TAG "[initialisations]"
36
37
38void TEST_ON_CACHED_QUREGS(quregCache quregs, auto testFunc) {
39
40 for (auto& [label, qureg]: quregs) {
41
42 DYNAMIC_SECTION( label ) {
43
44 testFunc(qureg);
45 }
46 }
47}
48
49
50void TEST_ON_CACHED_QUREGS(quregCache quregs, auto apiFunc, auto refState) {
51
52 // assumes refState is already initialised
53
54 auto testFunc = [&](Qureg qureg) {
55 apiFunc(qureg);
56 REQUIRE_AGREE( qureg, refState );
57 };
58
59 TEST_ON_CACHED_QUREGS(quregs, testFunc);
60}
61
62
63
64/**
65 * TESTS
66 *
67 * @ingroup unitinit
68 * @{
69 */
70
71
72TEST_CASE( "initBlankState", TEST_CATEGORY ) {
73
74 SECTION( LABEL_CORRECTNESS ) {
75
76 SECTION( LABEL_STATEVEC ) { TEST_ON_CACHED_QUREGS(getCachedStatevecs(), initBlankState, getRefStatevec()); }
77 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), initBlankState, getRefDensmatr()); }
78 }
79
80 /// @todo input validation
81}
82
83
84TEST_CASE( "initZeroState", TEST_CATEGORY ) {
85
86 SECTION( LABEL_CORRECTNESS ) {
87
88 qvector refVec = getRefStatevec(); refVec[0] = 1; // |0> = {1, 0...}
89 qmatrix refMat = getRefDensmatr(); refMat[0][0] = 1; // |0><0| = {{1,0...},{0...}...}
90
91 SECTION( LABEL_STATEVEC ) { TEST_ON_CACHED_QUREGS(getCachedStatevecs(), initZeroState, refVec); }
92 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), initZeroState, refMat); }
93 }
94
95 /// @todo input validation
96}
97
98
99TEST_CASE( "initPlusState", TEST_CATEGORY ) {
100
101 SECTION( LABEL_CORRECTNESS ) {
102
103 int numQubits = getNumCachedQubits();
104 qreal vecElem = 1. / std::sqrt(getPow2(numQubits));
105 qreal matElem = 1. / getPow2(numQubits);
106
107 qvector refVec = getConstantVector(getPow2(numQubits), vecElem); // |+> = 1/sqrt(2^N) {1, ...}
108 qmatrix refMat = getConstantMatrix(getPow2(numQubits), matElem); // |+><+| = 1/2^N {{1, ...}, ...}
109
110 SECTION( LABEL_STATEVEC ) { TEST_ON_CACHED_QUREGS(getCachedStatevecs(), initPlusState, refVec); }
111 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), initPlusState, refMat); }
112 }
113
114 /// @todo input validation
115}
116
117
118TEST_CASE( "initClassicalState", TEST_CATEGORY ) {
119
120 SECTION( LABEL_CORRECTNESS ) {
121
122 int numQubits = getNumCachedQubits();
123 int numInds = (int) getPow2(numQubits);
124 int stateInd = GENERATE_COPY( range(0,numInds) );
125
126 qvector refVec = getRefStatevec(); refVec[stateInd] = 1; // |i> = {0, ..., 1, 0, ...}
127 qmatrix refMat = getRefDensmatr(); refMat[stateInd][stateInd] = 1; // |i><i|
128
129 auto apiFunc = [&](Qureg qureg) { initClassicalState(qureg, stateInd); };
130
131 SECTION( LABEL_STATEVEC ) { TEST_ON_CACHED_QUREGS(getCachedStatevecs(), apiFunc, refVec); }
132 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), apiFunc, refMat); }
133 }
134
135 /// @todo input validation
136}
137
138
139TEST_CASE( "initDebugState", TEST_CATEGORY ) {
140
141 SECTION( LABEL_CORRECTNESS ) {
142
143 qvector refVec = getRefStatevec(); setToDebugState(refVec); // |debug>
144 qmatrix refMat = getRefDensmatr(); setToDebugState(refMat); // ||debug>>
145
146 SECTION( LABEL_STATEVEC ) { TEST_ON_CACHED_QUREGS(getCachedStatevecs(), initDebugState, refVec); }
147 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), initDebugState, refMat); }
148 }
149
150 /// @todo input validation
151}
152
153
154TEST_CASE( "initRandomPureState", TEST_CATEGORY ) {
155
156 SECTION( LABEL_CORRECTNESS ) {
157
158 // this test does not use reference states
159 GENERATE( range(0,10) );
160
161 auto testFunc = [&](Qureg qureg) {
162
163 initRandomPureState(qureg);
164
165 /// @todo
166 /// these not-all-same-amp checks can be made much more rigorous,
167 /// by e.g. asserting distinct nodes haven't generated all the same
168 /// amplitudes (we currently observe this by eye)
169 syncQuregFromGpu(qureg);
170 REQUIRE( qureg.cpuAmps[0] != qureg.cpuAmps[1] );
171
172 qreal prob = calcTotalProb(qureg);
173 REQUIRE_AGREE( prob, 1 );
174
175 qreal purity = calcPurity(qureg);
176 REQUIRE_AGREE( purity, 1 );
177 };
178
179 SECTION( LABEL_STATEVEC ) { TEST_ON_CACHED_QUREGS(getCachedStatevecs(), testFunc); }
180 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), testFunc); }
181 }
182
183 /// @todo input validation
184}
185
186
187TEST_CASE( "initRandomMixedState", TEST_CATEGORY ) {
188
189 SECTION( LABEL_CORRECTNESS ) {
190
191 // this test does not use reference states
192
193 GENERATE( range(0,10) );
194 int numPureStates = GENERATE( 1, 2, 10 );
195
196 auto testFunc = [&](Qureg qureg) {
197
198 initRandomMixedState(qureg, numPureStates);
199
200 /// @todo
201 /// these not-all-same-amp checks can be made much more rigorous,
202 /// by e.g. asserting distinct nodes haven't generated all the same
203 /// amplitudes (we currently observe this by eye)
204 syncQuregFromGpu(qureg);
205 REQUIRE( qureg.cpuAmps[0] != qureg.cpuAmps[1] ); // performed on all nodes
206
207 qreal prob = calcTotalProb(qureg);
208 REQUIRE_AGREE( prob, 1 );
209
210 qreal purity = calcPurity(qureg);
211 if (numPureStates == 1)
212 REQUIRE_AGREE( purity, 1 );
213 else
214 REQUIRE( purity < 1 );
215 };
216
217 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), testFunc); }
218 }
219
220 /// @todo input validation
221}
222
223
224TEST_CASE( "initArbitraryPureState", TEST_CATEGORY ) {
225
226 SECTION( LABEL_CORRECTNESS ) {
227
228 // works for unnormalised states
229 qvector refVec = getRandomVector(getPow2(getNumCachedQubits()));
230 qmatrix refMat = getOuterProduct(refVec, refVec);
231
232 auto apiFunc = [&](Qureg qureg) { initArbitraryPureState(qureg, refVec.data()); };
233
234 SECTION( LABEL_STATEVEC ) { TEST_ON_CACHED_QUREGS(getCachedStatevecs(), apiFunc, refVec); }
235 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), apiFunc, refMat); }
236 }
237
238 /// @todo input validation
239}
240
241
242
243
244TEST_CASE( "setQuregAmps", TEST_CATEGORY ) {
245
246 SECTION( LABEL_CORRECTNESS ) {
247
248 int numTotalAmps = getPow2(getNumCachedQubits());
249 int numSetAmps = GENERATE_COPY( range(0,numTotalAmps+1) );
250 int startInd = GENERATE_COPY( range(0,numTotalAmps-numSetAmps) );
251 qvector amps = getRandomVector(numSetAmps);
252
253 auto testFunc = [&](Qureg qureg) {
254
255 // initialise qureg randomly
256 qvector refVec = getRandomVector(numTotalAmps);
257 setQuregToReference(qureg, refVec);
258
259 // modify only subset of refVec amps and qureg...
260 setSubVector(refVec, amps, startInd);
261 setQuregAmps(qureg, startInd, amps.data(), numSetAmps);
262
263 // so that we simultaneously check targeted amps
264 // are modified while non-targeted are not
265 REQUIRE_AGREE( qureg, refVec );
266 };
267
268 SECTION( LABEL_STATEVEC ) { TEST_ON_CACHED_QUREGS(getCachedStatevecs(), testFunc); }
269 }
270
271 /// @todo input validation
272}
273
274
275TEST_CASE( "setDensityQuregFlatAmps", TEST_CATEGORY ) {
276
277 SECTION( LABEL_CORRECTNESS ) {
278
279 int numTotalRows = getPow2(getNumCachedQubits());
280 int numTotalAmps = numTotalRows * numTotalRows;
281
282 // systematic iteration is WAY too slow
283 GENERATE( range(0,1000) );
284 int numSetAmps = getRandomInt(0, numTotalAmps + 1);
285 int startInd = getRandomInt(0, numTotalAmps - numSetAmps);
286 qvector amps = getRandomVector(numSetAmps);
287
288 auto testFunc = [&](Qureg qureg) {
289
290 // initialise qureg randomly
291 qmatrix refMat = getRandomMatrix(numTotalRows);
292 setQuregToReference(qureg, refMat);
293
294 // overwrite a contiguous region of row-major refMat, column-wise
295 refMat = getTranspose(refMat);
296 setSubMatrix(refMat, amps, startInd);
297 refMat = getTranspose(refMat);
298
299 // modify the same contiguous region of column-major qureg
300 setDensityQuregFlatAmps(qureg, startInd, amps.data(), numSetAmps);
301
302 // check that both targeted and non-targeted amps agree
303 REQUIRE_AGREE( qureg, refMat );
304 };
305
306 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), testFunc); }
307 }
308
309 /// @todo input validation
310}
311
312
313TEST_CASE( "setDensityQuregAmps", TEST_CATEGORY ) {
314
315 //void setDensityQuregAmps(Qureg qureg, qindex startRow, qindex startCol, qcomp** amps, qindex numRows, qindex numCols);
316
317
318 SECTION( LABEL_CORRECTNESS ) {
319
320 int numTotalRowsCols = getPow2(getNumCachedQubits());
321
322 // systematic iteration is WAY too slow
323 GENERATE( range(0,1000) );
324 int numSetRows = getRandomInt(1, numTotalRowsCols+1);
325 int numSetCols = getRandomInt(1, numTotalRowsCols+1);
326 int startRow = getRandomInt(0, numTotalRowsCols - numSetRows);
327 int startCol = getRandomInt(0, numTotalRowsCols - numSetCols);
328
329 // caution that amps is 'qmatrix' despite not being square
330 qmatrix amps = getRandomNonSquareMatrix(numSetRows, numSetCols);
331
332 auto testFunc = [&](Qureg qureg) {
333
334 // initialise qureg randomly
335 qmatrix refMat = getRandomMatrix(numTotalRowsCols);
336 setQuregToReference(qureg, refMat);
337
338 // API needs nested pointers
339 std::vector<qcomp*> rowPtrs(numSetRows);
340 for (size_t r=0; r<numSetRows; r++)
341 rowPtrs[r] = amps[r].data();
342
343 // overwrite a sub-matrix of refMat and Qureg
344 setSubMatrix(refMat, amps, startRow, startCol);
345 setDensityQuregAmps(qureg, startRow, startCol, rowPtrs.data(), numSetRows, numSetCols);
346
347 // check that both targeted and non-targeted amps agree
348 REQUIRE_AGREE( qureg, refMat );
349 };
350
351 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), testFunc); }
352
353 }
354
355 /// @todo input validation
356}
357
358
359TEST_CASE( "setQuregToRenormalized", TEST_CATEGORY ) {
360
361 SECTION( LABEL_CORRECTNESS ) {
362
363 GENERATE( range(0,10) );
364 qindex dim = getPow2(getNumCachedQubits());
365 qvector refVec = getRandomVector(dim);
366 qmatrix refMat = getRandomMatrix(dim);
367
368 // eliminate random chance of tr(refMat)=0, triggering validation
369 if (doScalarsAgree(getTrace(refMat), 0))
370 refMat[0][0] += 1/(qreal) dim;
371
372 // [=] stores current (pre-normalised) reference objects
373 auto funcVec = [=](Qureg qureg) { setQuregToReference(qureg, refVec); setQuregToRenormalized(qureg); };
374 auto funcMat = [=](Qureg qureg) { setQuregToReference(qureg, refMat); setQuregToRenormalized(qureg); };
375
376 // setQuregToRenormalized() makes statevectors become valid
377 refVec = getNormalised(refVec);
378
379 // but it only divides density matrices by the sum of the real-elems of their diagonals
380 refMat /= getReferenceProbability(refMat);
381
382 SECTION( LABEL_STATEVEC ) { TEST_ON_CACHED_QUREGS(getCachedStatevecs(), funcVec, refVec); }
383 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), funcMat, refMat); }
384 }
385
386 /// @todo input validation
387}
388
389
390TEST_CASE( "setQuregToPauliStrSum", TEST_CATEGORY ) {
391
392 SECTION( LABEL_CORRECTNESS ) {
393
394 GENERATE( range(0,10) );
395 int numQubits = getNumCachedQubits();
396 int numTerms = GENERATE_COPY( 1, numQubits, getPow2(2*numQubits) );
397 PauliStrSum sum = createRandomPauliStrSum(numQubits, numTerms);
398 qmatrix refMat = getMatrix(sum, numQubits);
399
400 auto apiFunc = [&](Qureg qureg) { setQuregToPauliStrSum(qureg, sum); };
401
402 SECTION( LABEL_DENSMATR ) { TEST_ON_CACHED_QUREGS(getCachedDensmatrs(), apiFunc, refMat); }
403 }
404
405 /// @todo input validation
406}
407
408
409/** @} (end defgroup) */
410
411
412
413/**
414 * @todo
415 * UNTESTED FUNCTIONS
416 */
417
418// these require we deploy the Quregs differently
419// to thoroughly test all QuEST control flows
420
421void initPureState(Qureg qureg, Qureg pure);
422
423void setQuregToClone(Qureg targetQureg, Qureg copyQureg);
424
425void setQuregToSuperposition(qcomp facOut, Qureg out, qcomp fac1, Qureg qureg1, qcomp fac2, Qureg qureg2);
426
427void setQuregToPartialTrace(Qureg out, Qureg in, int* traceOutQubits, int numTraceQubits);
428
429void setQuregToReducedDensityMatrix(Qureg out, Qureg in, int* retainQubits, int numRetainQubits);
qreal calcPurity(Qureg qureg)
qreal calcTotalProb(Qureg qureg)
void setDensityQuregFlatAmps(Qureg qureg, qindex startInd, qcomp *amps, qindex numAmps)
void setQuregToReducedDensityMatrix(Qureg out, Qureg in, int *retainQubits, int numRetainQubits)
void setQuregToPauliStrSum(Qureg qureg, PauliStrSum sum)
void setQuregAmps(Qureg qureg, qindex startInd, qcomp *amps, qindex numAmps)
void setQuregToPartialTrace(Qureg out, Qureg in, int *traceOutQubits, int numTraceQubits)
qreal setQuregToRenormalized(Qureg qureg)
void setQuregToClone(Qureg targetQureg, Qureg copyQureg)
void setDensityQuregAmps(Qureg qureg, qindex startRow, qindex startCol, qcomp **amps, qindex numRows, qindex numCols)
void setQuregToSuperposition(qcomp facOut, Qureg out, qcomp fac1, Qureg qureg1, qcomp fac2, Qureg qureg2)
void initArbitraryPureState(Qureg qureg, qcomp *amps)
void initRandomPureState(Qureg qureg)
void initPlusState(Qureg qureg)
void initZeroState(Qureg qureg)
void initPureState(Qureg qureg, Qureg pure)
void initDebugState(Qureg qureg)
void initRandomMixedState(Qureg qureg, qindex numPureStates)
void initClassicalState(Qureg qureg, qindex ind)
void initBlankState(Qureg qureg)
void syncQuregFromGpu(Qureg qureg)
Definition qureg.cpp:397
void setSubMatrix(qmatrix &dest, qmatrix sub, size_t r, size_t c)
Definition qmatrix.cpp:203
int getRandomInt(int min, int maxExcl)
Definition random.cpp:76
TEST_CASE("initBlankState", TEST_CATEGORY)
Definition qureg.h:42