diff options
Diffstat (limited to 'libcxx/src/random.cpp')
| -rw-r--r-- | libcxx/src/random.cpp | 178 | 
1 files changed, 178 insertions, 0 deletions
| diff --git a/libcxx/src/random.cpp b/libcxx/src/random.cpp new file mode 100644 index 000000000000..04adc59f9bc9 --- /dev/null +++ b/libcxx/src/random.cpp @@ -0,0 +1,178 @@ +//===-------------------------- random.cpp --------------------------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#include <__config> + +#if defined(_LIBCPP_USING_WIN32_RANDOM) +// Must be defined before including stdlib.h to enable rand_s(). +#define _CRT_RAND_S +#endif // defined(_LIBCPP_USING_WIN32_RANDOM) + +#include "random" +#include "system_error" + +#if defined(__sun__) +#define rename solaris_headers_are_broken +#endif // defined(__sun__) + +#include <errno.h> +#include <stdio.h> +#include <stdlib.h> + +#if defined(_LIBCPP_USING_GETENTROPY) +#include <sys/random.h> +#elif defined(_LIBCPP_USING_DEV_RANDOM) +#include <fcntl.h> +#include <unistd.h> +#elif defined(_LIBCPP_USING_NACL_RANDOM) +#include <nacl/nacl_random.h> +#endif + + +_LIBCPP_BEGIN_NAMESPACE_STD + +#if defined(_LIBCPP_USING_GETENTROPY) + +random_device::random_device(const string& __token) +{ +    if (__token != "/dev/urandom") +        __throw_system_error(ENOENT, ("random device not supported " + __token).c_str()); +} + +random_device::~random_device() +{ +} + +unsigned +random_device::operator()() +{ +    unsigned r; +    size_t n = sizeof(r); +    int err = getentropy(&r, n); +    if (err) +        __throw_system_error(errno, "random_device getentropy failed"); +    return r; +} + +#elif defined(_LIBCPP_USING_ARC4_RANDOM) + +random_device::random_device(const string& __token) +{ +    if (__token != "/dev/urandom") +        __throw_system_error(ENOENT, ("random device not supported " + __token).c_str()); +} + +random_device::~random_device() +{ +} + +unsigned +random_device::operator()() +{ +    return arc4random(); +} + +#elif defined(_LIBCPP_USING_DEV_RANDOM) + +random_device::random_device(const string& __token) +    : __f_(open(__token.c_str(), O_RDONLY)) +{ +    if (__f_ < 0) +        __throw_system_error(errno, ("random_device failed to open " + __token).c_str()); +} + +random_device::~random_device() +{ +    close(__f_); +} + +unsigned +random_device::operator()() +{ +    unsigned r; +    size_t n = sizeof(r); +    char* p = reinterpret_cast<char*>(&r); +    while (n > 0) +    { +        ssize_t s = read(__f_, p, n); +        if (s == 0) +            __throw_system_error(ENODATA, "random_device got EOF"); +        if (s == -1) +        { +            if (errno != EINTR) +                __throw_system_error(errno, "random_device got an unexpected error"); +            continue; +        } +        n -= static_cast<size_t>(s); +        p += static_cast<size_t>(s); +    } +    return r; +} + +#elif defined(_LIBCPP_USING_NACL_RANDOM) + +random_device::random_device(const string& __token) +{ +    if (__token != "/dev/urandom") +        __throw_system_error(ENOENT, ("random device not supported " + __token).c_str()); +    int error = nacl_secure_random_init(); +    if (error) +        __throw_system_error(error, ("random device failed to open " + __token).c_str()); +} + +random_device::~random_device() +{ +} + +unsigned +random_device::operator()() +{ +    unsigned r; +    size_t n = sizeof(r); +    size_t bytes_written; +    int error = nacl_secure_random(&r, n, &bytes_written); +    if (error != 0) +        __throw_system_error(error, "random_device failed getting bytes"); +    else if (bytes_written != n) +        __throw_runtime_error("random_device failed to obtain enough bytes"); +    return r; +} + +#elif defined(_LIBCPP_USING_WIN32_RANDOM) + +random_device::random_device(const string& __token) +{ +    if (__token != "/dev/urandom") +        __throw_system_error(ENOENT, ("random device not supported " + __token).c_str()); +} + +random_device::~random_device() +{ +} + +unsigned +random_device::operator()() +{ +    unsigned r; +    errno_t err = rand_s(&r); +    if (err) +        __throw_system_error(err, "random_device rand_s failed."); +    return r; +} + +#else +#error "Random device not implemented for this architecture" +#endif + +double +random_device::entropy() const _NOEXCEPT +{ +    return 0; +} + +_LIBCPP_END_NAMESPACE_STD | 
