mirror of
https://github.com/claunia/Claunia.ReedSolomon.git
synced 2025-12-16 19:24:45 +00:00
Add ReedSolomonBenchmark.
This commit is contained in:
@@ -2,6 +2,8 @@
|
|||||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Claunia.ReedSolomon", "Claunia.ReedSolomon\Claunia.ReedSolomon.csproj", "{4213781F-6943-446C-B7A4-4E93C3227389}"
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Claunia.ReedSolomon", "Claunia.ReedSolomon\Claunia.ReedSolomon.csproj", "{4213781F-6943-446C-B7A4-4E93C3227389}"
|
||||||
EndProject
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ReedSolomonBenchmark", "ReedSolomonBenchmark\ReedSolomonBenchmark.csproj", "{D29AB600-D6D9-464D-A0DE-C8CEE1FDE73C}"
|
||||||
|
EndProject
|
||||||
Global
|
Global
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
Debug|Any CPU = Debug|Any CPU
|
Debug|Any CPU = Debug|Any CPU
|
||||||
@@ -12,5 +14,9 @@ Global
|
|||||||
{4213781F-6943-446C-B7A4-4E93C3227389}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
{4213781F-6943-446C-B7A4-4E93C3227389}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
{4213781F-6943-446C-B7A4-4E93C3227389}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
{4213781F-6943-446C-B7A4-4E93C3227389}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
{4213781F-6943-446C-B7A4-4E93C3227389}.Release|Any CPU.Build.0 = Release|Any CPU
|
{4213781F-6943-446C-B7A4-4E93C3227389}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{D29AB600-D6D9-464D-A0DE-C8CEE1FDE73C}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{D29AB600-D6D9-464D-A0DE-C8CEE1FDE73C}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{D29AB600-D6D9-464D-A0DE-C8CEE1FDE73C}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{D29AB600-D6D9-464D-A0DE-C8CEE1FDE73C}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
EndGlobal
|
EndGlobal
|
||||||
|
|||||||
14
ReedSolomonBenchmark/Program.cs
Normal file
14
ReedSolomonBenchmark/Program.cs
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
/**
|
||||||
|
* Benchmark of Reed-Solomon encoding.
|
||||||
|
*
|
||||||
|
* Copyright 2015, Backblaze, Inc. All rights reserved.
|
||||||
|
* Copyright © 2019 Natalia Portillo
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace ReedSolomonBenchmark
|
||||||
|
{
|
||||||
|
internal class Program
|
||||||
|
{
|
||||||
|
public static void Main(string[] args) => new ReedSolomonBenchmark().Run();
|
||||||
|
}
|
||||||
|
}
|
||||||
35
ReedSolomonBenchmark/Properties/AssemblyInfo.cs
Normal file
35
ReedSolomonBenchmark/Properties/AssemblyInfo.cs
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
using System.Reflection;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
|
||||||
|
// General Information about an assembly is controlled through the following
|
||||||
|
// set of attributes. Change these attribute values to modify the information
|
||||||
|
// associated with an assembly.
|
||||||
|
[assembly: AssemblyTitle("ReedSolomonBenchmark")]
|
||||||
|
[assembly: AssemblyDescription("")]
|
||||||
|
[assembly: AssemblyConfiguration("")]
|
||||||
|
[assembly: AssemblyCompany("")]
|
||||||
|
[assembly: AssemblyProduct("ReedSolomonBenchmark")]
|
||||||
|
[assembly: AssemblyCopyright("Copyright © 2019")]
|
||||||
|
[assembly: AssemblyTrademark("")]
|
||||||
|
[assembly: AssemblyCulture("")]
|
||||||
|
|
||||||
|
// Setting ComVisible to false makes the types in this assembly not visible
|
||||||
|
// to COM components. If you need to access a type in this assembly from
|
||||||
|
// COM, set the ComVisible attribute to true on that type.
|
||||||
|
[assembly: ComVisible(false)]
|
||||||
|
|
||||||
|
// The following GUID is for the ID of the typelib if this project is exposed to COM
|
||||||
|
[assembly: Guid("D29AB600-D6D9-464D-A0DE-C8CEE1FDE73C")]
|
||||||
|
|
||||||
|
// Version information for an assembly consists of the following four values:
|
||||||
|
//
|
||||||
|
// Major Version
|
||||||
|
// Minor Version
|
||||||
|
// Build Number
|
||||||
|
// Revision
|
||||||
|
//
|
||||||
|
// You can specify all the values or you can default the Build and Revision Numbers
|
||||||
|
// by using the '*' as shown below:
|
||||||
|
// [assembly: AssemblyVersion("1.0.*")]
|
||||||
|
[assembly: AssemblyVersion("1.0.0.0")]
|
||||||
|
[assembly: AssemblyFileVersion("1.0.0.0")]
|
||||||
252
ReedSolomonBenchmark/ReedSolomonBenchmark.cs
Normal file
252
ReedSolomonBenchmark/ReedSolomonBenchmark.cs
Normal file
@@ -0,0 +1,252 @@
|
|||||||
|
/**
|
||||||
|
* Benchmark of Reed-Solomon encoding.
|
||||||
|
*
|
||||||
|
* Copyright 2015, Backblaze, Inc. All rights reserved.
|
||||||
|
* Copyright © 2019 Natalia Portillo
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Text;
|
||||||
|
using Claunia.ReedSolomon;
|
||||||
|
|
||||||
|
namespace ReedSolomonBenchmark
|
||||||
|
{
|
||||||
|
internal sealed class ReedSolomonBenchmark
|
||||||
|
{
|
||||||
|
const int DATA_COUNT = 17;
|
||||||
|
const int PARITY_COUNT = 3;
|
||||||
|
const int TOTAL_COUNT = DATA_COUNT + PARITY_COUNT;
|
||||||
|
const int BUFFER_SIZE = 200 * 1000;
|
||||||
|
const int PROCESSOR_CACHE_SIZE = 10 * 1024 * 1024;
|
||||||
|
const int TWICE_PROCESSOR_CACHE_SIZE = 2 * PROCESSOR_CACHE_SIZE;
|
||||||
|
const int NUMBER_OF_BUFFER_SETS = TWICE_PROCESSOR_CACHE_SIZE / DATA_COUNT / BUFFER_SIZE + 1;
|
||||||
|
const long MEASUREMENT_DURATION = 2 * 1000;
|
||||||
|
|
||||||
|
static readonly Random Random = new Random();
|
||||||
|
|
||||||
|
int nextBuffer;
|
||||||
|
|
||||||
|
internal void Run()
|
||||||
|
{
|
||||||
|
Console.WriteLine("preparing...");
|
||||||
|
BufferSet[] bufferSets = new BufferSet [NUMBER_OF_BUFFER_SETS];
|
||||||
|
|
||||||
|
for(int iBufferSet = 0; iBufferSet < NUMBER_OF_BUFFER_SETS; iBufferSet++)
|
||||||
|
bufferSets[iBufferSet] = new BufferSet();
|
||||||
|
|
||||||
|
byte[] tempBuffer = new byte [BUFFER_SIZE];
|
||||||
|
|
||||||
|
List<string> summaryLines = new List<string>();
|
||||||
|
var csv = new StringBuilder();
|
||||||
|
csv.Append("Outer,Middle,Inner,Multiply,Encode,Check\n");
|
||||||
|
|
||||||
|
foreach(ICodingLoop codingLoop in CodingLoopBase.ALL_CODING_LOOPS)
|
||||||
|
{
|
||||||
|
var encodeAverage = new Measurement();
|
||||||
|
|
||||||
|
{
|
||||||
|
string testName = codingLoop.GetType().Name + " encodeParity";
|
||||||
|
Console.WriteLine("\nTEST: " + testName);
|
||||||
|
var codec = new ReedSolomon(DATA_COUNT, PARITY_COUNT, codingLoop);
|
||||||
|
Console.WriteLine(" warm up...");
|
||||||
|
DoOneEncodeMeasurement(codec, bufferSets);
|
||||||
|
DoOneEncodeMeasurement(codec, bufferSets);
|
||||||
|
Console.WriteLine(" testing...");
|
||||||
|
|
||||||
|
for(int iMeasurement = 0; iMeasurement < 10; iMeasurement++)
|
||||||
|
encodeAverage.Add(DoOneEncodeMeasurement(codec, bufferSets));
|
||||||
|
|
||||||
|
Console.WriteLine("\nAVERAGE: {0}", encodeAverage);
|
||||||
|
summaryLines.Add($" {testName,-45} {encodeAverage}");
|
||||||
|
}
|
||||||
|
|
||||||
|
// The encoding test should have filled all of the buffers with
|
||||||
|
// correct parity, so we can benchmark parity checking.
|
||||||
|
var checkAverage = new Measurement();
|
||||||
|
|
||||||
|
{
|
||||||
|
string testName = codingLoop.GetType().Name + " isParityCorrect";
|
||||||
|
Console.WriteLine("\nTEST: " + testName);
|
||||||
|
var codec = new ReedSolomon(DATA_COUNT, PARITY_COUNT, codingLoop);
|
||||||
|
Console.WriteLine(" warm up...");
|
||||||
|
DoOneEncodeMeasurement(codec, bufferSets);
|
||||||
|
DoOneEncodeMeasurement(codec, bufferSets);
|
||||||
|
Console.WriteLine(" testing...");
|
||||||
|
|
||||||
|
for(int iMeasurement = 0; iMeasurement < 10; iMeasurement++)
|
||||||
|
checkAverage.Add(DoOneCheckMeasurement(codec, bufferSets, tempBuffer));
|
||||||
|
|
||||||
|
Console.WriteLine("\nAVERAGE: {0}", checkAverage);
|
||||||
|
summaryLines.Add($" {testName,-45} {checkAverage}");
|
||||||
|
}
|
||||||
|
|
||||||
|
csv.Append(CodingLoopNameToCsvPrefix(codingLoop.GetType().Name));
|
||||||
|
csv.Append(encodeAverage.GetRate());
|
||||||
|
csv.Append(",");
|
||||||
|
csv.Append(checkAverage.GetRate());
|
||||||
|
csv.Append("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
Console.WriteLine("\n");
|
||||||
|
Console.WriteLine(csv.ToString());
|
||||||
|
|
||||||
|
Console.WriteLine("\nSummary:\n");
|
||||||
|
|
||||||
|
foreach(string line in summaryLines)
|
||||||
|
Console.WriteLine(line);
|
||||||
|
}
|
||||||
|
|
||||||
|
Measurement DoOneEncodeMeasurement(ReedSolomon codec, BufferSet[] bufferSets)
|
||||||
|
{
|
||||||
|
long passesCompleted = 0;
|
||||||
|
long bytesEncoded = 0;
|
||||||
|
long encodingTime = 0;
|
||||||
|
|
||||||
|
while(encodingTime < MEASUREMENT_DURATION)
|
||||||
|
{
|
||||||
|
BufferSet bufferSet = bufferSets[nextBuffer];
|
||||||
|
nextBuffer = (nextBuffer + 1) % bufferSets.Length;
|
||||||
|
byte[][] shards = bufferSet.Buffers;
|
||||||
|
DateTime startTime = DateTime.UtcNow;
|
||||||
|
codec.EncodeParity(shards, 0, BUFFER_SIZE);
|
||||||
|
DateTime endTime = DateTime.UtcNow;
|
||||||
|
encodingTime += (long)(endTime - startTime).TotalMilliseconds;
|
||||||
|
bytesEncoded += BUFFER_SIZE * DATA_COUNT;
|
||||||
|
passesCompleted += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
double seconds = encodingTime / 1000.0;
|
||||||
|
double megabytes = bytesEncoded / 1000000.0;
|
||||||
|
var result = new Measurement(megabytes, seconds);
|
||||||
|
Console.WriteLine(" {0} passes, {1}", passesCompleted, result);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
Measurement DoOneCheckMeasurement(ReedSolomon codec, BufferSet[] bufferSets, byte[] tempBuffer)
|
||||||
|
{
|
||||||
|
long passesCompleted = 0;
|
||||||
|
long bytesChecked = 0;
|
||||||
|
long checkingTime = 0;
|
||||||
|
|
||||||
|
while(checkingTime < MEASUREMENT_DURATION)
|
||||||
|
{
|
||||||
|
BufferSet bufferSet = bufferSets[nextBuffer];
|
||||||
|
nextBuffer = (nextBuffer + 1) % bufferSets.Length;
|
||||||
|
byte[][] shards = bufferSet.Buffers;
|
||||||
|
DateTime startTime = DateTime.UtcNow;
|
||||||
|
|
||||||
|
if(!codec.IsParityCorrect(shards, 0, BUFFER_SIZE, tempBuffer))
|
||||||
|
throw new Exception("parity not correct");
|
||||||
|
|
||||||
|
DateTime endTime = DateTime.UtcNow;
|
||||||
|
checkingTime += (long)(endTime - startTime).TotalMilliseconds;
|
||||||
|
bytesChecked += BUFFER_SIZE * DATA_COUNT;
|
||||||
|
passesCompleted += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
double seconds = checkingTime / 1000.0;
|
||||||
|
double megabytes = bytesChecked / 1000000.0;
|
||||||
|
var result = new Measurement(megabytes, seconds);
|
||||||
|
Console.WriteLine(" {0} passes, {1}", passesCompleted, result);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Converts a name like "OutputByteInputTableCodingLoop" to "output,byte,input,table,".</summary>
|
||||||
|
static string CodingLoopNameToCsvPrefix(string className)
|
||||||
|
{
|
||||||
|
List<string> names = SplitCamelCase(className);
|
||||||
|
|
||||||
|
return names[0] + "," + names[1] + "," + names[2] + "," + names[3] + ",";
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Converts a name like "OutputByteInputTableCodingLoop" to a List of words: { "output", "byte", "input",
|
||||||
|
/// "table", "coding", "loop" }
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="className"></param>
|
||||||
|
/// <returns></returns>
|
||||||
|
static List<string> SplitCamelCase(string className)
|
||||||
|
{
|
||||||
|
string remaining = className;
|
||||||
|
List<string> result = new List<string>();
|
||||||
|
|
||||||
|
while(!string.IsNullOrEmpty(remaining))
|
||||||
|
{
|
||||||
|
bool found = false;
|
||||||
|
|
||||||
|
for(int i = 1; i < remaining.Length; i++)
|
||||||
|
if(char.IsUpper(remaining[i]))
|
||||||
|
{
|
||||||
|
result.Add(remaining.Substring(0, i));
|
||||||
|
remaining = remaining.Substring(i);
|
||||||
|
found = true;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(found)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
result.Add(remaining);
|
||||||
|
remaining = "";
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class BufferSet
|
||||||
|
{
|
||||||
|
public readonly byte[][] Buffers;
|
||||||
|
|
||||||
|
public BufferSet()
|
||||||
|
{
|
||||||
|
Buffers = new byte [TOTAL_COUNT][];
|
||||||
|
|
||||||
|
for(int iBuffer = 0; iBuffer < TOTAL_COUNT; iBuffer++)
|
||||||
|
{
|
||||||
|
Buffers[iBuffer] = new byte[BUFFER_SIZE];
|
||||||
|
byte[] buffer = Buffers[iBuffer];
|
||||||
|
|
||||||
|
for(int iByte = 0; iByte < BUFFER_SIZE; iByte++)
|
||||||
|
buffer[iByte] = (byte)Random.Next(256);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] bigBuffer = new byte [TOTAL_COUNT * BUFFER_SIZE];
|
||||||
|
|
||||||
|
for(int i = 0; i < TOTAL_COUNT * BUFFER_SIZE; i++)
|
||||||
|
bigBuffer[i] = (byte)Random.Next(256);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class Measurement
|
||||||
|
{
|
||||||
|
double megabytes;
|
||||||
|
double seconds;
|
||||||
|
|
||||||
|
public Measurement()
|
||||||
|
{
|
||||||
|
megabytes = 0.0;
|
||||||
|
seconds = 0.0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Measurement(double megabytes, double seconds)
|
||||||
|
{
|
||||||
|
this.megabytes = megabytes;
|
||||||
|
this.seconds = seconds;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Add(Measurement other)
|
||||||
|
{
|
||||||
|
megabytes += other.megabytes;
|
||||||
|
seconds += other.seconds;
|
||||||
|
}
|
||||||
|
|
||||||
|
public double GetRate() => megabytes / seconds;
|
||||||
|
|
||||||
|
public override string ToString() => $"{GetRate():F1} MB/s";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
60
ReedSolomonBenchmark/ReedSolomonBenchmark.csproj
Normal file
60
ReedSolomonBenchmark/ReedSolomonBenchmark.csproj
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<Project ToolsVersion="4.0" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||||
|
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
|
||||||
|
<PropertyGroup>
|
||||||
|
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
|
||||||
|
<Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
|
||||||
|
<ProjectGuid>{D29AB600-D6D9-464D-A0DE-C8CEE1FDE73C}</ProjectGuid>
|
||||||
|
<OutputType>Exe</OutputType>
|
||||||
|
<AppDesignerFolder>Properties</AppDesignerFolder>
|
||||||
|
<RootNamespace>ReedSolomonBenchmark</RootNamespace>
|
||||||
|
<AssemblyName>ReedSolomonBenchmark</AssemblyName>
|
||||||
|
<TargetFrameworkVersion>v4.0</TargetFrameworkVersion>
|
||||||
|
<FileAlignment>512</FileAlignment>
|
||||||
|
</PropertyGroup>
|
||||||
|
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
|
||||||
|
<PlatformTarget>AnyCPU</PlatformTarget>
|
||||||
|
<DebugSymbols>true</DebugSymbols>
|
||||||
|
<DebugType>full</DebugType>
|
||||||
|
<Optimize>false</Optimize>
|
||||||
|
<OutputPath>bin\Debug\</OutputPath>
|
||||||
|
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
||||||
|
<ErrorReport>prompt</ErrorReport>
|
||||||
|
<WarningLevel>4</WarningLevel>
|
||||||
|
</PropertyGroup>
|
||||||
|
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
|
||||||
|
<PlatformTarget>AnyCPU</PlatformTarget>
|
||||||
|
<DebugType>pdbonly</DebugType>
|
||||||
|
<Optimize>true</Optimize>
|
||||||
|
<OutputPath>bin\Release\</OutputPath>
|
||||||
|
<DefineConstants>TRACE</DefineConstants>
|
||||||
|
<ErrorReport>prompt</ErrorReport>
|
||||||
|
<WarningLevel>4</WarningLevel>
|
||||||
|
</PropertyGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<Reference Include="System" />
|
||||||
|
<Reference Include="System.Core" />
|
||||||
|
<Reference Include="System.Data" />
|
||||||
|
<Reference Include="System.Xml" />
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<Compile Include="Program.cs" />
|
||||||
|
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||||
|
<Compile Include="ReedSolomonBenchmark.cs" />
|
||||||
|
</ItemGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\Claunia.ReedSolomon\Claunia.ReedSolomon.csproj">
|
||||||
|
<Project>{4213781f-6943-446c-b7a4-4e93c3227389}</Project>
|
||||||
|
<Name>Claunia.ReedSolomon</Name>
|
||||||
|
</ProjectReference>
|
||||||
|
</ItemGroup>
|
||||||
|
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
||||||
|
<!-- To modify your build process, add your task inside one of the targets below and uncomment it.
|
||||||
|
Other similar extension points exist, see Microsoft.Common.targets.
|
||||||
|
<Target Name="BeforeBuild">
|
||||||
|
</Target>
|
||||||
|
<Target Name="AfterBuild">
|
||||||
|
</Target>
|
||||||
|
-->
|
||||||
|
|
||||||
|
</Project>
|
||||||
Reference in New Issue
Block a user