|
@@ -27,7 +27,11 @@
|
|
#include <wincrypt.h>
|
|
#include <wincrypt.h>
|
|
static HCRYPTPROV hCryptProv;
|
|
static HCRYPTPROV hCryptProv;
|
|
#else
|
|
#else
|
|
-static int fd; /* a file handle to either /dev/urandom or /dev/random */
|
|
|
|
|
|
+# ifdef CONFIG_GETRANDOM
|
|
|
|
+# include <sys/random.h>
|
|
|
|
+# endif
|
|
|
|
+/* This is -1 for getrandom(), or a file handle for /dev/{u,}random. */
|
|
|
|
+static int fd;
|
|
#endif
|
|
#endif
|
|
|
|
|
|
int qcrypto_random_init(Error **errp)
|
|
int qcrypto_random_init(Error **errp)
|
|
@@ -40,15 +44,20 @@ int qcrypto_random_init(Error **errp)
|
|
return -1;
|
|
return -1;
|
|
}
|
|
}
|
|
#else
|
|
#else
|
|
- /* TBD perhaps also add support for BSD getentropy / Linux
|
|
|
|
- * getrandom syscalls directly */
|
|
|
|
|
|
+# ifdef CONFIG_GETRANDOM
|
|
|
|
+ if (getrandom(NULL, 0, 0) == 0) {
|
|
|
|
+ /* Use getrandom() */
|
|
|
|
+ fd = -1;
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+ /* Fall through to /dev/urandom case. */
|
|
|
|
+# endif
|
|
fd = open("/dev/urandom", O_RDONLY | O_CLOEXEC);
|
|
fd = open("/dev/urandom", O_RDONLY | O_CLOEXEC);
|
|
if (fd == -1 && errno == ENOENT) {
|
|
if (fd == -1 && errno == ENOENT) {
|
|
fd = open("/dev/random", O_RDONLY | O_CLOEXEC);
|
|
fd = open("/dev/random", O_RDONLY | O_CLOEXEC);
|
|
}
|
|
}
|
|
-
|
|
|
|
if (fd < 0) {
|
|
if (fd < 0) {
|
|
- error_setg(errp, "No /dev/urandom or /dev/random found");
|
|
|
|
|
|
+ error_setg_errno(errp, errno, "No /dev/urandom or /dev/random");
|
|
return -1;
|
|
return -1;
|
|
}
|
|
}
|
|
#endif
|
|
#endif
|
|
@@ -66,6 +75,24 @@ int qcrypto_random_bytes(uint8_t *buf G_GNUC_UNUSED,
|
|
return -1;
|
|
return -1;
|
|
}
|
|
}
|
|
#else
|
|
#else
|
|
|
|
+# ifdef CONFIG_GETRANDOM
|
|
|
|
+ if (likely(fd < 0)) {
|
|
|
|
+ while (1) {
|
|
|
|
+ ssize_t got = getrandom(buf, buflen, 0);
|
|
|
|
+ if (likely(got == buflen)) {
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+ if (got >= 0) {
|
|
|
|
+ buflen -= got;
|
|
|
|
+ buf += got;
|
|
|
|
+ } else if (errno != EINTR) {
|
|
|
|
+ error_setg_errno(errp, errno, "getrandom");
|
|
|
|
+ return -1;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ /* Fall through to /dev/urandom case. */
|
|
|
|
+# endif
|
|
while (1) {
|
|
while (1) {
|
|
ssize_t got = read(fd, buf, buflen);
|
|
ssize_t got = read(fd, buf, buflen);
|
|
if (likely(got == buflen)) {
|
|
if (likely(got == buflen)) {
|