using System;
using System.Diagnostics.CodeAnalysis;
using System.Security.Cryptography;
namespace Mhf.Server.Common
{
/*
* Original version by Stephen Toub and Shawn Farkas.
* Random pool and thread safety added by Markus Olsson (freakcode.com).
*
* Original source: http://msdn.microsoft.com/en-us/magazine/cc163367.aspx
*
* Some benchmarks (2009-03-18):
*
* Results produced by calling Next() 1 000 000 times on my machine (dual core 3Ghz)
*
* System.Random completed in 20.4993 ms (avg 0 ms) (first: 0.3454 ms)
* CryptoRandom with pool completed in 132.2408 ms (avg 0.0001 ms) (first: 0.025 ms)
* CryptoRandom without pool completed in 2 sec 587.708 ms (avg 0.0025 ms) (first: 1.4142 ms)
*
* |---------------------|------------------------------------|
* | Implementation | Slowdown compared to System.Random |
* |---------------------|------------------------------------|
* | System.Random | 0 |
* | CryptoRand w pool | 6,6x |
* | CryptoRand w/o pool | 19,5x |
* |---------------------|------------------------------------|
*
* ent (http://www.fourmilab.ch/) results for 16mb of data produced by this class:
*
* > Entropy = 7.999989 bits per byte.
* >
* > Optimum compression would reduce the size of this 16777216 byte file by 0 percent.
* >
* > Chi square distribution for 16777216 samples is 260.64,
* > and randomly would exceed this value 50.00 percent of the times.
* >
* > Arithmetic mean value of data bytes is 127.4974 (127.5 = random).
* > Monte Carlo value for Pi is 3.141838823 (error 0.01 percent).
* > Serial correlation coefficient is 0.000348 (totally uncorrelated = 0.0).
*
* your mileage may vary ;)
*
*/
///
/// A random number generator based on the RNGCryptoServiceProvider.
/// Adapted from the "Tales from the CryptoRandom" article in MSDN Magazine (September 2007)
/// but with explicit guarantee to be thread safe. Note that this implementation also includes
/// an optional (enabled by default) random buffer which provides a significant speed boost as
/// it greatly reduces the amount of calls into unmanaged land.
///
public class CryptoRandom : Random
{
private RNGCryptoServiceProvider _rng = new RNGCryptoServiceProvider();
private byte[] _buffer;
private int _bufferPosition;
///
/// Gets a value indicating whether this instance has random pool enabled.
///
///
/// true if this instance has random pool enabled; otherwise, false.
///
public bool IsRandomPoolEnabled { get; private set; }
///
/// Initializes a new instance of the class with.
/// Using this overload will enable the random buffer pool.
///
public CryptoRandom() : this(true)
{
}
///
/// Initializes a new instance of the class.
/// This method will disregard whatever value is passed as seed and it's only implemented
/// in order to be fully backwards compatible with .
/// Using this overload will enable the random buffer pool.
///
/// The ignored seed.
[SuppressMessage("Microsoft.Usage", "CA1801:ReviewUnusedParameters", MessageId = "ignoredSeed",
Justification = "Cannot remove this parameter as we implement the full API of System.Random")]
public CryptoRandom(int ignoredSeed) : this(true)
{
}
///
/// Initializes a new instance of the class with
/// optional random buffer.
///
/// set to true to enable the random pool buffer for increased performance.
public CryptoRandom(bool enableRandomPool)
{
IsRandomPoolEnabled = enableRandomPool;
}
private void InitBuffer()
{
if (IsRandomPoolEnabled)
{
if (_buffer == null || _buffer.Length != 512)
_buffer = new byte[512];
}
else
{
if (_buffer == null || _buffer.Length != 4)
_buffer = new byte[4];
}
_rng.GetBytes(_buffer);
_bufferPosition = 0;
}
///
/// Returns a nonnegative random number.
///
///
/// A 32-bit signed integer greater than or equal to zero and less than .
///
public override int Next()
{
// Mask away the sign bit so that we always return nonnegative integers
return (int) GetRandomUInt32() & 0x7FFFFFFF;
}
///
/// Returns a nonnegative random number less than the specified maximum.
///
/// The exclusive upper bound of the random number to be generated. must be greater than or equal to zero.
///
/// A 32-bit signed integer greater than or equal to zero, and less than ; that is, the range of return values ordinarily includes zero but not . However, if equals zero, is returned.
///
///
/// is less than zero.
///
public override int Next(int maxValue)
{
if (maxValue < 0)
throw new ArgumentOutOfRangeException("maxValue");
return Next(0, maxValue);
}
///
/// Returns a random number within a specified range.
///
/// The inclusive lower bound of the random number returned.
/// The exclusive upper bound of the random number returned. must be greater than or equal to .
///
/// A 32-bit signed integer greater than or equal to and less than ; that is, the range of return values includes but not . If equals , is returned.
///
///
/// is greater than .
///
public override int Next(int minValue, int maxValue)
{
if (minValue > maxValue)
throw new ArgumentOutOfRangeException("minValue");
if (minValue == maxValue)
return minValue;
long diff = maxValue - minValue;
while (true)
{
uint rand = GetRandomUInt32();
long max = 1 + (long) uint.MaxValue;
long remainder = max % diff;
if (rand < max - remainder)
return (int) (minValue + (rand % diff));
}
}
///
/// Returns a random number between 0.0 and 1.0.
///
///
/// A double-precision floating point number greater than or equal to 0.0, and less than 1.0.
///
public override double NextDouble()
{
return GetRandomUInt32() / (1.0 + uint.MaxValue);
}
///
/// Fills the elements of a specified array of bytes with random numbers.
///
/// An array of bytes to contain random numbers.
///
/// is null.
///
public override void NextBytes(byte[] buffer)
{
if (buffer == null)
throw new ArgumentNullException("buffer");
lock (this)
{
if (IsRandomPoolEnabled && _buffer == null)
InitBuffer();
// Can we fit the requested number of bytes in the buffer?
if (IsRandomPoolEnabled && _buffer.Length <= buffer.Length)
{
int count = buffer.Length;
EnsureRandomBuffer(count);
Buffer.BlockCopy(_buffer, _bufferPosition, buffer, 0, count);
_bufferPosition += count;
}
else
{
// Draw bytes directly from the RNGCryptoProvider
_rng.GetBytes(buffer);
}
}
}
///
/// Gets one random unsigned 32bit integer in a thread safe manner.
///
private uint GetRandomUInt32()
{
lock (this)
{
EnsureRandomBuffer(4);
uint rand = BitConverter.ToUInt32(_buffer, _bufferPosition);
_bufferPosition += 4;
return rand;
}
}
///
/// Ensures that we have enough bytes in the random buffer.
///
/// The number of required bytes.
private void EnsureRandomBuffer(int requiredBytes)
{
if (_buffer == null)
InitBuffer();
if (requiredBytes > _buffer.Length)
throw new ArgumentOutOfRangeException("requiredBytes", "cannot be greater than random buffer");
if ((_buffer.Length - _bufferPosition) < requiredBytes)
InitBuffer();
}
}
}