Implement getRandomBytes() function

This function allows getting arbitrary sized block of random bytes.
Primary motivation is support for --build-id=uuid in lld.

Differential revision: https://reviews.llvm.org/D23671

llvm-svn: 279807
This commit is contained in:
Eugene Leviant 2016-08-26 08:14:54 +00:00
parent 8f27f51192
commit ea877d40b4
3 changed files with 42 additions and 0 deletions

View File

@ -19,6 +19,7 @@
#include "llvm/Support/Compiler.h"
#include "llvm/Support/DataTypes.h" // Needed for uint64_t on Windows.
#include <random>
#include <system_error>
namespace llvm {
class StringRef;
@ -53,6 +54,9 @@ private:
friend class Module;
};
// Get random vector of specified size
std::error_code getRandomBytes(void *Buffer, size_t Size);
}
#endif

View File

@ -17,6 +17,11 @@
#include "llvm/Support/CommandLine.h"
#include "llvm/Support/Debug.h"
#include "llvm/Support/raw_ostream.h"
#ifdef LLVM_ON_WIN32
#include "Windows/WindowsSupport.h"
#else
#include "Unix/Unix.h"
#endif
using namespace llvm;
@ -55,3 +60,32 @@ RandomNumberGenerator::RandomNumberGenerator(StringRef Salt) {
uint_fast64_t RandomNumberGenerator::operator()() {
return Generator();
}
// Get random vector of specified size
std::error_code llvm::getRandomBytes(void *Buffer, size_t Size) {
#ifdef LLVM_ON_WIN32
HCRYPTPROV hProvider;
if (CryptAcquireContext(&hProvider, 0, 0, PROV_RSA_FULL,
CRYPT_VERIFYCONTEXT | CRYPT_SILENT)) {
ScopedCryptContext ScopedHandle(hProvider);
if (CryptGenRandom(hProvider, Size, static_cast<BYTE *>(Buffer)))
return std::error_code();
}
return std::error_code(GetLastError(), std::system_category());
#else
int Fd = open("/dev/urandom", O_RDONLY);
if (Fd != -1) {
std::error_code Ret;
ssize_t BytesRead = read(Fd, Buffer, Size);
if (BytesRead == -1)
Ret = std::error_code(errno, std::system_category());
else if (BytesRead != static_cast<ssize_t>(Size))
Ret = std::error_code(EIO, std::system_category());
if (close(Fd) == -1)
Ret = std::error_code(errno, std::system_category());
return Ret;
}
return std::error_code(errno, std::system_category());
#endif
}

View File

@ -48,6 +48,10 @@
# include <dlfcn.h>
#endif
#ifdef HAVE_FCNTL_H
# include <fcntl.h>
#endif
/// This function builds an error message into \p ErrMsg using the \p prefix
/// string and the Unix error number given by \p errnum. If errnum is -1, the
/// default then the value of errno is used.