|
| 1 | +// Compile and run with: |
| 2 | +// ``` |
| 3 | +// nvq++ --target quera quera_basic.cpp -o out.x |
| 4 | +// ./out.x |
| 5 | +// ``` |
| 6 | +// Assumes a valid set of credentials have been stored. |
| 7 | + |
| 8 | +#include "cudaq/algorithms/evolve.h" |
| 9 | +#include "cudaq/dynamics_integrators.h" |
| 10 | +#include "cudaq/schedule.h" |
| 11 | +#include <cmath> |
| 12 | +#include <map> |
| 13 | +#include <vector> |
| 14 | + |
| 15 | +// NOTE: QuEra Aquila system is available via Amazon Braket. |
| 16 | +// Credentials must be set before running this program. |
| 17 | +// Amazon Braket costs apply. |
| 18 | + |
| 19 | +// This example illustrates how to use QuEra's Aquila device on Braket with |
| 20 | +// CUDA-Q. It is a CUDA-Q implementation of the getting started materials for |
| 21 | +// Braket available here: |
| 22 | +// https://docs.aws.amazon.com/braket/latest/developerguide/braket-get-started-hello-ahs.html |
| 23 | + |
| 24 | +int main() { |
| 25 | + // Topology initialization |
| 26 | + const double a = 5.7e-6; |
| 27 | + std::vector<std::pair<double, double>> register_sites; |
| 28 | + |
| 29 | + auto make_coord = [a](double x, double y) { |
| 30 | + return std::make_pair(x * a, y * a); |
| 31 | + }; |
| 32 | + |
| 33 | + register_sites.push_back(make_coord(0.5, 0.5 + 1.0 / std::sqrt(2))); |
| 34 | + register_sites.push_back(make_coord(0.5 + 1.0 / std::sqrt(2), 0.5)); |
| 35 | + register_sites.push_back(make_coord(0.5 + 1.0 / std::sqrt(2), -0.5)); |
| 36 | + register_sites.push_back(make_coord(0.5, -0.5 - 1.0 / std::sqrt(2))); |
| 37 | + register_sites.push_back(make_coord(-0.5, -0.5 - 1.0 / std::sqrt(2))); |
| 38 | + register_sites.push_back(make_coord(-0.5 - 1.0 / std::sqrt(2), -0.5)); |
| 39 | + register_sites.push_back(make_coord(-0.5 - 1.0 / std::sqrt(2), 0.5)); |
| 40 | + register_sites.push_back(make_coord(-0.5, 0.5 + 1.0 / std::sqrt(2))); |
| 41 | + |
| 42 | + // Simulation Timing |
| 43 | + const double time_max = 4e-6; // seconds |
| 44 | + const double time_ramp = 1e-7; // seconds |
| 45 | + const double omega_max = 6.3e6; // rad/sec |
| 46 | + const double delta_start = -5 * omega_max; |
| 47 | + const double delta_end = 5 * omega_max; |
| 48 | + |
| 49 | + std::vector<std::complex<double>> steps = {0.0, time_ramp, |
| 50 | + time_max - time_ramp, time_max}; |
| 51 | + cudaq::schedule schedule(steps, {"t"}, {}); |
| 52 | + |
| 53 | + // Basic Rydberg Hamiltonian |
| 54 | + auto omega = cudaq::scalar_operator( |
| 55 | + [time_ramp, time_max, |
| 56 | + omega_max](const std::unordered_map<std::string, std::complex<double>> |
| 57 | + ¶meters) { |
| 58 | + double t = std::real(parameters.at("t")); |
| 59 | + return std::complex<double>( |
| 60 | + (t > time_ramp && t < time_max) ? omega_max : 0.0, 0.0); |
| 61 | + }); |
| 62 | + |
| 63 | + auto phi = cudaq::scalar_operator(0.0); |
| 64 | + |
| 65 | + auto delta = cudaq::scalar_operator( |
| 66 | + [time_ramp, time_max, delta_start, |
| 67 | + delta_end](const std::unordered_map<std::string, std::complex<double>> |
| 68 | + ¶meters) { |
| 69 | + double t = std::real(parameters.at("t")); |
| 70 | + return std::complex<double>( |
| 71 | + (t > time_ramp && t < time_max) ? delta_end : delta_start, 0.0); |
| 72 | + }); |
| 73 | + |
| 74 | + auto hamiltonian = |
| 75 | + cudaq::rydberg_hamiltonian(register_sites, omega, phi, delta); |
| 76 | + |
| 77 | + // Evolve the system |
| 78 | + auto result = cudaq::evolve_async(hamiltonian, schedule, 10).get(); |
| 79 | + result.get_sampling_result()->dump(); |
| 80 | + |
| 81 | + return 0; |
| 82 | +} |
0 commit comments