36TEST_CASE(
"calcDensityInnerProduct",
"[calculations]" ) {
41 SECTION(
"correctness" ) {
44 GENERATE( range(0,10) );
46 SECTION(
"density-matrix" ) {
60 for (
size_t i=0; i<r1.size(); i++)
61 prod += conj(r1[i]) * r2[i];
62 qreal densProd = pow(abs(prod),2);
64 REQUIRE( real(calcDensityInnerProduct(mat1,mat2)) == Approx(densProd).margin(100 * REAL_EPS) );
75 for (
size_t i=0; i<ref1.size(); i++)
76 for (
size_t j=0; j<ref1.size(); j++)
77 refProd += conj(ref1[i][j]) * ref2[i][j];
78 REQUIRE( imag(refProd) == Approx(0).margin(REAL_EPS) );
80 REQUIRE( real(calcDensityInnerProduct(mat1,mat2)) == Approx(real(refProd)).margin(100 * REAL_EPS) );
83 REQUIRE( real(calcDensityInnerProduct(mat1,mat2)) == Approx(real(calcDensityInnerProduct(mat2,mat1))).margin(100 * REAL_EPS) );
85 SECTION(
"unnormalised" ) {
95 for (
size_t i=0; i<ref1.size(); i++)
96 for (
size_t j=0; j<ref1.size(); j++)
97 refProd += conj(ref1[i][j]) * ref2[i][j];
99 REQUIRE( real(calcDensityInnerProduct(mat1,mat2)) == Approx(real(refProd)).margin(100 * REAL_EPS) );
103 SECTION(
"input validation" ) {
105 SECTION(
"dimensions" ) {
108 REQUIRE_THROWS_WITH( calcDensityInnerProduct(mat1,mat3), ContainsSubstring(
"differing numbers of qubits") );
145 SECTION(
"correctness" ) {
148 GENERATE( range(0,10) );
151 DiagonalOp op = createDiagonalOp(NUM_QUBITS,
getQuESTEnv());
152 for (
long long int i=0; i<op.numElemsPerNode; i++)
156 SECTION(
"state-vector" ) {
162 for (
size_t i=0; i<vecRef.size(); i++)
163 prod += conj(vecRef[i]) * sumRef[i];
165 qcomp res = calcExpecDiagonalOp(vec, op);
166 REQUIRE( real(res) == Approx(real(prod)).margin(REAL_EPS) );
167 REQUIRE( imag(res) == Approx(imag(prod)).margin(REAL_EPS) );
169 SECTION(
"density-matrix" ) {
174 for (
size_t i=0; i<matRef.size(); i++)
177 qcomp res = calcExpecDiagonalOp(mat, op);
178 REQUIRE( real(res) == Approx(real(tr)).margin(100*REAL_EPS) );
179 REQUIRE( imag(res) == Approx(imag(tr)).margin(100*REAL_EPS) );
184 SECTION(
"input validation" ) {
186 SECTION(
"mismatching size" ) {
188 DiagonalOp op = createDiagonalOp(NUM_QUBITS + 1,
getQuESTEnv());
190 REQUIRE_THROWS_WITH( calcExpecDiagonalOp(vec, op), ContainsSubstring(
"different number of qubits"));
191 REQUIRE_THROWS_WITH( calcExpecDiagonalOp(mat, op), ContainsSubstring(
"different number of qubits"));
325 Qureg vec =
createCustomQureg(NUM_QUBITS, 0, env.isDistributed, env.isGpuAccelerated, env.isMultithreaded);
326 Qureg mat =
createCustomQureg(NUM_QUBITS, 1, env.isDistributed, env.isGpuAccelerated, env.isMultithreaded);
336 SECTION(
"correctness" ) {
338 int numTargs = GENERATE( range(1,NUM_QUBITS+1) );
339 int* targs = GENERATE_COPY( sublists(range(0,NUM_QUBITS), numTargs) );
346 GENERATE( range(0,10) );
347 vector<pauliOpType> paulis(numTargs);
348 for (
int i=0; i<numTargs; i++)
354 QMatrix yMatr{{0,-qcomp(0,1)},{qcomp(0,1),0}};
357 for (
int i=0; i<numTargs; i++) {
359 if (paulis[i] == PAULI_I) fac = iMatr;
360 if (paulis[i] == PAULI_X) fac = xMatr;
361 if (paulis[i] == PAULI_Y) fac = yMatr;
362 if (paulis[i] == PAULI_Z) fac = zMatr;
366 SECTION(
"state-vector" ) {
373 for (
size_t i=0; i<vecRef.size(); i++)
374 prod += conj(vecRef[i]) * prodRef[i];
375 REQUIRE( imag(prod) == Approx(0).margin(REAL_EPS) );
377 qreal res = calcExpecPauliProd(vec, targs, paulis.data(), numTargs, vecWork);
378 REQUIRE( res == Approx(real(prod)).margin(REAL_EPS) );
380 SECTION(
"density-matrix" ) {
386 matRef = fullOp * matRef;
390 for (
size_t i=0; i<matRef.size(); i++)
391 tr += real(matRef[i][i]);
396 qreal res = calcExpecPauliProd(mat, targs, paulis.data(), numTargs, matWork);
399 REQUIRE( res == Approx(tr).margin(10*REAL_EPS) );
402 SECTION(
"input validation" ) {
404 pauliOpType pauliOpsToAvoidDeprecSegFault[100];
405 for (
int i=0; i<100; i++)
406 pauliOpsToAvoidDeprecSegFault[i] = PAULI_X;
408 SECTION(
"number of targets" ) {
410 int targs[NUM_QUBITS+1];
411 for (
int i=0; i<NUM_QUBITS+1; i++)
415 int numTargs = GENERATE( -1, 0 );
416 REQUIRE_THROWS_WITH( calcExpecPauliProd(vec, targs, pauliOpsToAvoidDeprecSegFault, numTargs, vecWork), ContainsSubstring(
"Invalid number of Paulis") );
419 numTargs = NUM_QUBITS + 1;
420 REQUIRE_THROWS_WITH( calcExpecPauliProd(vec, targs, pauliOpsToAvoidDeprecSegFault, numTargs, vecWork), ContainsSubstring(
"highest-index non-identity Pauli operator") && ContainsSubstring(
"exceeds the maximum target") );
422 SECTION(
"target indices" ) {
425 int targs[3] = {0, 1, 2};
427 int badInd = GENERATE( range(0,3) );
431 REQUIRE_THROWS_WITH( calcExpecPauliProd(vec, targs, pauliOpsToAvoidDeprecSegFault, numTargs, vecWork), ContainsSubstring(
"Pauli indices must be non-negative") );
434 targs[badInd] = NUM_QUBITS;
435 REQUIRE_THROWS_WITH( calcExpecPauliProd(vec, targs, pauliOpsToAvoidDeprecSegFault, numTargs, vecWork), ContainsSubstring(
"highest-index non-identity Pauli operator") );
439 REQUIRE_THROWS_WITH( calcExpecPauliProd(vec, targs, pauliOpsToAvoidDeprecSegFault, numTargs, vecWork), ContainsSubstring(
"exceed the maximum number of representable Pauli operators") );
441 SECTION(
"repetition in targets" ) {
444 int targs[3] = {0, 1, 1};
445 REQUIRE_THROWS_WITH( calcExpecPauliProd(vec, targs, pauliOpsToAvoidDeprecSegFault, numTargs, vecWork), ContainsSubstring(
"Indices must be unique") );
447 SECTION(
"pauli codes" ) {
450 int targs[3] = {0, 1, 2};
451 pauliOpType codes[3] = {PAULI_X, PAULI_Y, PAULI_Z};
454 codes[GENERATE( range(0,3) )] = (pauliOpType) GENERATE( -1, 4 );
455 REQUIRE_THROWS_WITH( calcExpecPauliProd(vec, targs, codes, numTargs, vecWork), ContainsSubstring(
"invalid Pauli code") );
510 SECTION(
"correctness" ) {
512 int numSumTerms = GENERATE( 1, 2, 10, 15 );
517 GENERATE( range(0,10) );
518 int totNumCodes = numSumTerms * NUM_QUBITS;
519 vector<pauliOpType> paulis(totNumCodes);
520 vector<qreal> coeffs(numSumTerms);
524 QMatrix pauliSum =
toQMatrix(coeffs.data(), paulis.data(), NUM_QUBITS, numSumTerms);
526 SECTION(
"state-vector" ) {
530 QVector sumRef = pauliSum * vecRef;
532 for (
size_t i=0; i<vecRef.size(); i++)
533 prod += conj(vecRef[i]) * sumRef[i];
534 REQUIRE( imag(prod) == Approx(0).margin(10*REAL_EPS) );
536 qreal res = calcExpecPauliSum(vec, paulis.data(), coeffs.data(), numSumTerms, vecWork);
537 REQUIRE( res == Approx(real(prod)).margin(10*REAL_EPS) );
539 SECTION(
"density-matrix" ) {
542 matRef = pauliSum * matRef;
544 for (
size_t i=0; i<matRef.size(); i++)
545 tr += real(matRef[i][i]);
548 qreal res = calcExpecPauliSum(mat, paulis.data(), coeffs.data(), numSumTerms, matWork);
549 REQUIRE( res == Approx(tr).margin(1E2*REAL_EPS) );
552 SECTION(
"input validation" ) {
562 SECTION(
"pauli codes" ) {
566 vector<qreal> coeffs(numSumTerms);
567 vector<pauliOpType> codes(numSumTerms*NUM_QUBITS);
568 for (
int i=0; i<numSumTerms*NUM_QUBITS; i++)
572 codes[GENERATE_COPY( range(0,numSumTerms*NUM_QUBITS) )] = (pauliOpType) GENERATE( -1, 4 );
573 REQUIRE_THROWS_WITH( calcExpecPauliSum(vec, codes.data(), coeffs.data(), numSumTerms, vecWork), ContainsSubstring(
"invalid Pauli code") );
627 SECTION(
"correctness" ) {
630 GENERATE( range(0,10) );
632 SECTION(
"state-vector" ) {
636 SECTION(
"normalised" ) {
649 for (
size_t i=0; i<vecRef.size(); i++)
650 dotProd += conj(vecRef[i]) * pureRef[i];
651 qreal refFid = pow(abs(dotProd), 2);
683 SECTION(
"density-matrix" ) {
704 for (
size_t i=0; i<r1.size(); i++)
705 dotProd += conj(r1[i]) * pureRef[i];
706 qreal refFid = pow(abs(dotProd), 2);
708 REQUIRE(
calcFidelity(mat,pure) == Approx(refFid).margin(100 * REAL_EPS) );
720 QVector rhs = matRef * pureRef;
722 for (
size_t i=0; i<rhs.size(); i++)
723 dotProd += conj(pureRef[i]) * rhs[i];
725 REQUIRE( imag(dotProd) == Approx(0).margin(REAL_EPS) );
726 REQUIRE(
calcFidelity(mat,pure) == Approx(real(dotProd)).margin(100 * REAL_EPS) );
750 SECTION(
"input validation" ) {
752 SECTION(
"dimensions" ) {
756 REQUIRE_THROWS_WITH(
calcFidelity(vec2,vec), ContainsSubstring(
"differing numbers of qubits") );
761 REQUIRE_THROWS_WITH(
calcFidelity(mat2,vec), ContainsSubstring(
"differing numbers of qubits") );
764 SECTION(
"density-matrices" ) {
767 REQUIRE_THROWS_WITH(
calcFidelity(mat,mat), ContainsSubstring(
"Quregs cannot both be density matrices") );
781TEST_CASE(
"calcHilbertSchmidtDistance",
"[calculations]" ) {
786 SECTION(
"correctness" ) {
789 GENERATE( range(0,10) );
791 SECTION(
"density-matrix" ) {
805 for (
size_t i=0; i<m1.size(); i++)
806 for (
size_t j=0; j<m1.size(); j++)
807 tr += pow(abs(m1[i][j] - m2[i][j]), 2);
809 qreal res = calcHilbertSchmidtDistance(mat1, mat2);
810 REQUIRE( res == Approx(sqrt(tr)) );
813 SECTION(
"normalised" ) {
822 for (
size_t i=0; i<ref1.size(); i++)
823 for (
size_t j=0; j<ref1.size(); j++)
824 tr += pow(abs(ref1[i][j] - ref2[i][j]), 2);
826 qreal res = calcHilbertSchmidtDistance(mat1, mat2);
827 REQUIRE( res == Approx(sqrt(tr)) );
829 SECTION(
"unnormalised" ) {
839 for (
size_t i=0; i<ref1.size(); i++)
840 for (
size_t j=0; j<ref1.size(); j++)
841 tr += pow(abs(ref1[i][j] - ref2[i][j]), 2);
843 qreal res = calcHilbertSchmidtDistance(mat1, mat2);
844 REQUIRE( res == Approx(sqrt(tr)) );
848 SECTION(
"input validation") {
850 SECTION(
"dimensions" ) {
853 REQUIRE_THROWS_WITH( calcHilbertSchmidtDistance(mat1,mat3), ContainsSubstring(
"differing numbers of qubits") );
963 SECTION(
"correctness" ) {
966 int numQubits = GENERATE_COPY( range(1,NUM_QUBITS+1) );
967 int* qubits = GENERATE_COPY( sublists(range(0,NUM_QUBITS), numQubits) );
969 int numOutcomes = 1<<numQubits;
970 vector<qreal> probs(numOutcomes);
973 SECTION(
"state-vector" ) {
975 SECTION(
"normalised" ) {
981 for (
size_t i=0; i<ref.size(); i++) {
983 for (
int q=0; q<numQubits; q++) {
984 int bit = (i >> qubits[q]) & 1;
985 outcome += bit * (1 << q);
987 refProbs[outcome] += pow(abs(ref[i]), 2);
990 calcProbOfAllOutcomes(probs.data(), vec, qubits, numQubits);
991 REQUIRE(
areEqual(refProbs, probs.data()) );
993 SECTION(
"unnormalised" ) {
999 for (
size_t i=0; i<ref.size(); i++) {
1001 for (
int q=0; q<numQubits; q++) {
1002 int bit = (i >> qubits[q]) & 1;
1003 outcome += bit * (1 << q);
1005 refProbs[outcome] += pow(abs(ref[i]), 2);
1008 calcProbOfAllOutcomes(probs.data(), vec, qubits, numQubits);
1009 REQUIRE(
areEqual(refProbs, probs.data()) );
1012 SECTION(
"density-matrix" ) {
1014 SECTION(
"normalised" ) {
1020 for (
size_t i=0; i<ref.size(); i++) {
1022 for (
int q=0; q<numQubits; q++) {
1023 int bit = (i >> qubits[q]) & 1;
1024 outcome += bit * (1 << q);
1026 refProbs[outcome] += real(ref[i][i]);
1029 calcProbOfAllOutcomes(probs.data(), mat, qubits, numQubits);
1030 REQUIRE(
areEqual(refProbs, probs.data()) );
1032 SECTION(
"unnormalised" ) {
1038 for (
size_t i=0; i<ref.size(); i++) {
1040 for (
int q=0; q<numQubits; q++) {
1041 int bit = (i >> qubits[q]) & 1;
1042 outcome += bit * (1 << q);
1044 refProbs[outcome] += real(ref[i][i]);
1047 calcProbOfAllOutcomes(probs.data(), mat, qubits, numQubits);
1048 REQUIRE(
areEqual(refProbs, probs.data()) );
1052 SECTION(
"input validation" ) {
1055 int qubits[] = {0, 1, 2};
1058 SECTION(
"number of qubits" ) {
1061 numQubits = GENERATE( -1, 0 );
1062 REQUIRE_THROWS_WITH( calcProbOfAllOutcomes(probs, mat, qubits, numQubits), ContainsSubstring(
"specified number of target qubits") && ContainsSubstring(
"invalid.") );
1065 numQubits = NUM_QUBITS + 1;
1066 REQUIRE_THROWS_WITH( calcProbOfAllOutcomes(probs, mat, qubits, numQubits), ContainsSubstring(
"target qubits") && ContainsSubstring(
"exceeds the number of qubits in the Qureg") );
1068 SECTION(
"qubit indices" ) {
1070 qubits[GENERATE_COPY(range(0,numQubits))] = GENERATE( -1, NUM_QUBITS );
1071 REQUIRE_THROWS_WITH( calcProbOfAllOutcomes(probs, mat, qubits, numQubits), ContainsSubstring(
"Invalid target qubit") );
1073 SECTION(
"repetition of qubits" ) {
1075 qubits[GENERATE_COPY(1,2)] = qubits[0];
1076 REQUIRE_THROWS_WITH( calcProbOfAllOutcomes(probs, mat, qubits, numQubits), ContainsSubstring(
"qubits must be unique") );
1095 SECTION(
"correctness" ) {
1097 int target = GENERATE( range(0,NUM_QUBITS) );
1098 int outcome = GENERATE( 0, 1 );
1100 SECTION(
"state-vector" ) {
1102 SECTION(
"normalised" ) {
1109 for (
size_t ind=0; ind<ref.size(); ind++) {
1110 int bit = (ind >> target) & 1;
1112 prob += pow(abs(ref[ind]), 2);
1115 REQUIRE( calcProbOfOutcome(vec, target, outcome) == Approx(prob) );
1117 SECTION(
"unnormalised" ) {
1124 for (
size_t ind=0; ind<ref.size(); ind++) {
1125 int bit = (ind >> target) & 1;
1127 prob += pow(abs(ref[ind]), 2);
1130 REQUIRE( calcProbOfOutcome(vec, target, outcome) == Approx(prob) );
1133 SECTION(
"density-matrix" ) {
1143 for (
size_t ind=0; ind<ref.size(); ind++) {
1144 int bit = (ind >> target) & 1;
1146 prob += pow(abs(ref[ind]), 2);
1149 REQUIRE( calcProbOfOutcome(mat, target, outcome) == Approx(prob) );
1151 SECTION(
"mixed" ) {
1158 for (
size_t ind=0; ind<ref.size(); ind++) {
1159 int bit = (ind >> target) & 1;
1161 tr += ref[ind][ind];
1164 REQUIRE( imag(tr) == Approx(0).margin(REAL_EPS) );
1166 REQUIRE( calcProbOfOutcome(mat, target, outcome) == Approx(real(tr)) );
1168 SECTION(
"unnormalised" ) {
1175 for (
size_t ind=0; ind<ref.size(); ind++) {
1176 int bit = (ind >> target) & 1;
1178 tr += real(ref[ind][ind]);
1181 REQUIRE( calcProbOfOutcome(mat, target, outcome) == Approx(tr) );
1185 SECTION(
"input validation" ) {
1187 SECTION(
"qubit indices" ) {
1189 int target = GENERATE( -1, NUM_QUBITS );
1190 REQUIRE_THROWS_WITH( calcProbOfOutcome(vec, target, 0), ContainsSubstring(
"Invalid target qubit") );
1192 SECTION(
"outcome value" ) {
1194 int outcome = GENERATE( -1, 2 );
1195 REQUIRE_THROWS_WITH( calcProbOfOutcome(vec, 0, outcome), ContainsSubstring(
"measurement outcome") && ContainsSubstring(
"invalid") );
1392 SECTION(
"correctness" ) {
1394 SECTION(
"density-matrix" ) {
1399 int row = GENERATE( range(0,1<<NUM_QUBITS) );
1400 int col = GENERATE( range(0,1<<NUM_QUBITS) );
1402 qcomp amp = getDensityAmp(mat,row,col);
1403 REQUIRE( amp == ref[row][col] );
1406 SECTION(
"input validation" ) {
1408 SECTION(
"state index" ) {
1410 int ind = GENERATE( -1, 1<<NUM_QUBITS );
1411 REQUIRE_THROWS_WITH( getDensityAmp(mat,ind,0), ContainsSubstring(
"The row and column indices") && ContainsSubstring(
"invalid") );
1412 REQUIRE_THROWS_WITH( getDensityAmp(mat,0,ind), ContainsSubstring(
"The row and column indices") && ContainsSubstring(
"invalid") );
1415 SECTION(
"state-vector" ) {
1418 REQUIRE_THROWS_WITH( getDensityAmp(vec,0,0), ContainsSubstring(
"Expected a density matrix Qureg but received a statevector") );