#include <iostream>
#include <iomanip>
#include <random>
#include <chrono>
#if !defined(EXRANDOM_HAVE_MPFR)
#define EXRANDOM_HAVE_MPFR 0
#endif
#if !defined(EXRANDOM_HAVE_MPREAL)
#define EXRANDOM_HAVE_MPREAL 0
#endif
#if EXRANDOM_HAVE_MPFR
#include <mpfr.h>
#if MPFR_VERSION >= MPFR_VERSION_NUM(3,1,0)
#define EXRANDOM_HAVE_GRANDOM 1
#else
#define EXRANDOM_HAVE_GRANDOM 0
#endif
#if MPFR_VERSION >= MPFR_VERSION_NUM(3,2,0)
#define EXRANDOM_HAVE_NRANDOM 1
#else
#define EXRANDOM_HAVE_NRANDOM 0
#endif
#else
#define EXRANDOM_HAVE_GRANDOM 0
#define EXRANDOM_HAVE_NRANDOM 0
#endif
#if EXRANDOM_HAVE_MPREAL
#include <mpreal.h>
#endif
#if EXRANDOM_HAVE_MPFR
int main() {
gmp_randstate_t r;
mpfr_t z, z2;
mpfr_rnd_t rnd = MPFR_RNDN;
double tx = 1e6;
int preclist[] = {4, 8, 16, 24, 32, 48, 53, 64, 128, 256,
(1<<10), (1<<12), (1<<14), (1<<16), (1<<18), (1<<20)};
gmp_randinit_mt(r);
gmp_randseed_ui(r, std::random_device()());
mpfr_init(z); mpfr_init(z2);
#if EXRANDOM_HAVE_NRANDOM
double timelistn[] = {
0.381,
0.365,
0.366,
0.367,
0.399,
0.406,
0.407,
0.405,
0.416,
0.417,
0.494,
1.11,
2.08,
6.1,
23.8,
91.4,
};
#endif
#if EXRANDOM_HAVE_GRANDOM
double timelistg[] = {
1.07,
1.11,
1.23,
1.23,
1.28,
1.37,
1.43,
1.96,
2.68,
3.61,
9.52,
48,
436,
4.09e+03,
3.65e+04,
3.52e+05,
};
#endif
#if EXRANDOM_HAVE_MPREAL
auto g = std::mt19937(std::random_device()());
double timelistm[] = {
1.25,
1.27,
1.28,
1.26,
1.28,
1.27,
1.29,
1.25,
1.37,
1.31,
1.38,
1.52,
1.54,
2.42,
13.7,
41.1,
};
#endif
std::cout << "timings (us) for nrandom vs grandom vs mpreal\n"
<< "prec nrandom grandom mpreal\n"
<< std::setprecision(3);
for (unsigned k = 0; k < sizeof(preclist)/sizeof(int); ++k) {
int prec = preclist[k];
double tn = -1, tg = -1, tm = -1;
#if EXRANDOM_HAVE_NRANDOM
{
mpfr_set_prec(z, prec);
size_t m = (size_t)(tx / timelistn[k] + 1);
auto t0 = std::chrono::high_resolution_clock::now();
for (size_t i = 0; i < m; ++i)
mpfr_nrandom(z, r, rnd);
auto t1 = std::chrono::high_resolution_clock::now();
double dt = double(std::chrono::duration_cast<std::chrono::nanoseconds>
(t1 - t0).count());
tn = dt/(m * 1000);
}
#endif
#if EXRANDOM_HAVE_GRANDOM
{
mpfr_set_prec(z, prec); mpfr_set_prec(z2, prec);
size_t m = (size_t)(tx / timelistg[k] + 1);
auto t0 = std::chrono::high_resolution_clock::now();
for (size_t i = 0; i < (m + 1)/2; ++i)
mpfr_grandom(z, z2, r, rnd);
auto t1 = std::chrono::high_resolution_clock::now();
double dt = double(std::chrono::duration_cast<std::chrono::nanoseconds>
(t1 - t0).count());
tg = dt/((m + 1)/2 * 2000);
}
#endif
#if EXRANDOM_HAVE_MPREAL
{
mpfr::mpreal::set_default_prec(prec);
size_t m = (size_t)(tx / timelistm[k] + 1);
auto t0 = std::chrono::high_resolution_clock::now();
for (size_t i = 0; i < m; ++i) N(g);
auto t1 = std::chrono::high_resolution_clock::now();
double dt = double(std::chrono::duration_cast<std::chrono::nanoseconds>
(t1 - t0).count());
tm = dt/(m * 1000);
}
#endif
std::cout << prec << " " << tn << " " << tg << " " << tm << "\n";
}
mpfr_clear(z); mpfr_clear(z2);
mpfr_free_cache();
gmp_randclear(r);
}
#else
int main() {
std::cerr << "MPFR version 3.1 or later required\n\n";
}
#endif