The Quantum Exact Simulation Toolkit v4.0.0
Loading...
Searching...
No Matches
test_gates.cpp
1/** @file
2 * Ported tests of the deprecated QuEST v3 interface,
3 * unit testing the "gates" 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 */
12
13// must include catch.hpp first, because quest's
14// deprecation library will include a definition
15// of Vector{}
16#include <catch2/catch_test_macros.hpp>
17#include <catch2/catch_approx.hpp>
18#include <catch2/matchers/catch_matchers_string.hpp>
19#include <catch2/generators/catch_generators_range.hpp>
20
21// must define preprocessors to enable quest's
22// deprecated v3 API, and disable the numerous
23// warnings issued by its compilation
24#define INCLUDE_DEPRECATED_FUNCTIONS 1
25#define DISABLE_DEPRECATION_WARNINGS 1
26#include "quest/include/quest.h"
27
28#include "test_utilities.hpp"
29
30/* allows concise use of ContainsSubstring in catch's REQUIRE_THROWS_WITH */
31using Catch::Matchers::ContainsSubstring;
32using Catch::Approx;
33
34
35/** @sa collapseToOutcome
36 * @ingroup deprecatedtests
37 * @author Tyson Jones
38 */
39TEST_CASE( "collapseToOutcome", "[gates]" ) {
40
41 Qureg vec = createForcedQureg(NUM_QUBITS);
42 Qureg mat = createForcedDensityQureg(NUM_QUBITS);
43
44 SECTION( "correctness" ) {
45
46 int qubit = GENERATE( range(0,NUM_QUBITS) );
47 int outcome = GENERATE( 0, 1 );
48
49 // repeat these random tests 10 times on every qubit, and for both outcomes
50 GENERATE( range(0,10) );
51
52 SECTION( "state-vector" ) {
53
54 // use a random L2 state for every qubit & outcome
55 QVector vecRef = getRandomStateVector(NUM_QUBITS);
56 toQureg(vec, vecRef);
57
58 // calculate prob of outcome
59 qreal prob = 0;
60 for (size_t ind=0; ind<vecRef.size(); ind++) {
61 int bit = (ind >> qubit) & 1; // target-th bit
62 if (bit == outcome)
63 prob += pow(abs(vecRef[ind]), 2);
64 }
65
66 // renormalise by the outcome prob
67 for (size_t ind=0; ind<vecRef.size(); ind++) {
68 int bit = (ind >> qubit) & 1; // target-th bit
69 if (bit == outcome)
70 vecRef[ind] /= sqrt(prob);
71 else
72 vecRef[ind] = 0;
73 }
74
75 qreal res = collapseToOutcome(vec, qubit, outcome);
76 REQUIRE( res == Approx(prob) );
77 REQUIRE( areEqual(vec, vecRef) );
78 }
79 SECTION( "density-matrix" ) {
80
81 // use a random density matrix for every qubit & outcome
82 QMatrix matRef = getRandomDensityMatrix(NUM_QUBITS);
83 toQureg(mat, matRef);
84
85 // prob is sum of diagonal amps (should be real) where target bit is outcome
86 qcomp tr = 0;
87 for (size_t ind=0; ind<matRef.size(); ind++) {
88 int bit = (ind >> qubit) & 1; // qubit-th bit
89 if (bit == outcome)
90 tr += matRef[ind][ind];
91 }
92 REQUIRE( imag(tr) == Approx(0).margin(REAL_EPS) );
93 qreal prob = real(tr);
94
95 // renorm (/prob) every |*outcome*><*outcome*| state, zeroing all others
96 for (size_t r=0; r<matRef.size(); r++) {
97 for (size_t c=0; c<matRef.size(); c++) {
98 int ketBit = (c >> qubit) & 1;
99 int braBit = (r >> qubit) & 1;
100
101 if (ketBit == outcome && braBit == outcome)
102 matRef[r][c] /= prob;
103 else
104 matRef[r][c] = 0;
105 }
106 }
107
108 qreal res = collapseToOutcome(mat, qubit, outcome);
109 REQUIRE( res == Approx(prob) );
110 REQUIRE( areEqual(mat, matRef) );
111 }
112 }
113 SECTION( "input validation" ) {
114
115 SECTION( "qubit index" ) {
116
117 int qubit = GENERATE( -1, NUM_QUBITS );
118 int outcome = 0;
119 REQUIRE_THROWS_WITH( collapseToOutcome(mat, qubit, outcome), ContainsSubstring("Invalid target qubit") );
120 }
121 SECTION( "outcome value" ) {
122
123 int qubit = 0;
124 int outcome = GENERATE( -1, 2 );
125 REQUIRE_THROWS_WITH( collapseToOutcome(mat, qubit, outcome), ContainsSubstring("qubit measurement outcome") && ContainsSubstring("invalid") );
126 }
127 SECTION( "outcome probability" ) {
128
129 initZeroState(vec);
130 REQUIRE_THROWS_WITH( collapseToOutcome(vec, 0, 1), ContainsSubstring("impossibly unlikely") );
131 initClassicalState(vec, 1);
132 REQUIRE_THROWS_WITH( collapseToOutcome(vec, 0, 0), ContainsSubstring("impossibly unlikely") );
133 }
134 }
135 destroyQureg(vec);
136 destroyQureg(mat);
137}
138
139
140
141/** @sa measure
142 * @ingroup deprecatedtests
143 * @author Tyson Jones
144 */
145TEST_CASE( "measure", "[gates]" ) {
146
147 Qureg vec = createForcedQureg(NUM_QUBITS);
148 Qureg mat = createForcedDensityQureg(NUM_QUBITS);
149
150 SECTION( "correctness" ) {
151
152 int qubit = GENERATE( range(0,NUM_QUBITS) );
153
154 // repeat these random tests 10 times on every qubit
155 GENERATE( range(0,10) );
156
157 SECTION( "state-vector" ) {
158
159 QVector vecRef = getRandomStateVector(NUM_QUBITS);
160 toQureg(vec, vecRef);
161
162 int outcome = measure(vec, qubit);
163 REQUIRE( (outcome == 0 || outcome == 1) );
164
165 // calculate prob of this outcome
166 qreal prob = 0;
167 for (size_t ind=0; ind<vecRef.size(); ind++) {
168 int bit = (ind >> qubit) & 1; // target-th bit
169 if (bit == outcome)
170 prob += pow(abs(vecRef[ind]), 2);
171 }
172
173 REQUIRE( prob > REAL_EPS );
174
175 // renormalise by the outcome prob
176 for (size_t ind=0; ind<vecRef.size(); ind++) {
177 int bit = (ind >> qubit) & 1; // target-th bit
178 if (bit == outcome)
179 vecRef[ind] /= sqrt(prob);
180 else
181 vecRef[ind] = 0;
182 }
183 REQUIRE( areEqual(vec, vecRef) );
184 }
185 SECTION( "density-matrix" ) {
186
187 QMatrix matRef = getRandomDensityMatrix(NUM_QUBITS);
188 toQureg(mat, matRef);
189
190 int outcome = measure(mat, qubit);
191 REQUIRE( (outcome == 0 || outcome == 1) );
192
193 // compute prob of this outcome
194 qreal prob = 0;
195 for (size_t ind=0; ind<matRef.size(); ind++) {
196 int bit = (ind >> qubit) & 1; // qubit-th bit
197 if (bit == outcome)
198 prob += real(matRef[ind][ind]);
199 }
200
201 REQUIRE( prob > REAL_EPS );
202
203 // renorm (/prob) every |*outcome*><*outcome*| state, zeroing all others
204 for (size_t r=0; r<matRef.size(); r++) {
205 for (size_t c=0; c<matRef.size(); c++) {
206 int ketBit = (c >> qubit) & 1;
207 int braBit = (r >> qubit) & 1;
208
209 if (ketBit == outcome && braBit == outcome)
210 matRef[r][c] /= prob;
211 else
212 matRef[r][c] = 0;
213 }
214 }
215
216 REQUIRE( areEqual(mat, matRef) );
217 }
218 }
219 SECTION( "input validation" ) {
220
221 SECTION( "qubit index" ) {
222
223 int qubit = GENERATE( -1, NUM_QUBITS );
224 REQUIRE_THROWS_WITH( measure(vec, qubit), ContainsSubstring("Invalid target qubit") );
225 }
226 }
227 destroyQureg(vec);
228 destroyQureg(mat);
229}
230
231
232
233/** @sa measureWithStats
234 * @ingroup deprecatedtests
235 * @author Tyson Jones
236 */
237TEST_CASE( "measureWithStats", "[gates]" ) {
238
239 Qureg vec = createForcedQureg(NUM_QUBITS);
240 Qureg mat = createForcedDensityQureg(NUM_QUBITS);
241
242 SECTION( "correctness" ) {
243
244 int qubit = GENERATE( range(0,NUM_QUBITS) );
245
246 // repeat these random tests 10 times on every qubit
247 GENERATE( range(0,10) );
248
249 SECTION( "state-vector" ) {
250
251 QVector vecRef = getRandomStateVector(NUM_QUBITS);
252 toQureg(vec, vecRef);
253
254 qreal res;
255 int outcome = measureWithStats(vec, qubit, &res);
256 REQUIRE( (outcome == 0 || outcome == 1) );
257
258 // calculate prob of this outcome
259 qreal prob = 0;
260 for (size_t ind=0; ind<vecRef.size(); ind++) {
261 int bit = (ind >> qubit) & 1; // target-th bit
262 if (bit == outcome)
263 prob += pow(abs(vecRef[ind]), 2);
264 }
265
266 REQUIRE( prob == Approx(res) );
267
268 // renormalise by the outcome prob
269 for (size_t ind=0; ind<vecRef.size(); ind++) {
270 int bit = (ind >> qubit) & 1; // target-th bit
271 if (bit == outcome)
272 vecRef[ind] /= sqrt(prob);
273 else
274 vecRef[ind] = 0;
275 }
276 REQUIRE( areEqual(vec, vecRef) );
277 }
278 SECTION( "density-matrix" ) {
279
280 QMatrix matRef = getRandomDensityMatrix(NUM_QUBITS);
281 toQureg(mat, matRef);
282
283 qreal res;
284 int outcome = measureWithStats(mat, qubit, &res);
285 REQUIRE( (outcome == 0 || outcome == 1) );
286
287 // compute prob of this outcome
288 qreal prob = 0;
289 for (size_t ind=0; ind<matRef.size(); ind++) {
290 int bit = (ind >> qubit) & 1; // qubit-th bit
291 if (bit == outcome)
292 prob += real(matRef[ind][ind]);
293 }
294
295 REQUIRE( prob == Approx(res) );
296
297 // renorm (/prob) every |*outcome*><*outcome*| state, zeroing all others
298 for (size_t r=0; r<matRef.size(); r++) {
299 for (size_t c=0; c<matRef.size(); c++) {
300 int ketBit = (c >> qubit) & 1;
301 int braBit = (r >> qubit) & 1;
302
303 if (ketBit == outcome && braBit == outcome)
304 matRef[r][c] /= prob;
305 else
306 matRef[r][c] = 0;
307 }
308 }
309
310 REQUIRE( areEqual(mat, matRef) );
311 }
312 }
313 SECTION( "input validation" ) {
314
315 SECTION( "qubit index" ) {
316
317 int qubit = GENERATE( -1, NUM_QUBITS );
318 qreal res;
319 REQUIRE_THROWS_WITH( measureWithStats(vec, qubit, &res), ContainsSubstring("Invalid target qubit") );
320 }
321 }
322 destroyQureg(vec);
323 destroyQureg(mat);
324}
TEST_CASE("collapseToOutcome", "[gates]")
bool areEqual(QVector a, QVector b)
vector< vector< qcomp > > QMatrix
vector< qcomp > QVector
void toQureg(Qureg qureg, QVector vec)
void initZeroState(Qureg qureg)
void initClassicalState(Qureg qureg, qindex stateInd)
Qureg createForcedQureg(int numQubits)
Definition qureg.cpp:293
Qureg createForcedDensityQureg(int numQubits)
Definition qureg.cpp:303
void destroyQureg(Qureg qureg)
Definition qureg.cpp:328
qvector getRandomStateVector(int numQb)
Definition random.cpp:274
qmatrix getRandomDensityMatrix(int numQb)
Definition random.cpp:286
Definition qureg.h:42