Blame src/unix/random-sysctl-linux.c

Packit Service e08953
/* Copyright libuv contributors. All rights reserved.
Packit Service e08953
 *
Packit Service e08953
 * Permission is hereby granted, free of charge, to any person obtaining a copy
Packit Service e08953
 * of this software and associated documentation files (the "Software"), to
Packit Service e08953
 * deal in the Software without restriction, including without limitation the
Packit Service e08953
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
Packit Service e08953
 * sell copies of the Software, and to permit persons to whom the Software is
Packit Service e08953
 * furnished to do so, subject to the following conditions:
Packit Service e08953
 *
Packit Service e08953
 * The above copyright notice and this permission notice shall be included in
Packit Service e08953
 * all copies or substantial portions of the Software.
Packit Service e08953
 *
Packit Service e08953
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit Service e08953
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit Service e08953
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
Packit Service e08953
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
Packit Service e08953
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
Packit Service e08953
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
Packit Service e08953
 * IN THE SOFTWARE.
Packit Service e08953
 */
Packit Service e08953
Packit Service e08953
#include "uv.h"
Packit Service e08953
#include "internal.h"
Packit Service e08953
Packit Service e08953
#include <errno.h>
Packit Service e08953
#include <string.h>
Packit Service e08953
Packit Service e08953
#include <syscall.h>
Packit Service e08953
#include <unistd.h>
Packit Service e08953
Packit Service e08953
Packit Service e08953
struct uv__sysctl_args {
Packit Service e08953
  int* name;
Packit Service e08953
  int nlen;
Packit Service e08953
  void* oldval;
Packit Service e08953
  size_t* oldlenp;
Packit Service e08953
  void* newval;
Packit Service e08953
  size_t newlen;
Packit Service e08953
  unsigned long unused[4];
Packit Service e08953
};
Packit Service e08953
Packit Service e08953
Packit Service e08953
int uv__random_sysctl(void* buf, size_t buflen) {
Packit Service e08953
  static int name[] = {1 /*CTL_KERN*/, 40 /*KERN_RANDOM*/, 6 /*RANDOM_UUID*/};
Packit Service e08953
  struct uv__sysctl_args args;
Packit Service e08953
  char uuid[16];
Packit Service e08953
  char* p;
Packit Service e08953
  char* pe;
Packit Service e08953
  size_t n;
Packit Service e08953
Packit Service e08953
  p = buf;
Packit Service e08953
  pe = p + buflen;
Packit Service e08953
Packit Service e08953
  while (p < pe) {
Packit Service e08953
    memset(&args, 0, sizeof(args));
Packit Service e08953
Packit Service e08953
    args.name = name;
Packit Service e08953
    args.nlen = ARRAY_SIZE(name);
Packit Service e08953
    args.oldval = uuid;
Packit Service e08953
    args.oldlenp = &n;
Packit Service e08953
    n = sizeof(uuid);
Packit Service e08953
Packit Service e08953
    /* Emits a deprecation warning with some kernels but that seems like
Packit Service e08953
     * an okay trade-off for the fallback of the fallback: this function is
Packit Service e08953
     * only called when neither getrandom(2) nor /dev/urandom are available.
Packit Service e08953
     * Fails with ENOSYS on kernels configured without CONFIG_SYSCTL_SYSCALL.
Packit Service e08953
     * At least arm64 never had a _sysctl system call and therefore doesn't
Packit Service e08953
     * have a SYS__sysctl define either.
Packit Service e08953
     */
Packit Service e08953
#ifdef SYS__sysctl
Packit Service e08953
    if (syscall(SYS__sysctl, &args) == -1)
Packit Service e08953
      return UV__ERR(errno);
Packit Service e08953
#else
Packit Service e08953
    {
Packit Service e08953
      (void) &arg;;
Packit Service e08953
      return UV_ENOSYS;
Packit Service e08953
    }
Packit Service e08953
#endif
Packit Service e08953
Packit Service e08953
    if (n != sizeof(uuid))
Packit Service e08953
      return UV_EIO;  /* Can't happen. */
Packit Service e08953
Packit Service e08953
    /* uuid[] is now a type 4 UUID. Bytes 6 and 8 (counting from zero) contain
Packit Service e08953
     * 4 and 5 bits of entropy, respectively. For ease of use, we skip those
Packit Service e08953
     * and only use 14 of the 16 bytes.
Packit Service e08953
     */
Packit Service e08953
    uuid[6] = uuid[14];
Packit Service e08953
    uuid[8] = uuid[15];
Packit Service e08953
Packit Service e08953
    n = pe - p;
Packit Service e08953
    if (n > 14)
Packit Service e08953
      n = 14;
Packit Service e08953
Packit Service e08953
    memcpy(p, uuid, n);
Packit Service e08953
    p += n;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  return 0;
Packit Service e08953
}