// random -*- C++ -*- // Copyright (C) 2012-2019 Free Software Foundation, Inc. // // This file is part of the GNU ISO C++ Library. This library is free // software; you can redistribute it and/or modify it under the // terms of the GNU General Public License as published by the // Free Software Foundation; either version 3, or (at your option) // any later version. // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // Under Section 7 of GPL version 3, you are granted additional // permissions described in the GCC Runtime Library Exception, version // 3.1, as published by the Free Software Foundation. // You should have received a copy of the GNU General Public License and // a copy of the GCC Runtime Library Exception along with this program; // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see // . #define _GLIBCXX_USE_CXX11_ABI 1 #define _CRT_RAND_S // define this before including to get rand_s #include #ifdef _GLIBCXX_USE_C99_STDINT_TR1 #if defined __i386__ || defined __x86_64__ # include #endif #include #include #ifdef _GLIBCXX_HAVE_UNISTD_H # include #endif #ifdef _GLIBCXX_HAVE_SYS_IOCTL_H # include #endif #ifdef _GLIBCXX_HAVE_LINUX_TYPES_H # include #endif #ifdef _GLIBCXX_HAVE_LINUX_RANDOM_H # include #endif #ifdef _GLIBCXX_USE_CRT_RAND_S # include #endif namespace std _GLIBCXX_VISIBILITY(default) { namespace { static unsigned long _M_strtoul(const std::string& __str) { unsigned long __ret = 5489UL; if (__str != "mt19937") { const char* __nptr = __str.c_str(); char* __endptr; __ret = std::strtoul(__nptr, &__endptr, 0); if (*__nptr == '\0' || *__endptr != '\0') std::__throw_runtime_error(__N("random_device::_M_strtoul" "(const std::string&)")); } return __ret; } #if (defined __i386__ || defined __x86_64__) && defined _GLIBCXX_X86_RDRAND unsigned int __attribute__ ((target("rdrnd"))) __x86_rdrand(void) { unsigned int retries = 100; unsigned int val; while (__builtin_ia32_rdrand32_step(&val) == 0) if (--retries == 0) std::__throw_runtime_error(__N("random_device::__x86_rdrand(void)")); return val; } #endif #ifdef _GLIBCXX_USE_CRT_RAND_S # pragma GCC poison _M_mt unsigned int __winxp_rand_s() { unsigned int val; if (::rand_s(&val) != 0) std::__throw_runtime_error(__N("random_device: rand_s failed")); return val; } #endif } void random_device::_M_init(const std::string& token) { const char *fname = token.c_str(); if (token == "default") { #if (defined __i386__ || defined __x86_64__) && defined _GLIBCXX_X86_RDRAND unsigned int eax, ebx, ecx, edx; // Check availability of cpuid and, for now at least, also the // CPU signature for Intel's if (__get_cpuid_max(0, &ebx) > 0 && ebx == signature_INTEL_ebx) { __cpuid(1, eax, ebx, ecx, edx); if (ecx & bit_RDRND) { _M_file = nullptr; return; } } #endif fname = "/dev/urandom"; } else if (token != "/dev/urandom" && token != "/dev/random") fail: std::__throw_runtime_error(__N("random_device::" "random_device(const std::string&)")); _M_file = static_cast(std::fopen(fname, "rb")); if (!_M_file) goto fail; } void random_device::_M_init_pretr1(const std::string& token [[gnu::unused]]) { #ifndef _GLIBCXX_USE_CRT_RAND_S _M_mt.seed(_M_strtoul(token)); #endif } void random_device::_M_fini() { if (_M_file) std::fclose(static_cast(_M_file)); } random_device::result_type random_device::_M_getval() { #if (defined __i386__ || defined __x86_64__) && defined _GLIBCXX_X86_RDRAND if (!_M_file) return __x86_rdrand(); #endif result_type __ret; void* p = &__ret; size_t n = sizeof(result_type); #ifdef _GLIBCXX_HAVE_UNISTD_H do { const int e = read(fileno(static_cast(_M_file)), p, n); if (e > 0) { n -= e; p = static_cast(p) + e; } else if (e != -1 || errno != EINTR) __throw_runtime_error(__N("random_device could not be read")); } while (n > 0); #else const size_t e = std::fread(p, n, 1, static_cast(_M_file)); if (e != 1) __throw_runtime_error(__N("random_device could not be read")); #endif return __ret; } random_device::result_type random_device::_M_getval_pretr1() { #ifdef _GLIBCXX_USE_CRT_RAND_S return __winxp_rand_s(); #else return _M_mt(); #endif } double random_device::_M_getentropy() const noexcept { #if defined _GLIBCXX_HAVE_SYS_IOCTL_H && defined RNDGETENTCNT if (!_M_file) return 0.0; const int fd = fileno(static_cast(_M_file)); if (fd < 0) return 0.0; int ent; if (ioctl(fd, RNDGETENTCNT, &ent) < 0) return 0.0; if (ent < 0) return 0.0; const int max = sizeof(result_type) * __CHAR_BIT__; if (ent > max) ent = max; return static_cast(ent); #else return 0.0; #endif } template class mersenne_twister_engine< uint_fast32_t, 32, 624, 397, 31, 0x9908b0dfUL, 11, 0xffffffffUL, 7, 0x9d2c5680UL, 15, 0xefc60000UL, 18, 1812433253UL>; } #endif