// Copyright (c) 2012-2013 The Cryptonote developers // Distributed under the MIT/X11 software license, see the accompanying // file COPYING or http://www.opensource.org/licenses/mit-license.php. #include #include #include #include "hash-ops.h" //#include "initializer.h" #include "random.h" static_assert(RANDOM_STATE_SIZE >= HASH_DATA_AREA, "Invalid RANDOM_STATE_SIZE"); #if defined(_WIN32) #include #include void generate_system_random_bytes(size_t n, void *result) { HCRYPTPROV prov; #define must_succeed(x) do if (!(x)) assert(0); while (0) if(!CryptAcquireContext(&prov, NULL, NULL, PROV_RSA_FULL, CRYPT_VERIFYCONTEXT | CRYPT_SILENT)) { int err = GetLastError(); assert(0); } must_succeed(CryptGenRandom(prov, (DWORD)n, result)); must_succeed(CryptReleaseContext(prov, 0)); #undef must_succeed } #else #include #include #include #include #include #include #include void generate_system_random_bytes(size_t n, void *result) { int fd; if ((fd = open("/dev/urandom", O_RDONLY | O_NOCTTY | O_CLOEXEC)) < 0) { err(EXIT_FAILURE, "open /dev/urandom"); } for (;;) { ssize_t res = read(fd, result, n); if ((size_t) res == n) { break; } if (res < 0) { if (errno != EINTR) { err(EXIT_FAILURE, "read /dev/urandom"); } } else if (res == 0) { errx(EXIT_FAILURE, "read /dev/urandom: end of file"); } else { result = padd(result, (size_t) res); n -= (size_t) res; } } if (close(fd) < 0) { err(EXIT_FAILURE, "close /dev/urandom"); } } #endif static union hash_state state; #if !defined(NDEBUG) static volatile int curstate; /* To catch thread safety problems. */ #endif /* FINALIZER(deinit_random) { #if !defined(NDEBUG) assert(curstate == 1); curstate = 0; #endif memset(&state, 0, sizeof(union hash_state)); } */ //INITIALIZER(init_random) { void init_random(void) { generate_system_random_bytes(32, &state); //REGISTER_FINA\LIZER(deinit_random); #if !defined(NDEBUG) assert(curstate == 0); curstate = 1; #endif } void grant_random_initialize(void) { static bool initalized = false; if(!initalized) { init_random(); initalized = true; } } void random_prng_initialize_with_seed(uint64_t seed) { grant_random_initialize(); #if !defined(NDEBUG) assert(curstate == 1); curstate = 3; #endif memset(&state, 0, sizeof state); memcpy(&state, &seed, sizeof seed); for(size_t i = 0, count = seed & 31; i < count; ++i) hash_permutation(&state); #if !defined(NDEBUG) assert(curstate == 3); curstate = 1; #endif } void random_prng_get_state(void *state_buffer, const size_t buffer_size) { grant_random_initialize(); #if !defined(NDEBUG) assert(curstate == 1); curstate = 4; #endif assert(sizeof state == buffer_size); memcpy(state_buffer, &state, buffer_size); #if !defined(NDEBUG) assert(curstate == 4); curstate = 1; #endif } void random_prng_set_state(const void *state_buffer, const size_t buffer_size) { grant_random_initialize(); #if !defined(NDEBUG) assert(curstate == 1); curstate = 5; #endif assert(sizeof state == buffer_size); memcpy(&state, state_buffer, buffer_size); #if !defined(NDEBUG) assert(curstate == 5); curstate = 1; #endif } void generate_random_bytes(size_t n, void *result) { grant_random_initialize(); #if !defined(NDEBUG) assert(curstate == 1); curstate = 2; #endif if (n == 0) { #if !defined(NDEBUG) assert(curstate == 2); curstate = 1; #endif return; } for (;;) { hash_permutation(&state); if (n <= HASH_DATA_AREA) { memcpy(result, &state, n); #if !defined(NDEBUG) assert(curstate == 2); curstate = 1; #endif return; } else { memcpy(result, &state, HASH_DATA_AREA); result = padd(result, HASH_DATA_AREA); n -= HASH_DATA_AREA; } } }