Skip to content

Commit f3b979b

Browse files
committed
[libc++] Use ioctl when available to get random_device entropy.
Implemented the idea from D94571 to improve entropy on Linux. Reviewed By: ldionne, #libc Differential Revision: https://reviews.llvm.org/D94953
1 parent 266820b commit f3b979b

File tree

2 files changed

+25
-5
lines changed

2 files changed

+25
-5
lines changed

libcxx/src/random.cpp

+19
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
#define _CRT_RAND_S
1414
#endif // defined(_LIBCPP_USING_WIN32_RANDOM)
1515

16+
#include "limits"
1617
#include "random"
1718
#include "system_error"
1819

@@ -29,6 +30,10 @@
2930
#elif defined(_LIBCPP_USING_DEV_RANDOM)
3031
#include <fcntl.h>
3132
#include <unistd.h>
33+
#if __has_include(<sys/ioctl.h>) && __has_include(<linux/random.h>)
34+
#include <sys/ioctl.h>
35+
#include <linux/random.h>
36+
#endif
3237
#elif defined(_LIBCPP_USING_NACL_RANDOM)
3338
#include <nacl/nacl_random.h>
3439
#endif
@@ -172,7 +177,21 @@ random_device::operator()()
172177
double
173178
random_device::entropy() const _NOEXCEPT
174179
{
180+
#if defined(_LIBCPP_USING_DEV_RANDOM) && defined(RNDGETENTCNT)
181+
int ent;
182+
if (::ioctl(__f_, RNDGETENTCNT, &ent) < 0)
183+
return 0;
184+
185+
if (ent < 0)
175186
return 0;
187+
188+
if (ent > std::numeric_limits<result_type>::digits)
189+
return std::numeric_limits<result_type>::digits;
190+
191+
return ent;
192+
#else
193+
return 0;
194+
#endif
176195
}
177196

178197
_LIBCPP_END_NAMESPACE_STD

libcxx/test/std/numerics/rand/rand.device/entropy.pass.cpp

+6-5
Original file line numberDiff line numberDiff line change
@@ -16,14 +16,15 @@
1616

1717
#include <random>
1818
#include <cassert>
19+
#include <climits>
1920

2021
#include "test_macros.h"
2122

22-
int main(int, char**)
23-
{
24-
std::random_device r;
25-
double e = r.entropy();
26-
((void)e); // Prevent unused warning
23+
int main(int, char**) {
24+
std::random_device r;
25+
double e = r.entropy();
26+
assert(e >= 0);
27+
assert(e <= sizeof(typename std::random_device::result_type) * CHAR_BIT);
2728

2829
return 0;
2930
}

0 commit comments

Comments
 (0)