2025-07-03 12:53:49 +04:00

96 lines
2.9 KiB
C++

#include "il2cpp-config.h"
#include "vm/Random.h"
#include "os/Cryptography.h"
namespace il2cpp
{
namespace vm
{
bool Random::Open()
{
return il2cpp::os::Cryptography::OpenCryptographyProvider();
}
void* Random::Create()
{
il2cpp::os::Cryptography::OpenCryptographyProvider();
return il2cpp::os::Cryptography::GetCryptographyProvider();
}
void Random::Free(void* handle)
{
il2cpp::os::Cryptography::ReleaseCryptographyProvider(handle);
}
/**
* mono_rand_try_get_bytes:
* @handle: A pointer to an RNG handle. Handle is set to NULL on failure.
* @buffer: A buffer into which to write random data.
* @buffer_size: Number of bytes to write into buffer.
* @error: Set on error.
*
* Returns: FALSE on failure and sets @error, TRUE on success.
*
* Extracts bytes from an RNG handle.
*/
bool Random::TryGetBytes(void* *handle, unsigned char *buffer, int buffer_size)
{
IL2CPP_ASSERT(handle);
void* provider = *handle;
if (!il2cpp::os::Cryptography::FillBufferWithRandomBytes(provider, buffer_size, buffer))
{
il2cpp::os::Cryptography::ReleaseCryptographyProvider(provider);
/* we may have lost our context with CryptoAPI, but all hope isn't lost yet! */
provider = il2cpp::os::Cryptography::GetCryptographyProvider();
if (!il2cpp::os::Cryptography::FillBufferWithRandomBytes(provider, buffer_size, buffer))
{
il2cpp::os::Cryptography::ReleaseCryptographyProvider(provider);
*handle = 0;
//mono_error_set_execution_engine(error, "Failed to gen random bytes (%d)", GetLastError());
return false;
}
}
return true;
}
/**
* mono_rand_try_get_uint32:
* @handle: A pointer to an RNG handle. Handle is set to NULL on failure.
* @val: A pointer to a 32-bit unsigned int, to which the result will be written.
* @min: Result will be greater than or equal to this value.
* @max: Result will be less than or equal to this value.
*
* Returns: FALSE on failure, TRUE on success.
*
* Extracts one 32-bit unsigned int from an RNG handle.
*/
bool Random::TryGetUnsignedInt32(void* *handle, uint32_t *val, uint32_t min, uint32_t max)
{
IL2CPP_ASSERT(val);
if (!TryGetBytes(handle, (unsigned char*)val, sizeof(uint32_t)))
return false;
double randomDouble = ((double)*val) / (((double)UINT32_MAX) + 1); // Range is [0,1)
*val = (uint32_t)(randomDouble * (max - min + 1) + min);
IL2CPP_ASSERT(*val >= min);
IL2CPP_ASSERT(*val <= max);
return true;
}
uint32_t Random::Next(void** handle, uint32_t min, uint32_t max)
{
uint32_t val;
bool ok = TryGetUnsignedInt32(handle, &val, min, max);
IL2CPP_ASSERT(ok);
return val;
}
} /* namespace vm */
} /* namespace il2cpp */