12#include "../../composer/composer/Estimators/ExecutionEstimator.h"
42 Estimators::ExecutionEstimator<>::InitializeRegressors();
57 Simulators::SimulatorsFactory::InitGpuLibraryWithMute();
61 Estimators::ExecutionEstimator<>::InitializeRegressors();
92 int simType,
int simExecType) {
116 const char *jsonConfig) {
141 std::shared_ptr<Circuits::Circuit<>> circuit;
143 if (circuitStr[0] ==
'{' || circuitStr[0] ==
'[') {
150 std::string qasmInput(circuitStr);
152 if (parser.
Failed())
return nullptr;
162 if (configJson.is_object()) {
163 const auto configObject = configJson.as_object();
168 if (configObject.contains(
"shots") &&
169 configObject.at(
"shots").is_number()) {
170 auto number = configObject.at(
"shots");
171 nrShots = number.is_int64() ? (size_t)number.as_int64()
172 : (size_t)number.as_uint64();
176 bool configured =
false;
179 "matrix_product_state_max_bond_dimension", configJson);
180 if (!maxBondDim.empty()) {
182 if (network->GetSimulator()) network->GetSimulator()->Clear();
183 network->Configure(
"matrix_product_state_max_bond_dimension",
187 const std::string singularValueThreshold =
189 "matrix_product_state_truncation_threshold", configJson);
190 if (!singularValueThreshold.empty()) {
192 if (network->GetSimulator()) network->GetSimulator()->Clear();
193 network->Configure(
"matrix_product_state_truncation_threshold",
194 singularValueThreshold.c_str());
198 "mps_sample_measure_algorithm", configJson);
199 if (!mpsSample.empty()) {
201 if (network->GetSimulator()) network->GetSimulator()->Clear();
202 network->Configure(
"mps_sample_measure_algorithm", mpsSample.c_str());
205 if (configured || !network->GetSimulator()) network->CreateSimulator();
208 auto start = std::chrono::high_resolution_clock::now();
209 auto results = network->RepeatedExecuteOnHost(circuit, 0, nrShots);
210 auto end = std::chrono::high_resolution_clock::now();
212 std::chrono::duration<double> duration = end - start;
213 double time_taken = duration.count();
214 std::string timeStr = std::to_string(time_taken);
220 boost::json::object jsonResult;
221 jsonResult.reserve(results.size());
223 for (
auto &result : results) {
224 boost::json::string bits;
225 bits.reserve(result.first.size());
226 for (
const auto bit : result.first) bits.append(bit ?
"1" :
"0");
228 jsonResult.emplace(std::move(bits), std::move(result.second));
231 boost::json::object response;
234 response.emplace(
"counts", std::move(jsonResult));
235 response.emplace(
"time_taken", timeStr);
237 auto simulatorType = network->GetLastSimulatorType();
239 switch (simulatorType) {
242 response.emplace(
"simulator",
"aer");
246 response.emplace(
"simulator",
"qcsim");
250 response.emplace(
"simulator",
"composite_aer");
254 response.emplace(
"simulator",
"composite_qcsim");
257 response.emplace(
"simulator",
"quest");
261 response.emplace(
"simulator",
"gpu_simulator");
265 response.emplace(
"simulator",
"unknown");
269 auto simulationType = network->GetLastSimulationType();
270 switch (simulationType) {
272 response.emplace(
"method",
"statevector");
275 response.emplace(
"method",
"matrix_product_state");
278 response.emplace(
"method",
"stabilizer");
281 response.emplace(
"method",
"tensor_network");
284 response.emplace(
"method",
"pauli_propagation");
287 response.emplace(
"method",
"extended_stabilizer");
290 response.emplace(
"method",
"path_integral");
293 response.emplace(
"method",
"unknown");
297 const std::string responseStr = boost::json::serialize(response);
298 const size_t responseSize = responseStr.length();
299 char *result =
new char[responseSize + 1];
301 const char *responseData = responseStr.c_str();
302 std::copy(responseData, responseData + responseSize, result);
304 result[responseSize] = 0;
313 const char *observableStr,
const char *jsonConfig) {
314 if (simpleSim == 0 || !circuitStr || !observableStr || !jsonConfig ||
320 std::shared_ptr<Circuits::Circuit<>> circuit;
322 if (circuitStr[0] ==
'{' || circuitStr[0] ==
'[') {
327 std::string qasmInput(circuitStr);
329 if (parser.
Failed())
return nullptr;
334 bool configured =
false;
337 "matrix_product_state_max_bond_dimension", configJson);
338 if (!maxBondDim.empty()) {
340 if (network->GetSimulator()) network->GetSimulator()->Clear();
341 network->Configure(
"matrix_product_state_max_bond_dimension",
345 const std::string singularValueThreshold =
347 "matrix_product_state_truncation_threshold", configJson);
348 if (!singularValueThreshold.empty()) {
350 if (network->GetSimulator()) network->GetSimulator()->Clear();
351 network->Configure(
"matrix_product_state_truncation_threshold",
352 singularValueThreshold.c_str());
356 "mps_sample_measure_algorithm", configJson);
357 if (!mpsSample.empty()) {
359 if (network->GetSimulator()) network->GetSimulator()->Clear();
360 network->Configure(
"mps_sample_measure_algorithm", mpsSample.c_str());
363 if (configured || !network->GetSimulator()) network->CreateSimulator();
366 std::vector<std::string> paulis;
367 std::string obsStr(observableStr);
368 std::stringstream ss(obsStr);
370 while (std::getline(ss, item,
';')) {
371 if (!item.empty()) paulis.push_back(item);
374 auto start = std::chrono::high_resolution_clock::now();
375 auto expectations = network->ExecuteOnHostExpectations(circuit, 0, paulis);
376 auto end = std::chrono::high_resolution_clock::now();
378 std::chrono::duration<double> duration = end - start;
379 double time_taken = duration.count();
380 std::string timeStr = std::to_string(time_taken);
382 boost::json::object response;
385 boost::json::array jsonExpectations;
386 jsonExpectations.reserve(expectations.size());
387 for (
double val : expectations) {
388 jsonExpectations.push_back(val);
391 response.emplace(
"expectation_values", std::move(jsonExpectations));
392 response.emplace(
"time_taken", timeStr);
394 auto simulatorType = network->GetLastSimulatorType();
396 switch (simulatorType) {
399 response.emplace(
"simulator",
"aer");
403 response.emplace(
"simulator",
"qcsim");
406 response.emplace(
"simulator",
"quest");
410 response.emplace(
"simulator",
"composite_aer");
414 response.emplace(
"simulator",
"composite_qcsim");
418 response.emplace(
"simulator",
"gpu_simulator");
422 response.emplace(
"simulator",
"unknown");
426 auto simulationType = network->GetLastSimulationType();
427 switch (simulationType) {
429 response.emplace(
"method",
"statevector");
432 response.emplace(
"method",
"matrix_product_state");
435 response.emplace(
"method",
"stabilizer");
438 response.emplace(
"method",
"tensor_network");
441 response.emplace(
"method",
"pauli_propagation");
444 response.emplace(
"method",
"extended_stabilizer");
447 response.emplace(
"method",
"path_integral");
450 response.emplace(
"method",
"unknown");
454 const std::string responseStr = boost::json::serialize(response);
455 const size_t responseSize = responseStr.length();
456 char *result =
new char[responseSize + 1];
458 const char *responseData = responseStr.c_str();
459 std::copy(responseData, responseData + responseSize, result);
461 result[responseSize] = 0;
470 if (result)
delete[] result;
625 int ApplyP(
void *sim,
int qubit,
double theta) {
628 simulator->
ApplyP(qubit, theta);
636 int ApplyRx(
void *sim,
int qubit,
double theta) {
639 simulator->
ApplyRx(qubit, theta);
647 int ApplyRy(
void *sim,
int qubit,
double theta) {
650 simulator->
ApplyRy(qubit, theta);
658 int ApplyRz(
void *sim,
int qubit,
double theta) {
661 simulator->
ApplyRz(qubit, theta);
669 int ApplyU(
void *sim,
int qubit,
double theta,
double phi,
double lambda,
673 simulator->
ApplyU(qubit, theta, phi, lambda, gamma);
681 int ApplyCX(
void *sim,
int controlQubit,
int targetQubit) {
684 simulator->
ApplyCX(controlQubit, targetQubit);
692 int ApplyCY(
void *sim,
int controlQubit,
int targetQubit) {
695 simulator->
ApplyCY(controlQubit, targetQubit);
703 int ApplyCZ(
void *sim,
int controlQubit,
int targetQubit) {
706 simulator->
ApplyCZ(controlQubit, targetQubit);
714 int ApplyCH(
void *sim,
int controlQubit,
int targetQubit) {
717 simulator->
ApplyCH(controlQubit, targetQubit);
725 int ApplyCSX(
void *sim,
int controlQubit,
int targetQubit) {
728 simulator->
ApplyCSx(controlQubit, targetQubit);
736 int ApplyCSXDG(
void *sim,
int controlQubit,
int targetQubit) {
747 int ApplyCP(
void *sim,
int controlQubit,
int targetQubit,
double theta) {
750 simulator->
ApplyCP(controlQubit, targetQubit, theta);
758 int ApplyCRx(
void *sim,
int controlQubit,
int targetQubit,
double theta) {
761 simulator->
ApplyCRx(controlQubit, targetQubit, theta);
769 int ApplyCRy(
void *sim,
int controlQubit,
int targetQubit,
double theta) {
772 simulator->
ApplyCRy(controlQubit, targetQubit, theta);
780 int ApplyCRz(
void *sim,
int controlQubit,
int targetQubit,
double theta) {
783 simulator->
ApplyCRz(controlQubit, targetQubit, theta);
791 int ApplyCCX(
void *sim,
int controlQubit1,
int controlQubit2,
795 simulator->
ApplyCCX(controlQubit1, controlQubit2, targetQubit);
814 int ApplyCSwap(
void *sim,
int controlQubit,
int qubit1,
int qubit2) {
817 simulator->
ApplyCSwap(controlQubit, qubit1, qubit2);
825 int ApplyCU(
void *sim,
int controlQubit,
int targetQubit,
double theta,
826 double phi,
double lambda,
double gamma) {
829 simulator->
ApplyCU(controlQubit, targetQubit, theta, phi, lambda, gamma);
858 if (!sim || !key || !value)
return 0;
868 if (!sim || !key)
return nullptr;
871 if (value.empty())
return nullptr;
874 const size_t valueSize = value.length();
875 char *result =
new char[valueSize + 1];
876 std::copy(value.c_str(), value.c_str() + valueSize, result);
877 result[valueSize] = 0;
885 if (!sim || nrQubits == 0)
return 0;
889 return static_cast<unsigned long int>(res);
899 return static_cast<unsigned long int>(res);
915 unsigned long long int Measure(
void *sim,
const unsigned long int *qubits,
916 unsigned long int nrQubits) {
917 if (!sim || !qubits || nrQubits == 0)
return 0;
920 const size_t res = simulator->Measure(qubitVector);
921 return static_cast<unsigned long long int>(res);
928 unsigned long int nrQubits) {
929 if (!sim || !qubits || nrQubits == 0)
return 0;
932 simulator->ApplyReset(qubitVector);
940 if (!sim)
return 0.0;
942 const double res = simulator->
Probability(outcome);
950 if (vec)
delete[] vec;
957 if (vec)
delete[] vec;
963 double *
Amplitude(
void *sim,
unsigned long long int outcome) {
964 if (!sim)
return nullptr;
966 const std::complex<double> amp = simulator->
Amplitude(outcome);
968 double *result =
new double[2];
969 result[0] = amp.real();
970 result[1] = amp.imag();
978 if (!sim)
return nullptr;
982 double *result =
new double[probabilities.size()];
983 std::copy(probabilities.begin(), probabilities.end(), result);
991 unsigned long int nrQubits) {
992 if (!sim || !qubits || nrQubits == 0)
return nullptr;
995 const auto probabilities = simulator->Probabilities(qubitVector);
997 double *result =
new double[probabilities.size()];
998 std::copy(probabilities.begin(), probabilities.end(), result);
1003__declspec(dllexport)
1006 const unsigned long long int *qubits,
1007 unsigned long int nrQubits,
1008 unsigned long int shots) {
1009 if (!sim || !qubits || nrQubits == 0 || shots == 0)
return nullptr;
1013 const auto counts = simulator->SampleCounts(qubitVector, shots);
1015 unsigned long long int *result =
1016 new unsigned long long int[counts.size() * 2];
1018 for (
const auto &count : counts) {
1019 result[index] = count.first;
1021 result[index] = count.second;
1028__declspec(dllexport)
1031 if (!sim)
return -1;
1033 return static_cast<int>(simulator->GetType());
1037__declspec(dllexport)
1040 if (!sim)
return -1;
1042 return static_cast<int>(simulator->GetSimulationType());
1046__declspec(dllexport)
1056__declspec(dllexport)
1066__declspec(dllexport)
1076__declspec(dllexport)
1086__declspec(dllexport)
1096__declspec(dllexport)
1106__declspec(dllexport)
1115__declspec(dllexport)
1120 return simulator->
IsQcsim() ? 1 : 0;
1124__declspec(dllexport)
1129 return static_cast<unsigned long long int>(simulator->MeasureNoCollapse());
int ApplyK(void *sim, int qubit)
double Probability(void *sim, unsigned long long int outcome)
char * GetConfiguration(void *sim, const char *key)
int InitializeSimulator(void *sim)
int RestoreState(void *sim)
void FreeDoubleVector(double *vec)
int ApplyRx(void *sim, int qubit, double theta)
int ApplyReset(void *sim, const unsigned long int *qubits, unsigned long int nrQubits)
int ApplyX(void *sim, int qubit)
int ApplyU(void *sim, int qubit, double theta, double phi, double lambda, double gamma)
int AddOptimizationSimulator(unsigned long int simHandle, int simType, int simExecType)
unsigned long int CreateSimulator(int simType, int simExecType)
int ApplyCRy(void *sim, int controlQubit, int targetQubit, double theta)
int ApplyTDG(void *sim, int qubit)
int ApplyCSXDG(void *sim, int controlQubit, int targetQubit)
static std::atomic_bool isInitialized
void FreeResult(char *result)
int ApplyS(void *sim, int qubit)
int ApplyCX(void *sim, int controlQubit, int targetQubit)
unsigned long int AllocateQubits(void *sim, unsigned long int nrQubits)
char * SimpleExecute(unsigned long int simpleSim, const char *circuitStr, const char *jsonConfig)
int ApplyCRz(void *sim, int controlQubit, int targetQubit, double theta)
unsigned long int GetNumberOfQubits(void *sim)
double * AllProbabilities(void *sim)
void * GetMaestroObjectWithMute()
unsigned long long int MeasureNoCollapse(void *sim)
int ApplyCP(void *sim, int controlQubit, int targetQubit, double theta)
int ApplySXDG(void *sim, int qubit)
void * GetMaestroObject()
int GetMultithreading(void *sim)
int ApplySDG(void *sim, int qubit)
unsigned long long int Measure(void *sim, const unsigned long int *qubits, unsigned long int nrQubits)
int ApplyCSwap(void *sim, int controlQubit, int qubit1, int qubit2)
void DestroySimpleSimulator(unsigned long int simHandle)
int ApplyCCX(void *sim, int controlQubit1, int controlQubit2, int targetQubit)
int ApplyY(void *sim, int qubit)
double * Amplitude(void *sim, unsigned long long int outcome)
int ApplyZ(void *sim, int qubit)
int ApplyH(void *sim, int qubit)
int ApplyCY(void *sim, int controlQubit, int targetQubit)
char * SimpleEstimate(unsigned long int simpleSim, const char *circuitStr, const char *observableStr, const char *jsonConfig)
double * Probabilities(void *sim, const unsigned long long int *qubits, unsigned long int nrQubits)
int SetMultithreading(void *sim, int multithreading)
int ApplyCU(void *sim, int controlQubit, int targetQubit, double theta, double phi, double lambda, double gamma)
void DestroySimulator(unsigned long int simHandle)
int ApplySwap(void *sim, int qubit1, int qubit2)
void * GetSimulator(unsigned long int simHandle)
static std::unique_ptr< Maestro > maestroInstance
int ApplyRy(void *sim, int qubit, double theta)
int ApplyP(void *sim, int qubit, double theta)
int SaveStateToInternalDestructive(void *sim)
int ApplyCH(void *sim, int controlQubit, int targetQubit)
int FlushSimulator(void *sim)
unsigned long int CreateSimpleSimulator(int nrQubits)
int GetSimulationType(void *sim)
int ResetSimulator(void *sim)
unsigned long long int * SampleCounts(void *sim, const unsigned long long int *qubits, unsigned long int nrQubits, unsigned long int shots)
int ApplySX(void *sim, int qubit)
int ApplyCZ(void *sim, int controlQubit, int targetQubit)
int ApplyRz(void *sim, int qubit, double theta)
int GetSimulatorType(void *sim)
int RestoreInternalDestructiveSavedState(void *sim)
int ApplyT(void *sim, int qubit)
int ApplyCRx(void *sim, int controlQubit, int targetQubit, double theta)
int ConfigureSimulator(void *sim, const char *key, const char *value)
int ApplyCSX(void *sim, int controlQubit, int targetQubit)
int ClearSimulator(void *sim)
int RemoveAllOptimizationSimulatorsAndAdd(unsigned long int simHandle, int simType, int simExecType)
void FreeULLIVector(unsigned long long int *vec)
static boost::json::value ParseString(const char *str)
Parses a string containing json.
std::shared_ptr< Circuits::Circuit< Time > > ParseCircuit(const char *str) const
static std::string GetConfigString(const std::string &config, const boost::json::value &jsonConfig)
Interface class for a quantum computing simulator.
virtual void ApplyCRz(Types::qubit_t ctrl_qubit, Types::qubit_t tgt_qubit, double theta)=0
Applies a CRz gate to the qubits.
virtual void ApplySDG(Types::qubit_t qubit)=0
Applies a S dagger gate to the qubit.
virtual void ApplyCCX(Types::qubit_t qubit0, Types::qubit_t qubit1, Types::qubit_t qubit2)=0
Applies a controlled controlled not gate to the qubits.
virtual void ApplyX(Types::qubit_t qubit)=0
Applies a not gate to the qubit.
virtual void ApplyCSxDAG(Types::qubit_t ctrl_qubit, Types::qubit_t tgt_qubit)=0
Applies a CSx dagger gate to the qubits.
virtual void ApplyU(Types::qubit_t qubit, double theta, double phi, double lambda, double gamma)=0
Applies a U gate to the qubit.
virtual void ApplyP(Types::qubit_t qubit, double lambda)=0
Applies a phase shift gate to the qubit.
virtual void ApplySx(Types::qubit_t qubit)=0
Applies a Sx gate to the qubit.
virtual void ApplyTDG(Types::qubit_t qubit)=0
Applies a T dagger gate to the qubit.
virtual void ApplyCU(Types::qubit_t ctrl_qubit, Types::qubit_t tgt_qubit, double theta, double phi, double lambda, double gamma)=0
Applies a controlled U gate to the qubits.
virtual void ApplyCRy(Types::qubit_t ctrl_qubit, Types::qubit_t tgt_qubit, double theta)=0
Applies a CRy gate to the qubits.
virtual void ApplyCX(Types::qubit_t ctrl_qubit, Types::qubit_t tgt_qubit)=0
Applies a CX gate to the qubits.
virtual void ApplyRy(Types::qubit_t qubit, double theta)=0
Applies a Ry gate to the qubit.
virtual void ApplyCH(Types::qubit_t ctrl_qubit, Types::qubit_t tgt_qubit)=0
Applies a CH gate to the qubits.
virtual void ApplySwap(Types::qubit_t qubit0, Types::qubit_t qubit1)=0
Applies a swap gate to the qubits.
virtual void ApplyCSx(Types::qubit_t ctrl_qubit, Types::qubit_t tgt_qubit)=0
Applies a CSx gate to the qubits.
virtual void ApplyK(Types::qubit_t qubit)=0
Applies a K gate to the qubit.
virtual void ApplyY(Types::qubit_t qubit)=0
Applies a Y gate to the qubit.
virtual void ApplyT(Types::qubit_t qubit)=0
Applies a T gate to the qubit.
virtual void ApplyS(Types::qubit_t qubit)=0
Applies a S gate to the qubit.
virtual void ApplyCSwap(Types::qubit_t ctrl_qubit, Types::qubit_t qubit0, Types::qubit_t qubit1)=0
Applies a controlled swap gate to the qubits.
virtual void ApplyZ(Types::qubit_t qubit)=0
Applies a Z gate to the qubit.
virtual void ApplyRz(Types::qubit_t qubit, double theta)=0
Applies a Rz gate to the qubit.
virtual void ApplyCRx(Types::qubit_t ctrl_qubit, Types::qubit_t tgt_qubit, double theta)=0
Applies a CRx gate to the qubits.
virtual void ApplyCY(Types::qubit_t ctrl_qubit, Types::qubit_t tgt_qubit)=0
Applies a CY gate to the qubits.
virtual void ApplyCZ(Types::qubit_t ctrl_qubit, Types::qubit_t tgt_qubit)=0
Applies a CZ gate to the qubits.
virtual void ApplyH(Types::qubit_t qubit)=0
Applies a Hadamard gate to the qubit.
virtual void ApplyRx(Types::qubit_t qubit, double theta)=0
Applies a Rx gate to the qubit.
virtual void ApplySxDAG(Types::qubit_t qubit)=0
Applies a Sx dagger gate to the qubit.
virtual void ApplyCP(Types::qubit_t ctrl_qubit, Types::qubit_t tgt_qubit, double lambda)=0
Applies a CP gate to the qubits.
virtual bool IsQcsim() const =0
Returns if the simulator is a qcsim simulator.
virtual void SaveStateToInternalDestructive()=0
Saves the state to internal storage.
virtual void RestoreState()=0
Restores the state from the internally saved state.
virtual void SaveState()=0
Saves the state to internal storage.
virtual double Probability(Types::qubit_t outcome)=0
Returns the probability of the specified outcome.
virtual void Initialize()=0
Initializes the state.
virtual size_t AllocateQubits(size_t num_qubits)=0
Allocates qubits.
virtual std::vector< double > AllProbabilities()=0
Returns the probabilities of all possible outcomes.
virtual void RestoreInternalDestructiveSavedState()=0
Restores the state from the internally saved state.
virtual void SetMultithreading(bool multithreading=true)=0
Enable/disable multithreading.
virtual std::complex< double > Amplitude(Types::qubit_t outcome)=0
Returns the amplitude of the specified state.
virtual size_t GetNumberOfQubits() const =0
Returns the number of qubits.
virtual void Flush()=0
Flushes the applied operations.
virtual void Configure(const char *key, const char *value)=0
Configures the state.
virtual void Clear()=0
Clears the state.
virtual bool GetMultithreading() const =0
Get the multithreading flag.
virtual void Reset()=0
Just resets the state to 0.
virtual std::string GetConfiguration(const char *key) const =0
Returns configuration value.
static bool InitQuestLibrary()
static bool InitGpuLibrary()
std::shared_ptr< Circuits::Circuit< Time > > ParseAndTranslate(const std::string &qasmInputStr)
SimulationType
The type of simulation.
@ kExtendedStabilizer
Extended stabilizer simulation type.
@ kStatevector
statevector simulation type
@ kMatrixProductState
matrix product state simulation type
@ kStabilizer
Clifford gates simulation type.
@ kPauliPropagator
Pauli propagator simulation type.
@ kTensorNetwork
Tensor network simulation type.
@ kPathIntegral
Path integral simulation type.
SimulatorType
The type of simulator.
@ kCompositeQCSim
composite qcsim simulator type
@ kQCSim
qcsim simulator type
@ kQiskitAer
qiskit aer simulator type
@ kQuestSim
quest simulator type
@ kCompositeQiskitAer
composite qiskit aer simulator type
@ kGpuSim
gpu simulator type
std::vector< qubit_t > qubits_vector
The type of a vector of qubits.