Skip to content

Commit

Permalink
Updated MemoryDiagnoserIsAccurateForMultiThreadedBenchmarks test.
Browse files Browse the repository at this point in the history
  • Loading branch information
timcassell committed Apr 24, 2024
1 parent 2c94c02 commit c192cb5
Showing 1 changed file with 36 additions and 12 deletions.
48 changes: 36 additions & 12 deletions tests/BenchmarkDotNet.IntegrationTests/MemoryDiagnoserTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -253,44 +253,68 @@ public void AllocationQuantumIsNotAnIssueForNetCore21Plus(IToolchain toolchain)

public class MultiThreadedAllocation
{
public const int Size = 1_000_000;
public const int ThreadsCount = 10;
public const int Size = 1024;
public const int ThreadsCount = 1;

// We cache the threads in GlobalSetup and reuse them for each benchmark invocation
// to avoid measuring the cost of thread create and join, which varies across different runtimes.
private Thread[] threads;
private volatile bool keepRunning = true;
private readonly Barrier barrier = new (ThreadsCount + 1);
private readonly CountdownEvent countdownEvent = new (ThreadsCount);

[IterationSetup]
public void SetupIteration()
[GlobalSetup]
public void Setup()
{
threads = Enumerable.Range(0, ThreadsCount)
.Select(_ => new Thread(() => GC.KeepAlive(new byte[Size])))
.Select(_ => new Thread(() =>
{
while (keepRunning)
{
barrier.SignalAndWait();
GC.KeepAlive(new byte[Size]);
countdownEvent.Signal();
}
}))
.ToArray();
foreach (var thread in threads)
{
thread.Start();
}
}

[Benchmark]
public void Allocate()
[GlobalCleanup]
public void Cleanup()
{
keepRunning = false;
barrier.SignalAndWait();
foreach (var thread in threads)
{
thread.Start();
thread.Join();
}
}

[Benchmark]
public void Allocate()
{
countdownEvent.Reset(ThreadsCount);
barrier.SignalAndWait();
countdownEvent.Wait();
}
}

[Theory(Skip = "Test is currently failing on all toolchains.")]
[TheoryEnvSpecific("Full Framework cannot measure precisely enough", EnvRequirement.DotNetCoreOnly)]
[MemberData(nameof(GetToolchains))]
[Trait(Constants.Category, Constants.BackwardCompatibilityCategory)]
public void MemoryDiagnoserIsAccurateForMultiThreadedBenchmarks(IToolchain toolchain)
{
long objectAllocationOverhead = IntPtr.Size * 2; // pointer to method table + object header word
long arraySizeOverhead = IntPtr.Size; // array length
long memoryAllocatedPerArray = (MultiThreadedAllocation.Size + objectAllocationOverhead + arraySizeOverhead);
long threadStartAndJoinOverhead = 112; // this is more or less a magic number taken from memory profiler
long allocatedMemoryPerThread = memoryAllocatedPerArray + threadStartAndJoinOverhead;

AssertAllocations(toolchain, typeof(MultiThreadedAllocation), new Dictionary<string, long>
{
{ nameof(MultiThreadedAllocation.Allocate), allocatedMemoryPerThread * MultiThreadedAllocation.ThreadsCount }
{ nameof(MultiThreadedAllocation.Allocate), memoryAllocatedPerArray * MultiThreadedAllocation.ThreadsCount }
});
}

Expand Down

0 comments on commit c192cb5

Please sign in to comment.