source code
/* The Computer Language Benchmarks Game
https://salsa.debian.org/benchmarksgame-team/benchmarksgame/
contributed by Isaac Gouy, transliterated from Oleg Mazurov's Java program
concurrency fix and minor improvements by Peperud
parallel and small optimisations by Anthony Lloyd
15% improvement by optimizing inner loop and compiler options by Thom Kiesewetter
*/
using System.Numerics;
using System.Runtime.CompilerServices;
public static class FannkuchRedux
{
private static int taskCount;
private static int[] fact, chkSums, maxFlips;
private static void FirstPermutation(int[] p, int[] pp, int[] count, int idx, int n)
{
for (var i = 0; i < n; ++i) p[i] = i;
for (var i = count.Length - 1; i > 0; --i)
{
var d = idx / fact[i];
count[i] = d;
if (d > 0)
{
idx %= fact[i];
new Vector<int>(p).CopyTo(pp);
new Vector<int>(p, 4).CopyTo(pp, 4);
for (var j = 0; j <= i; ++j) p[j] = pp[(j + d) % (i + 1)];
}
}
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static int NextPermutation(int[] p, int[] count)
{
var first = p[1];
p[1] = p[0];
p[0] = first;
var i = 1;
while (++count[i] > i)
{
count[i++] = 0;
var next = p[1];
p[0] = next;
for (var j = 1; j < i;) p[j] = p[++j];
p[i] = first;
first = next;
}
return first;
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static int CountFlips(int first, int[] p, int[] pp)
{
if (first == 0) return 0;
if (p[first] == 0) return 1;
new Vector<int>(p).CopyTo(pp);
new Vector<int>(p, 4).CopyTo(pp, 4);
var flips = 1;
var tp = pp[first];
while (tp != 0)
{
var hi = first - 1;
pp[first] = first;
first = tp;
for (var lo = 1; lo < hi; lo++, hi--)
{
var t = pp[lo];
pp[lo] = pp[hi];
pp[hi] = t;
}
tp = pp[first];
flips++;
}
return flips;
}
private static void Run(int n, int taskSize)
{
int[] p = new int[16], pp = new int[16];
var count = new int[n];
int taskId, chkSum = 0, maxFlip = 0;
while ((taskId = Interlocked.Decrement(ref taskCount)) >= 0)
{
FirstPermutation(p, pp, count, taskId * taskSize, n);
var flips = CountFlips(p[0], p, pp);
chkSum += flips;
if (flips > maxFlip) maxFlip = flips;
for (var i = 1; i < taskSize; i++)
{
flips = CountFlips(NextPermutation(p, count), p, pp);
chkSum += (1 - i % 2 * 2) * flips;
if (flips > maxFlip) maxFlip = flips;
}
}
chkSums[-taskId - 1] = chkSum;
maxFlips[-taskId - 1] = maxFlip;
}
public static void Main(string[] args)
{
var n = args.Length > 0 ? int.Parse(args[0]) : 7;
fact = new int[n + 1];
fact[0] = 1;
var factN = 1;
for (var i = 1; i < fact.Length; i++) fact[i] = factN *= i;
taskCount = n > 9
? factN / (7 * 6 * 5 * 4 * 3 * 2)
: Environment.ProcessorCount;
var taskSize = factN / taskCount;
var nThreads = Environment.ProcessorCount;
chkSums = new int[nThreads];
maxFlips = new int[nThreads];
var threads = new Thread[nThreads];
for (var i = 1; i < nThreads; i++)
(threads[i] = new Thread(() => Run(n, taskSize))).Start();
Run(n, taskSize);
int chkSum = chkSums[0], maxFlip = maxFlips[0];
for (var i = 1; i < threads.Length; i++)
{
threads[i].Join();
chkSum += chkSums[i];
if (maxFlips[i] > maxFlip) maxFlip = maxFlips[i];
}
Console.Out.WriteLineAsync(chkSum + "\nPfannkuchen(" + n + ") = " + maxFlip);
}
}
notes, command-line, and program output
NOTES:
64-bit Ubuntu quad core
.NET SDK 8.0.301
Host Version: 8.0.6
Commit: 3b8b000a0e
<OutputType>Exe
<TargetFramework>net8.0
<ImplicitUsings>enable
<Nullable>enable
<AllowUnsafeBlocks>true
<ServerGarbageCollection>true
<ConcurrentGarbageCollection>true
<PublishAot>false
<OptimizationPreference>Speed
<IlcInstructionSet>native
Wed, 29 May 2024 21:20:48 GMT
MAKE:
cp fannkuchredux.csharpcore-7.csharpcore Program.cs
cp Include/csharpcore/program.csproj .
mkdir obj
cp Include/csharpcore/project.assets.json ./obj
~/dotnet/dotnet build -c Release --use-current-runtime
Determining projects to restore...
Restored /home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/program.csproj (in 840 ms).
/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/Program.cs(16,26): warning CS8618: Non-nullable field 'fact' must contain a non-null value when exiting constructor. Consider declaring the field as nullable. [/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/program.csproj]
/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/Program.cs(16,32): warning CS8618: Non-nullable field 'chkSums' must contain a non-null value when exiting constructor. Consider declaring the field as nullable. [/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/program.csproj]
/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/Program.cs(16,41): warning CS8618: Non-nullable field 'maxFlips' must contain a non-null value when exiting constructor. Consider declaring the field as nullable. [/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/program.csproj]
program -> /home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/bin/Release/net8.0/linux-x64/program.dll
Build succeeded.
/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/Program.cs(16,26): warning CS8618: Non-nullable field 'fact' must contain a non-null value when exiting constructor. Consider declaring the field as nullable. [/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/program.csproj]
/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/Program.cs(16,32): warning CS8618: Non-nullable field 'chkSums' must contain a non-null value when exiting constructor. Consider declaring the field as nullable. [/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/program.csproj]
/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/Program.cs(16,41): warning CS8618: Non-nullable field 'maxFlips' must contain a non-null value when exiting constructor. Consider declaring the field as nullable. [/home/dunham/all-benchmarksgame/benchmarksgame_i53330/fannkuchredux/tmp/program.csproj]
3 Warning(s)
0 Error(s)
Time Elapsed 00:00:05.31
7.03s to complete and log all make actions
COMMAND LINE:
./bin/Release/net8.0/linux-x64/program 10
UNEXPECTED OUTPUT
1,2c1,2
< 64380
< Pfannkuchen(10) = 22
---
> 73196
> Pfannkuchen(10) = 38
PROGRAM OUTPUT:
64380
Pfannkuchen(10) = 22