| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215 |
- #include <assert.h>
- #include <limits.h>
- #include <stdint.h>
- #include <stdlib.h>
- #include <sys/types.h>
- #ifdef __EMSCRIPTEN__
- # include <emscripten.h>
- #endif
- #include "core.h"
- #include "crypto_stream_chacha20.h"
- #include "randombytes.h"
- #ifndef RANDOMBYTES_CUSTOM_IMPLEMENTATION
- # ifdef RANDOMBYTES_DEFAULT_IMPLEMENTATION
- # include "randombytes_internal.h"
- # endif
- # include "randombytes_sysrandom.h"
- #endif
- #include "private/common.h"
- /* C++Builder defines a "random" macro */
- #undef random
- static const randombytes_implementation *implementation;
- #ifndef RANDOMBYTES_DEFAULT_IMPLEMENTATION
- # ifdef __EMSCRIPTEN__
- # define RANDOMBYTES_DEFAULT_IMPLEMENTATION NULL
- # else
- # define RANDOMBYTES_DEFAULT_IMPLEMENTATION &randombytes_sysrandom_implementation
- # endif
- #endif
- #ifdef __EMSCRIPTEN__
- static const char *
- javascript_implementation_name(void)
- {
- return "js";
- }
- static uint32_t
- javascript_random(void)
- {
- return EM_ASM_INT_V({
- return Module.getRandomValue();
- });
- }
- static void
- javascript_stir(void)
- {
- EM_ASM({
- if (Module.getRandomValue === undefined) {
- try {
- var window_ = 'object' === typeof window ? window : self;
- var crypto_ = typeof window_.crypto !== 'undefined' ? window_.crypto : window_.msCrypto;
- var randomValuesStandard = function() {
- var buf = new Uint32Array(1);
- crypto_.getRandomValues(buf);
- return buf[0] >>> 0;
- };
- randomValuesStandard();
- Module.getRandomValue = randomValuesStandard;
- } catch (e) {
- try {
- var crypto = require('crypto');
- var randomValueNodeJS = function() {
- var buf = crypto['randomBytes'](4);
- return (buf[0] << 24 | buf[1] << 16 | buf[2] << 8 | buf[3]) >>> 0;
- };
- randomValueNodeJS();
- Module.getRandomValue = randomValueNodeJS;
- } catch (e) {
- throw 'No secure random number generator found';
- }
- }
- }
- });
- }
- static void
- javascript_buf(void * const buf, const size_t size)
- {
- unsigned char *p = (unsigned char *) buf;
- size_t i;
- for (i = (size_t) 0U; i < size; i++) {
- p[i] = (unsigned char) randombytes_random();
- }
- }
- #endif
- static void
- randombytes_init_if_needed(void)
- {
- if (implementation == NULL) {
- #ifdef __EMSCRIPTEN__
- static randombytes_implementation javascript_implementation;
- javascript_implementation.implementation_name = javascript_implementation_name;
- javascript_implementation.random = javascript_random;
- javascript_implementation.stir = javascript_stir;
- javascript_implementation.buf = javascript_buf;
- implementation = &javascript_implementation;
- #else
- implementation = RANDOMBYTES_DEFAULT_IMPLEMENTATION;
- #endif
- randombytes_stir();
- }
- }
- int
- randombytes_set_implementation(const randombytes_implementation *impl)
- {
- implementation = impl;
- return 0;
- }
- const char *
- randombytes_implementation_name(void)
- {
- randombytes_init_if_needed();
- return implementation->implementation_name();
- }
- uint32_t
- randombytes_random(void)
- {
- randombytes_init_if_needed();
- return implementation->random();
- }
- void
- randombytes_stir(void)
- {
- randombytes_init_if_needed();
- if (implementation->stir != NULL) {
- implementation->stir();
- }
- }
- uint32_t
- randombytes_uniform(const uint32_t upper_bound)
- {
- uint32_t min;
- uint32_t r;
- randombytes_init_if_needed();
- if (implementation->uniform != NULL) {
- return implementation->uniform(upper_bound);
- }
- if (upper_bound < 2) {
- return 0;
- }
- min = (1U + ~upper_bound) % upper_bound; /* = 2**32 mod upper_bound */
- do {
- r = randombytes_random();
- } while (r < min);
- /* r is now clamped to a set whose size mod upper_bound == 0
- * the worst case (2**31+1) requires ~ 2 attempts */
- return r % upper_bound;
- }
- void
- randombytes_buf(void * const buf, const size_t size)
- {
- randombytes_init_if_needed();
- if (size > (size_t) 0U) {
- implementation->buf(buf, size);
- }
- }
- void
- randombytes_buf_deterministic(void * const buf, const size_t size,
- const unsigned char seed[randombytes_SEEDBYTES])
- {
- static const unsigned char nonce[crypto_stream_chacha20_ietf_NONCEBYTES] = {
- 'L', 'i', 'b', 's', 'o', 'd', 'i', 'u', 'm', 'D', 'R', 'G'
- };
- COMPILER_ASSERT(randombytes_SEEDBYTES == crypto_stream_chacha20_ietf_KEYBYTES);
- #if SIZE_MAX > 0x4000000000ULL
- COMPILER_ASSERT(randombytes_BYTES_MAX <= 0x4000000000ULL);
- if (size > 0x4000000000ULL) {
- sodium_misuse();
- }
- #endif
- crypto_stream_chacha20_ietf((unsigned char *) buf, (unsigned long long) size,
- nonce, seed);
- }
- size_t
- randombytes_seedbytes(void)
- {
- return randombytes_SEEDBYTES;
- }
- int
- randombytes_close(void)
- {
- if (implementation != NULL && implementation->close != NULL) {
- return implementation->close();
- }
- return 0;
- }
- void
- randombytes(unsigned char * const buf, const unsigned long long buf_len)
- {
- assert(buf_len <= SIZE_MAX);
- randombytes_buf(buf, (size_t) buf_len);
- }
|