open5gs/lib/core/ogs-rand.c

132 lines
3.2 KiB
C

/*
* Copyright (C) 2019 by Sukchan Lee <acetcom@gmail.com>
*
* This file is part of Open5GS.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
#include "core-config-private.h"
#if HAVE_SYS_PARAM_H
#include <sys/param.h>
#endif
#if HAVE_SYS_RANDOM_H
#include <sys/random.h>
#endif
#if HAVE_FCNTL_H
#include <fcntl.h>
#endif
#if HAVE_UNISTD_H
#include <unistd.h>
#endif
#include "ogs-core.h"
void ogs_random(void *buf, size_t buflen)
{
#ifdef _WIN32
void arc4random_buf(void *buf, size_t n);
arc4random_buf(buf, buflen);
#elif defined(HAVE_GETRANDOM)
int rc = getrandom(buf, buflen, GRND_NONBLOCK);
if (rc < 0) {
ogs_log_message(OGS_LOG_FATAL, ogs_errno, "getrandom() failed");
ogs_assert_if_reached();
}
#elif defined(HAVE_ARC4RANDOM_BUF)
arc4random_buf(buf, buflen);
#elif defined(OGS_DEV_RANDOM)
int fd = -1;
/* On BSD/OS 4.1, /dev/random gives out 8 bytes at a time, then
* gives EOF, so reading 'buflen' bytes may require opening the
* device several times. */
do {
int rc;
if (fd == -1)
if ((fd = open(OGS_DEV_RANDOM, O_RDONLY)) == -1) {
ogs_log_message(OGS_LOG_FATAL, ogs_errno, "open() failed");
ogs_assert_if_reached();
}
do {
rc = read(fd, buf, buflen);
} while (rc == -1 && errno == EINTR);
if (rc < 0) {
ogs_log_message(OGS_LOG_ERROR, ogs_errno, "read() failed");
close(fd);
return;
}
else if (rc == 0) {
close(fd);
fd = -1; /* force open() again */
}
else {
buf = (unsigned char *)buf + rc;
buflen -= rc;
}
} while (buflen > 0);
close(fd);
#else
#error cannot find a method which finds a pseudo-random number
#endif
}
uint32_t ogs_random32(void)
{
#ifdef _WIN32
uint32_t arc4random(void);
return arc4random();
#elif defined(HAVE_ARC4RANDOM_BUF)
return arc4random();
#else
uint32_t rand32;
ogs_random(&rand32, sizeof(rand32));
return rand32;
#endif
}
#ifdef _WIN32
#define ARC4RANDOM_NO_INCLUDES
#define ARC4_LOCK_() do {} while(0)
#define ARC4_UNLOCK_() do {} while(0)
#define ARC4RANDOM_UINT32 uint32_t
#define ARC4RANDOM_NOSTIR
#define ARC4RANDOM_NOADDRANDOM
#define ARC4RANDOM_NOUNIFORM
/**
* Stolen from libevent library.
* https://github.com/libevent/libevent/blob/master/evutil.c
*/
void *(*volatile evutil_memset_volatile_)(void *, int, size_t) = memset;
static void evutil_memclear_(void *mem, size_t len)
{
evutil_memset_volatile_(mem, 0, len);
}
#include "arc4random.c"
#endif