From aef813b02680c3402e87e4cae20e5e63571c5216 Mon Sep 17 00:00:00 2001 From: maxisoft Date: Sun, 19 Nov 2023 12:10:27 +0100 Subject: [PATCH] Enhance GaussianRandom class thread-safety and compliance with GetNonZeroBytes specification This commit modifies the GaussianRandom class to use Interlocked.CompareExchange instead of a bool field to ensure thread-safety when accessing the stored value for the next Gaussian number. This commit also changes the GetNonZeroBytes method to use a Span parameter and a stack-allocated buffer to ensure that no zero bytes are generated, as required by the RandomNumberGenerator base class. Thus, the compliance and performance of the GaussianRandom class are improved. --- ASFFreeGames/RandomUtils.cs | 31 +++++++++++++++++++++++-------- 1 file changed, 23 insertions(+), 8 deletions(-) diff --git a/ASFFreeGames/RandomUtils.cs b/ASFFreeGames/RandomUtils.cs index de69199..378d6aa 100644 --- a/ASFFreeGames/RandomUtils.cs +++ b/ASFFreeGames/RandomUtils.cs @@ -4,6 +4,7 @@ using System.Runtime.CompilerServices; using System.Runtime.InteropServices; using System.Security.Cryptography; +using System.Threading; namespace Maxisoft.ASF; @@ -12,19 +13,32 @@ namespace Maxisoft.ASF; public static class RandomUtils { internal sealed class GaussianRandom : RandomNumberGenerator { // A flag to indicate if there is a stored value for the next Gaussian number - private bool HasNextGaussian; + private int HasNextGaussian; + + private const int True = 1; + private const int False = 0; // The stored value for the next Gaussian number private double NextGaussianValue; public override void GetBytes(byte[] data) => Fill(data); - public override void GetNonZeroBytes(byte[] data) => Fill(data); + public override void GetNonZeroBytes(Span data) { + Fill(data); + Span buffer = stackalloc byte[1]; - private double NextDouble() { - if (HasNextGaussian) { - HasNextGaussian = false; + for (int i = 0; i < data.Length; i++) { + while (data[i] == default(byte)) { + Fill(buffer); + data[i] = buffer[0]; + } + } + } + public override void GetNonZeroBytes(byte[] data) => GetNonZeroBytes((Span) data); + + private double NextDouble() { + if (Interlocked.CompareExchange(ref HasNextGaussian, False, True) == True) { return NextGaussianValue; } @@ -34,12 +48,13 @@ private double NextDouble() { double u1 = ulongs[0] / (double) ulong.MaxValue; double u2 = ulongs[1] / (double) ulong.MaxValue; - // Apply the Box-Muller formula + // Box-Muller formula double r = Math.Sqrt(-2.0f * Math.Log(u1)); double theta = 2.0 * Math.PI * u2; - NextGaussianValue = r * Math.Sin(theta); - HasNextGaussian = true; + if (Interlocked.CompareExchange(ref HasNextGaussian, True, False) == False) { + NextGaussianValue = r * Math.Sin(theta); + } return r * Math.Cos(theta); }