Skip to content

Commit 97768fc

Browse files
committed
util: don't block on getrandom()
1 parent e3c72c2 commit 97768fc

File tree

2 files changed

+28
-4
lines changed

2 files changed

+28
-4
lines changed

src/shared/missing.h

+8
Original file line numberDiff line numberDiff line change
@@ -149,6 +149,14 @@ static inline int getrandom(void *buffer, size_t count, unsigned flags) {
149149
}
150150
#endif
151151

152+
#ifndef GRND_NONBLOCK
153+
#define GRND_NONBLOCK 0x0001
154+
#endif
155+
156+
#ifndef GRND_RANDOM
157+
#define GRND_RANDOM 0x0002
158+
#endif
159+
152160
#ifndef BTRFS_IOCTL_MAGIC
153161
#define BTRFS_IOCTL_MAGIC 0x94
154162
#endif

src/shared/util.c

+20-4
Original file line numberDiff line numberDiff line change
@@ -2470,19 +2470,35 @@ int dev_urandom(void *p, size_t n) {
24702470
int r, fd;
24712471
ssize_t k;
24722472

2473-
/* Use the syscall unless we know we don't have it, or when
2474-
* the requested size is too large for it. */
2473+
/* Gathers some randomness from the kernel. This call will
2474+
* never block, and will always return some data from the
2475+
* kernel, regardless if the random pool is fully initialized
2476+
* or not. It thus makes no guarantee for the quality of the
2477+
* returned entropy, but is good enough for or usual usecases
2478+
* of seeding the hash functions for hashtable */
2479+
2480+
/* Use the getrandom() syscall unless we know we don't have
2481+
* it, or when the requested size is too large for it. */
24752482
if (have_syscall != 0 || (size_t) (int) n != n) {
2476-
r = getrandom(p, n, 0);
2483+
r = getrandom(p, n, GRND_NONBLOCK);
24772484
if (r == (int) n) {
24782485
have_syscall = true;
24792486
return 0;
24802487
}
24812488

24822489
if (r < 0) {
24832490
if (errno == ENOSYS)
2484-
/* we lack the syscall, continue with reading from /dev/urandom */
2491+
/* we lack the syscall, continue with
2492+
* reading from /dev/urandom */
24852493
have_syscall = false;
2494+
else if (errno == EAGAIN)
2495+
/* not enough entropy for now. Let's
2496+
* remember to use the syscall the
2497+
* next time, again, but also read
2498+
* from /dev/urandom for now, which
2499+
* doesn't care about the current
2500+
* amount of entropy. */
2501+
have_syscall = true;
24862502
else
24872503
return -errno;
24882504
} else

0 commit comments

Comments
 (0)