OSDN Git Service

Added XOR table generator "V2", highly experimental.
authorLoRd_MuldeR <mulder2@gmx.de>
Fri, 29 Sep 2017 19:17:44 +0000 (21:17 +0200)
committerLoRd_MuldeR <mulder2@gmx.de>
Fri, 29 Sep 2017 19:17:44 +0000 (21:17 +0200)
tools/GenTables/GenTables.sln
tools/GenTables/GenTables_XOR_V2.vcxproj [new file with mode: 0644]
tools/GenTables/GenTables_XOR_V2.vcxproj.filters [new file with mode: 0644]
tools/GenTables/src/gen_table_xor_v2.c [new file with mode: 0644]

index 6f31439..393adbb 100644 (file)
@@ -1,12 +1,14 @@
 
 Microsoft Visual Studio Solution File, Format Version 12.00
 # Visual Studio 15
-VisualStudioVersion = 15.0.26430.16
+VisualStudioVersion = 15.0.26730.16
 MinimumVisualStudioVersion = 10.0.40219.1
 Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "GenTables_XOR", "GenTables_XOR.vcxproj", "{0167CFFA-8052-4BA8-9DA6-EE0E7EDF3A74}"
 EndProject
 Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "GenTables_MIX", "GenTables_MIX.vcxproj", "{CB02E071-218F-4714-A623-99DBFC115A22}"
 EndProject
+Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "GenTables_XOR_V2", "GenTables_XOR_V2.vcxproj", "{83CA38F1-6018-4FD0-B7CC-192BD3F29715}"
+EndProject
 Global
        GlobalSection(SolutionConfigurationPlatforms) = preSolution
                Debug|x64 = Debug|x64
@@ -31,8 +33,19 @@ Global
                {CB02E071-218F-4714-A623-99DBFC115A22}.Release|x64.Build.0 = Release|x64
                {CB02E071-218F-4714-A623-99DBFC115A22}.Release|x86.ActiveCfg = Release|Win32
                {CB02E071-218F-4714-A623-99DBFC115A22}.Release|x86.Build.0 = Release|Win32
+               {83CA38F1-6018-4FD0-B7CC-192BD3F29715}.Debug|x64.ActiveCfg = Debug|x64
+               {83CA38F1-6018-4FD0-B7CC-192BD3F29715}.Debug|x64.Build.0 = Debug|x64
+               {83CA38F1-6018-4FD0-B7CC-192BD3F29715}.Debug|x86.ActiveCfg = Debug|Win32
+               {83CA38F1-6018-4FD0-B7CC-192BD3F29715}.Debug|x86.Build.0 = Debug|Win32
+               {83CA38F1-6018-4FD0-B7CC-192BD3F29715}.Release|x64.ActiveCfg = Release|x64
+               {83CA38F1-6018-4FD0-B7CC-192BD3F29715}.Release|x64.Build.0 = Release|x64
+               {83CA38F1-6018-4FD0-B7CC-192BD3F29715}.Release|x86.ActiveCfg = Release|Win32
+               {83CA38F1-6018-4FD0-B7CC-192BD3F29715}.Release|x86.Build.0 = Release|Win32
        EndGlobalSection
        GlobalSection(SolutionProperties) = preSolution
                HideSolutionNode = FALSE
        EndGlobalSection
+       GlobalSection(ExtensibilityGlobals) = postSolution
+               SolutionGuid = {EDD7E6EE-D331-40DB-852C-A4FDC8AF68BC}
+       EndGlobalSection
 EndGlobal
diff --git a/tools/GenTables/GenTables_XOR_V2.vcxproj b/tools/GenTables/GenTables_XOR_V2.vcxproj
new file mode 100644 (file)
index 0000000..2df4bb4
--- /dev/null
@@ -0,0 +1,207 @@
+<?xml version="1.0" encoding="utf-8"?>
+<Project DefaultTargets="Build" ToolsVersion="14.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
+  <ItemGroup Label="ProjectConfigurations">
+    <ProjectConfiguration Include="Debug|Win32">
+      <Configuration>Debug</Configuration>
+      <Platform>Win32</Platform>
+    </ProjectConfiguration>
+    <ProjectConfiguration Include="Release|Win32">
+      <Configuration>Release</Configuration>
+      <Platform>Win32</Platform>
+    </ProjectConfiguration>
+    <ProjectConfiguration Include="Debug|x64">
+      <Configuration>Debug</Configuration>
+      <Platform>x64</Platform>
+    </ProjectConfiguration>
+    <ProjectConfiguration Include="Release|x64">
+      <Configuration>Release</Configuration>
+      <Platform>x64</Platform>
+    </ProjectConfiguration>
+  </ItemGroup>
+  <ItemGroup>
+    <ClCompile Include="src\gen_table_xor_v2.c" />
+  </ItemGroup>
+  <ItemGroup>
+    <ClInclude Include="include\boxmuller.h" />
+    <ClInclude Include="include\common.h" />
+    <ClInclude Include="include\hd_table.h" />
+    <ClInclude Include="include\msws.h" />
+    <ClInclude Include="include\thread_utils.h" />
+  </ItemGroup>
+  <PropertyGroup Label="Globals">
+    <ProjectGuid>{83CA38F1-6018-4FD0-B7CC-192BD3F29715}</ProjectGuid>
+    <Keyword>Win32Proj</Keyword>
+    <RootNamespace>GenTables_XOR</RootNamespace>
+    <WindowsTargetPlatformVersion>10.0.10586.0</WindowsTargetPlatformVersion>
+  </PropertyGroup>
+  <Import Project="$(VCTargetsPath)\Microsoft.Cpp.Default.props" />
+  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'" Label="Configuration">
+    <ConfigurationType>Application</ConfigurationType>
+    <UseDebugLibraries>true</UseDebugLibraries>
+    <PlatformToolset>v141_xp</PlatformToolset>
+    <CharacterSet>Unicode</CharacterSet>
+  </PropertyGroup>
+  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'" Label="Configuration">
+    <ConfigurationType>Application</ConfigurationType>
+    <UseDebugLibraries>false</UseDebugLibraries>
+    <PlatformToolset>v141_xp</PlatformToolset>
+    <WholeProgramOptimization>true</WholeProgramOptimization>
+    <CharacterSet>Unicode</CharacterSet>
+  </PropertyGroup>
+  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'" Label="Configuration">
+    <ConfigurationType>Application</ConfigurationType>
+    <UseDebugLibraries>true</UseDebugLibraries>
+    <PlatformToolset>v141_xp</PlatformToolset>
+    <CharacterSet>Unicode</CharacterSet>
+  </PropertyGroup>
+  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'" Label="Configuration">
+    <ConfigurationType>Application</ConfigurationType>
+    <UseDebugLibraries>false</UseDebugLibraries>
+    <PlatformToolset>v141_xp</PlatformToolset>
+    <WholeProgramOptimization>true</WholeProgramOptimization>
+    <CharacterSet>Unicode</CharacterSet>
+  </PropertyGroup>
+  <Import Project="$(VCTargetsPath)\Microsoft.Cpp.props" />
+  <ImportGroup Label="ExtensionSettings">
+  </ImportGroup>
+  <ImportGroup Label="Shared">
+  </ImportGroup>
+  <ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
+    <Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
+  </ImportGroup>
+  <ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
+    <Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
+  </ImportGroup>
+  <ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
+    <Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
+  </ImportGroup>
+  <ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
+    <Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
+  </ImportGroup>
+  <PropertyGroup Label="UserMacros" />
+  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
+    <LinkIncremental>true</LinkIncremental>
+    <OutDir>$(SolutionDir)\bin\$(Platform)\$(Configuration)\</OutDir>
+    <IntDir>$(SolutionDir)\obj\$(Platform)\$(Configuration)\$(ProjectName)\</IntDir>
+  </PropertyGroup>
+  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
+    <LinkIncremental>true</LinkIncremental>
+    <OutDir>$(SolutionDir)\bin\$(Platform)\$(Configuration)\</OutDir>
+    <IntDir>$(SolutionDir)\obj\$(Platform)\$(Configuration)\$(ProjectName)\</IntDir>
+  </PropertyGroup>
+  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
+    <LinkIncremental>false</LinkIncremental>
+    <OutDir>$(SolutionDir)\bin\$(Platform)\$(Configuration)\</OutDir>
+    <IntDir>$(SolutionDir)\obj\$(Platform)\$(Configuration)\$(ProjectName)\</IntDir>
+  </PropertyGroup>
+  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
+    <LinkIncremental>false</LinkIncremental>
+    <OutDir>$(SolutionDir)\bin\$(Platform)\$(Configuration)\</OutDir>
+    <IntDir>$(SolutionDir)\obj\$(Platform)\$(Configuration)\$(ProjectName)\</IntDir>
+  </PropertyGroup>
+  <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
+    <ClCompile>
+      <PrecompiledHeader>
+      </PrecompiledHeader>
+      <WarningLevel>Level3</WarningLevel>
+      <Optimization>Disabled</Optimization>
+      <PreprocessorDefinitions>WIN32;_DEBUG;_CONSOLE;PTW32_STATIC_LIB;%(PreprocessorDefinitions)</PreprocessorDefinitions>
+      <AdditionalIncludeDirectories>$(SolutionDir)\include;$(SolutionDir)\..\..\..\Prerequisites\PthreadsW32\include</AdditionalIncludeDirectories>
+      <DisableSpecificWarnings>4996</DisableSpecificWarnings>
+      <RuntimeLibrary>MultiThreadedDebug</RuntimeLibrary>
+    </ClCompile>
+    <Link>
+      <SubSystem>Console</SubSystem>
+      <GenerateDebugInformation>true</GenerateDebugInformation>
+      <AdditionalLibraryDirectories>$(SolutionDir)\..\..\..\Prerequisites\PthreadsW32\lib\$(Platform)\shared;%(AdditionalLibraryDirectories)</AdditionalLibraryDirectories>
+      <AdditionalDependencies>pthreadVC2.$(PlatformToolset).lib;%(AdditionalDependencies)</AdditionalDependencies>
+    </Link>
+  </ItemDefinitionGroup>
+  <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
+    <ClCompile>
+      <PrecompiledHeader>
+      </PrecompiledHeader>
+      <WarningLevel>Level3</WarningLevel>
+      <Optimization>Disabled</Optimization>
+      <PreprocessorDefinitions>_DEBUG;_CONSOLE;PTW32_STATIC_LIB;%(PreprocessorDefinitions)</PreprocessorDefinitions>
+      <AdditionalIncludeDirectories>$(SolutionDir)\include;$(SolutionDir)\..\..\..\Prerequisites\PthreadsW32\include</AdditionalIncludeDirectories>
+      <DisableSpecificWarnings>4996</DisableSpecificWarnings>
+      <RuntimeLibrary>MultiThreadedDebug</RuntimeLibrary>
+    </ClCompile>
+    <Link>
+      <SubSystem>Console</SubSystem>
+      <GenerateDebugInformation>true</GenerateDebugInformation>
+      <AdditionalLibraryDirectories>$(SolutionDir)\..\..\..\Prerequisites\PthreadsW32\lib\$(Platform)\shared;%(AdditionalLibraryDirectories)</AdditionalLibraryDirectories>
+      <AdditionalDependencies>pthreadVC2.$(PlatformToolset).lib;%(AdditionalDependencies)</AdditionalDependencies>
+    </Link>
+  </ItemDefinitionGroup>
+  <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
+    <ClCompile>
+      <WarningLevel>Level3</WarningLevel>
+      <PrecompiledHeader>NotUsing</PrecompiledHeader>
+      <Optimization>Full</Optimization>
+      <FunctionLevelLinking>true</FunctionLevelLinking>
+      <IntrinsicFunctions>true</IntrinsicFunctions>
+      <PreprocessorDefinitions>WIN32;NDEBUG;_CONSOLE;PTW32_STATIC_LIB;%(PreprocessorDefinitions);</PreprocessorDefinitions>
+      <EnableEnhancedInstructionSet>AdvancedVectorExtensions2</EnableEnhancedInstructionSet>
+      <RuntimeLibrary>MultiThreaded</RuntimeLibrary>
+      <BufferSecurityCheck>false</BufferSecurityCheck>
+      <InlineFunctionExpansion>AnySuitable</InlineFunctionExpansion>
+      <FavorSizeOrSpeed>Speed</FavorSizeOrSpeed>
+      <OmitFramePointers>true</OmitFramePointers>
+      <AdditionalIncludeDirectories>$(SolutionDir)\include;$(SolutionDir)\..\..\..\Prerequisites\PthreadsW32\include</AdditionalIncludeDirectories>
+      <DisableSpecificWarnings>4996</DisableSpecificWarnings>
+      <ControlFlowGuard>false</ControlFlowGuard>
+      <FloatingPointModel>Fast</FloatingPointModel>
+      <CreateHotpatchableImage>false</CreateHotpatchableImage>
+      <WholeProgramOptimization>false</WholeProgramOptimization>
+      <ExceptionHandling>false</ExceptionHandling>
+      <MinimalRebuild>false</MinimalRebuild>
+    </ClCompile>
+    <Link>
+      <SubSystem>Console</SubSystem>
+      <EnableCOMDATFolding>true</EnableCOMDATFolding>
+      <OptimizeReferences>true</OptimizeReferences>
+      <GenerateDebugInformation>false</GenerateDebugInformation>
+      <AdditionalLibraryDirectories>$(SolutionDir)\..\..\..\Prerequisites\PthreadsW32\lib\$(Platform)\static;%(AdditionalLibraryDirectories)</AdditionalLibraryDirectories>
+      <AdditionalDependencies>pthreadVC2.$(PlatformToolset).lib;%(AdditionalDependencies)</AdditionalDependencies>
+      <LinkTimeCodeGeneration>UseLinkTimeCodeGeneration</LinkTimeCodeGeneration>
+    </Link>
+  </ItemDefinitionGroup>
+  <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
+    <ClCompile>
+      <WarningLevel>Level3</WarningLevel>
+      <PrecompiledHeader>NotUsing</PrecompiledHeader>
+      <Optimization>Full</Optimization>
+      <FunctionLevelLinking>true</FunctionLevelLinking>
+      <IntrinsicFunctions>true</IntrinsicFunctions>
+      <PreprocessorDefinitions>NDEBUG;_CONSOLE;PTW32_STATIC_LIB;%(PreprocessorDefinitions);</PreprocessorDefinitions>
+      <AdditionalIncludeDirectories>$(SolutionDir)\include;$(SolutionDir)\..\..\..\Prerequisites\PthreadsW32\include</AdditionalIncludeDirectories>
+      <DisableSpecificWarnings>4996</DisableSpecificWarnings>
+      <RuntimeLibrary>MultiThreaded</RuntimeLibrary>
+      <BufferSecurityCheck>false</BufferSecurityCheck>
+      <ControlFlowGuard>false</ControlFlowGuard>
+      <EnableEnhancedInstructionSet>AdvancedVectorExtensions2</EnableEnhancedInstructionSet>
+      <FloatingPointModel>Fast</FloatingPointModel>
+      <CreateHotpatchableImage>false</CreateHotpatchableImage>
+      <InlineFunctionExpansion>AnySuitable</InlineFunctionExpansion>
+      <FavorSizeOrSpeed>Speed</FavorSizeOrSpeed>
+      <OmitFramePointers>true</OmitFramePointers>
+      <WholeProgramOptimization>true</WholeProgramOptimization>
+      <ExceptionHandling>false</ExceptionHandling>
+      <MinimalRebuild>false</MinimalRebuild>
+    </ClCompile>
+    <Link>
+      <SubSystem>Console</SubSystem>
+      <EnableCOMDATFolding>true</EnableCOMDATFolding>
+      <OptimizeReferences>true</OptimizeReferences>
+      <GenerateDebugInformation>false</GenerateDebugInformation>
+      <AdditionalLibraryDirectories>$(SolutionDir)\..\..\..\Prerequisites\PthreadsW32\lib\$(Platform)\static;%(AdditionalLibraryDirectories)</AdditionalLibraryDirectories>
+      <AdditionalDependencies>pthreadVC2.$(PlatformToolset).lib;%(AdditionalDependencies)</AdditionalDependencies>
+      <LinkTimeCodeGeneration>UseLinkTimeCodeGeneration</LinkTimeCodeGeneration>
+    </Link>
+  </ItemDefinitionGroup>
+  <Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
+  <ImportGroup Label="ExtensionTargets">
+  </ImportGroup>
+</Project>
\ No newline at end of file
diff --git a/tools/GenTables/GenTables_XOR_V2.vcxproj.filters b/tools/GenTables/GenTables_XOR_V2.vcxproj.filters
new file mode 100644 (file)
index 0000000..8033083
--- /dev/null
@@ -0,0 +1,39 @@
+<?xml version="1.0" encoding="utf-8"?>
+<Project ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
+  <ItemGroup>
+    <Filter Include="Source Files">
+      <UniqueIdentifier>{4FC737F1-C7A5-4376-A066-2A32D752A2FF}</UniqueIdentifier>
+      <Extensions>cpp;c;cc;cxx;def;odl;idl;hpj;bat;asm;asmx</Extensions>
+    </Filter>
+    <Filter Include="Header Files">
+      <UniqueIdentifier>{93995380-89BD-4b04-88EB-625FBE52EBFB}</UniqueIdentifier>
+      <Extensions>h;hh;hpp;hxx;hm;inl;inc;xsd</Extensions>
+    </Filter>
+    <Filter Include="Resource Files">
+      <UniqueIdentifier>{67DA6AB6-F800-4c08-8B7A-83BB121AAD01}</UniqueIdentifier>
+      <Extensions>rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms</Extensions>
+    </Filter>
+  </ItemGroup>
+  <ItemGroup>
+    <ClInclude Include="include\thread_utils.h">
+      <Filter>Header Files</Filter>
+    </ClInclude>
+    <ClInclude Include="include\common.h">
+      <Filter>Header Files</Filter>
+    </ClInclude>
+    <ClInclude Include="include\hd_table.h">
+      <Filter>Header Files</Filter>
+    </ClInclude>
+    <ClInclude Include="include\boxmuller.h">
+      <Filter>Header Files</Filter>
+    </ClInclude>
+    <ClInclude Include="include\msws.h">
+      <Filter>Header Files</Filter>
+    </ClInclude>
+  </ItemGroup>
+  <ItemGroup>
+    <ClCompile Include="src\gen_table_xor_v2.c">
+      <Filter>Source Files</Filter>
+    </ClCompile>
+  </ItemGroup>
+</Project>
\ No newline at end of file
diff --git a/tools/GenTables/src/gen_table_xor_v2.c b/tools/GenTables/src/gen_table_xor_v2.c
new file mode 100644 (file)
index 0000000..fc79279
--- /dev/null
@@ -0,0 +1,518 @@
+/* ----------------------------------------------------------------------------------------------- */
+/* MHash-384 - Generate tables utility program                                                     */
+/* Copyright(c) 2016-2017 LoRd_MuldeR <mulder2@gmx.de>                                             */
+/*                                                                                                 */
+/* Permission is hereby granted, free of charge, to any person obtaining a copy of this software   */
+/* and associated documentation files(the "Software"), to deal in the Software without             */
+/* restriction, including without limitation the rights to use, copy, modify, merge, publish,      */
+/* distribute, sublicense, and / or sell copies of the Software, and to permit persons to whom the */
+/* Software is furnished to do so, subject to the following conditions:                            */
+/*                                                                                                 */
+/* The above copyright notice and this permission notice shall be included in all copies or        */
+/* substantial portions of the Software.                                                           */
+/*                                                                                                 */
+/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING   */
+/* BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND      */
+/* NONINFRINGEMENT.IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,     */
+/* DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,  */
+/* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.         */
+/* ----------------------------------------------------------------------------------------------- */
+
+#include "common.h"
+#include "thread_utils.h"
+#include "msws.h"
+#include "boxmuller.h"
+
+#include <stdio.h>
+#include <stdint.h>
+#include <string.h>
+#include <time.h>
+#include <stdbool.h>
+#include <io.h>
+#include <float.h>
+#include <math.h>
+
+//-----------------------------------------------------------------------------
+// Const
+//-----------------------------------------------------------------------------
+
+#define HASH_LEN 384U
+#define DISTANCE_MIN 192U
+
+#define THREAD_COUNT 8U
+
+#undef ENABLE_TRACE
+
+#define ROW_NUM (UINT8_MAX+2)           /*total number of rows*/
+#define ROW_LEN (HASH_LEN / CHAR_BIT)   /*number of bits per row*/
+
+#define __DISTANCE_STR(X) #X
+#define _DISTANCE_STR(X) __DISTANCE_STR(X)
+#define DISTANCE_STR _DISTANCE_STR(DISTANCE_MIN)
+
+#define MAGIC_NUMBER 0x837EAF3A7F3A4DFDui64
+#define THREAD_ID (pthread_getw32threadid_np(pthread_self()))
+
+//-----------------------------------------------------------------------------
+// Globals
+//-----------------------------------------------------------------------------
+
+static const char SPINNER[4] = { '/', '-', '\\', '|' };
+static const double SQRT2 = 1.41421356237309504880168872420969807856967187537694;
+
+static size_t g_spinpos = 0;
+
+//-----------------------------------------------------------------------------
+// Utility Functions
+//-----------------------------------------------------------------------------
+
+#ifdef ENABLE_TRACE
+#define TRACE(X, ...) printf("[%04X] " X "\n", THREAD_ID, __VA_ARGS__)
+#else
+#define TRACE(X, ...) __noop()
+#endif
+
+static inline void print_row(uint8_t *const row_buffer)
+{
+       for (size_t w = 0; w < ROW_LEN; ++w)
+       {
+               printf("%02X", row_buffer[w]);
+       }
+       puts("");
+}
+
+static inline void flip_bit_at(uint8_t *const row_buffer, const size_t pos)
+{
+       row_buffer[pos >> 3] ^= ((uint8_t)(1U << (pos & 0x7)));
+}
+
+static inline void flip_rand_n(uint8_t *const row_buffer, msws_t rand, const uint32_t n)
+{
+       bool taken[HASH_LEN];
+       memset(&taken, 0, sizeof(bool) * HASH_LEN);
+       for (uint_fast32_t i = 0; i < n; ++i)
+       {
+               size_t next;
+               do
+               {
+                       next = msws_uint32_max(rand, HASH_LEN);
+               }
+               while (taken[next]);
+               flip_bit_at(row_buffer, next);
+               taken[next] = true;
+       }
+}
+
+static inline uint_fast32_t get_distance_rows(const uint8_t table[ROW_NUM][ROW_LEN], const uint_fast16_t index_1, const uint_fast16_t index_2)
+{
+       return hamming_distance(table[index_1], table[index_2], ROW_LEN);
+}
+
+#define ERROR_ACC(MAX,ACC) (((MAX) << 20U) | (ACC))
+static inline uint_fast32_t get_error_table(const uint8_t table[ROW_NUM][ROW_LEN], const uint_fast32_t limit)
+{
+       uint_fast32_t error_max = 0U, error_acc = 0U;
+       for (uint_fast16_t i = 0; i < ROW_NUM; i++)
+       {
+               for (uint_fast16_t j = i + 1U; j < ROW_NUM; j++)
+               {
+                       const uint_fast32_t dist = get_distance_rows(table, i, j);
+                       if (dist < DISTANCE_MIN)
+                       {
+                               const uint_fast32_t current = DISTANCE_MIN - dist;
+                               error_acc += current;
+                               if (current > error_max)
+                               {
+                                       error_max = current;
+                               }
+                               if (ERROR_ACC(error_max, error_acc) >= limit)
+                               {
+                                       return limit; /*early termination*/
+                               }
+                       }
+               }
+       }
+       return ERROR_ACC(error_max, error_acc);
+}
+
+static inline void copy_table(uint8_t dst[ROW_NUM][ROW_LEN], const uint8_t src[ROW_NUM][ROW_LEN])
+{
+       for (uint_fast16_t i = 0; i < ROW_NUM; i++)
+       {
+               memcpy(&dst[i][0], &src[i][0], sizeof(uint8_t) * ROW_LEN);
+       }
+}
+
+//static void dump_table(FILE *out)
+//{
+//     fputs("uint8_t MHASH_384_TABLE_XOR[UINT8_MAX+2][MHASH_384_LEN] =\n{\n", out);
+//     for (size_t i = 0; i < ROW_NUM; i++)
+//     {
+//             fputs("\t{ ", out);
+//             for (size_t j = 0; j < ROW_LEN; j++)
+//             {
+//                     if (j > 0)
+//                     {
+//                             fputc(',', out);
+//                     }
+//                     fprintf(out, "0x%02X", g_table[i][j]);
+//             }
+//             fprintf(out, " }%s /*%02X*/\n", (i != (ROW_NUM - 1)) ? "," : " ", (uint32_t)(i % 0x100));
+//     }
+//     fputs("};\n", out);
+//}
+
+//-----------------------------------------------------------------------------
+// Thread function
+//-----------------------------------------------------------------------------
+
+typedef struct
+{
+       uint8_t table[ROW_NUM][ROW_LEN];
+       sem_t *stop;
+       pthread_mutex_t *mutex;
+}
+thread_data_t;
+
+static void* thread_main(void *const param)
+{
+       thread_data_t *const data = (thread_data_t*)param;
+       msws_t rand;
+       bxmller_t bxmller;
+       uint8_t backup[ROW_LEN];
+       const uint_fast32_t error_initial = get_error_table(data->table, UINT_FAST32_MAX);
+       TRACE("Startup (Initail error: %08X)", error_initial);
+       while(error_initial)
+       {
+               msws_init(rand, make_seed());
+               gaussian_noise_init(&bxmller);
+               for (int_fast16_t round = 0; round < 29989; ++round)
+               {
+                       if (!(round & 0x3FF))
+                       {
+                               if (SEM_TRYWAIT(data->stop))
+                               {
+                                       return NULL;
+                               }
+                       }
+                       for (uint_fast16_t row_index = 0U; row_index < ROW_NUM; ++row_index)
+                       {
+                               memcpy(backup, data->table[row_index], sizeof(uint8_t) * ROW_LEN);
+                               for (uint_fast16_t rand_loop = 0; rand_loop < 99991U; ++rand_loop)
+                               {
+                                       msws_bytes(rand, data->table[row_index], ROW_LEN);
+                                       const uint_fast32_t error = get_error_table(data->table, error_initial);
+                                       if (error < error_initial)
+                                       {
+                                               TRACE("Improved by rand-init (%08X -> %08X)", error_initial, error);
+                                               goto success;
+                                       }
+                               }
+                               memcpy(data->table[row_index], backup, sizeof(uint8_t) * ROW_LEN);
+                       }
+               }
+               TRACE("Restarting");
+       }
+
+success:
+       if (get_error_table(data->table, UINT_FAST32_MAX) >= error_initial)
+       {
+               fprintf(stderr, "ERROR MISCOMPARE!\n");
+               abort();
+       }
+
+       MUTEX_LOCK(data->mutex);
+       if (SEM_TRYWAIT(data->stop))
+       {
+               MUTEX_UNLOCK(data->mutex);
+               return NULL;
+       }
+
+       SEM_POST(data->stop, THREAD_COUNT);
+       MUTEX_UNLOCK(data->mutex);
+       return data->table; /*success*/
+}
+
+static void* thread_spin(void *const param)
+{
+       unsigned long delay = 1U;
+       sem_t *const stop = ((sem_t*)param);
+       for (;;)
+       {
+               if (SEM_TRYWAIT(stop))
+               {
+                       printf("\b\b\b[!]");
+                       return NULL;
+               }
+               _sleep(delay);
+               if (delay >= 500)
+               {
+                       printf("\b\b\b[%c]", SPINNER[g_spinpos]);
+                       g_spinpos = (g_spinpos + 1) % 4;
+               }
+               else
+               {
+                       delay *= 2U;
+               }
+       }
+}
+
+//-----------------------------------------------------------------------------
+// Save / Load
+//-----------------------------------------------------------------------------
+
+static bool save_table_data(const uint8_t table[ROW_NUM][ROW_LEN], const wchar_t *const filename)
+{
+       wchar_t filename_temp[_MAX_PATH];
+       swprintf_s(filename_temp, _MAX_PATH, L"%s~%X", filename, make_seed());
+       FILE *const file = _wfopen(filename_temp, L"wb");
+       if (file)
+       {
+               bool success = true;
+               const uint64_t magic_number = MAGIC_NUMBER;
+               const uint32_t hash_len = HASH_LEN, distance_min = DISTANCE_MIN;
+               fwrite(&magic_number, sizeof(uint64_t), 1, file);
+               fwrite(&hash_len, sizeof(uint32_t), 1, file);
+               fwrite(&distance_min, sizeof(uint32_t), 1, file);
+               for (uint32_t i = 0; i < ROW_NUM; ++i)
+               {
+                       const uint32_t checksum = adler32(&table[i][0], ROW_LEN);
+                       fwrite(&checksum, sizeof(uint32_t), 1, file);
+                       fwrite(&table[i][0], sizeof(uint8_t), ROW_LEN, file);
+               }
+               if (ferror(file))
+               {
+                       printf("ERROR: Failed to write table data!\n");
+                       success = false;
+               }
+               fclose(file);
+               if (success)
+               {
+                       for (size_t i = 0; i < 42; ++i)
+                       {
+                               if (_wremove(filename))
+                               {
+                                       if (errno != ENOENT)
+                                       {
+                                               printf("ERROR: Failed to delete existing file!\n");
+                                               sched_yield();
+                                               continue;
+                                       }
+                               }
+                               break;
+                       }
+                       if (_wrename(filename_temp, filename))
+                       {
+                               printf("ERROR: Failed to rename temp file!\n");
+                               success = false;
+                       }
+               }
+               else
+               {
+                       _wremove(filename_temp);
+               }
+               return success;
+       }
+       else
+       {
+               printf("ERROR: Failed to open table file for writing!\n");
+               return false;
+       }
+}
+
+static bool load_table_data(uint8_t table[ROW_NUM][ROW_LEN], const wchar_t *const filename)
+{
+       FILE *const file = _wfopen(filename, L"rb");
+       if (file)
+       {
+               bool success = true;
+               uint64_t magic_number;
+               uint32_t hash_len, distance_min;
+               fread(&magic_number, sizeof(uint64_t), 1, file);
+               fread(&hash_len, sizeof(uint32_t), 1, file);
+               fread(&distance_min, sizeof(uint32_t), 1, file);
+               if (ferror(file) || feof(file))
+               {
+                       printf("ERROR: Failed to read the table header!\n");
+                       success = false;
+                       goto failed;
+               }
+               if (magic_number != MAGIC_NUMBER)
+               {
+                       printf("ERROR: Table file format could not be recognized!\n");
+                       success = false;
+                       goto failed;
+               }
+               if ((hash_len != HASH_LEN) || (distance_min != DISTANCE_MIN))
+               {
+                       printf("ERROR: Table properties are incompatibe with this instance!\n");
+                       success = false;
+                       goto failed;
+               }
+               for (uint_fast16_t i = 0; i < ROW_NUM; ++i)
+               {
+                       uint32_t checksum_expected;
+                       if ((fread(&checksum_expected, sizeof(uint32_t), 1, file) != 1) || (fread(&table[i][0], sizeof(uint8_t), ROW_LEN, file) != ROW_LEN))
+                       {
+                               printf("ERROR: Failed to read table data from file!\n");
+                               success = false;
+                               goto failed;
+                       }
+                       if (adler32(&table[i][0], ROW_LEN) != checksum_expected)
+                       {
+                               printf("ERROR: Table checksum does *not* match table contents!\n");
+                               success = false;
+                               goto failed;
+                       }
+                       for (uint_fast16_t j = 0; j < i; j++)
+                       {
+                               if (get_distance_rows(table, i, j) < DISTANCE_MIN)
+                               {
+                                       printf("ERROR: Table distance verification has failed!\n");
+                                       success = false;
+                                       goto failed;
+                               }
+                       }
+               }
+       failed:
+               fclose(file);
+               return success;
+       }
+       else
+       {
+               printf("ERROR: Failed to open table file for reading!\n");
+               return false;
+       }
+}
+
+//-----------------------------------------------------------------------------
+// MAIN
+//-----------------------------------------------------------------------------
+
+static uint8_t g_table[ROW_NUM][ROW_LEN];
+static thread_data_t thread_data[THREAD_COUNT];
+static pthread_t thread_id[THREAD_COUNT + 1];
+
+int wmain(int argc, wchar_t *argv[])
+{
+       sem_t stop_flag;
+       pthread_mutex_t stop_mutex;
+       FILE *file_out = NULL;
+
+       printf("MHash GenTableXOR V2 [%s]\n\n", __DATE__);
+       printf("HashLen: %d, Distance Min: %d, Threads: %d, MSVC: %u\n\n", HASH_LEN, DISTANCE_MIN, THREAD_COUNT, _MSC_FULL_VER);
+
+       if ((HASH_LEN % (8 * sizeof(uint32_t))) != 0)
+       {
+               crit_exit("FATAL: Hash length must be a multiple of 32 bits!");
+       }
+
+       if (argc < 2)
+       {
+               printf("Table file not specified!\n\n");
+               printf("Usage:\n");
+               printf("   GenTables_XOR.exe <table_file>\n\n");
+               return 1;
+       }
+
+       for (size_t i = 0; i < ROW_NUM; i++)
+       {
+               memset(&g_table[i][0], 0, sizeof(uint8_t) * ROW_LEN);
+       }
+
+       memset(&thread_id, 0, sizeof(pthread_t) * (THREAD_COUNT + 1));
+       memset(&thread_data, 0, sizeof(thread_data_t) * THREAD_COUNT);
+
+       SEM_INIT(&stop_flag);
+       MUTEX_INIT(&stop_mutex);
+
+       if (_waccess(argv[1], 4) == 0)
+       {
+               printf("Loading existing table data and proceeding...\n");
+               if (!load_table_data(g_table, argv[1]))
+               {
+                       return 1;
+               }
+       }
+       else
+       {
+               msws_t rand;
+               printf("Generating new initial random table...\n");
+               msws_init(rand, make_seed());
+               for (size_t i = 0; i < ROW_NUM; i++)
+               {
+                       msws_bytes(rand, &g_table[i][0], ROW_LEN);
+               }
+       }
+
+       uint_fast32_t error = get_error_table(g_table, UINT_FAST32_MAX);
+       while(error > 0)
+       {
+               char time_string[64];
+               printf("\aRemaining error: %08X [%c]", error, SPINNER[g_spinpos]);
+               g_spinpos = (g_spinpos + 1) % 4;
+
+               PTHREAD_CREATE(&thread_id[THREAD_COUNT], NULL, thread_spin, &stop_flag);
+               for (size_t t = 0; t < THREAD_COUNT; t++)
+               {
+                       thread_data[t].stop = &stop_flag;
+                       thread_data[t].mutex = &stop_mutex;
+                       copy_table(thread_data[t].table, g_table);
+                       PTHREAD_CREATE(&thread_id[t], NULL, thread_main, &thread_data[t]);
+                       PTHREAD_SET_PRIORITY(thread_id[t], -15);
+               }
+
+               for (size_t t = 0; t < THREAD_COUNT; t++)
+               {
+                       void *return_value = NULL;
+                       PTHREAD_JOIN(thread_id[t], &return_value);
+                       if (return_value)
+                       {
+                               copy_table(g_table, thread_data[t].table);
+                               error = get_error_table(g_table, UINT_FAST32_MAX);
+                       }
+               }
+
+               PTHREAD_JOIN(thread_id[THREAD_COUNT], NULL);
+               get_time_str(time_string, 64);
+               printf("\b\b\b[#] - %s\n", time_string);
+
+               if (!save_table_data(g_table, argv[1]))
+               {
+                       getchar(); /*failed to save current table data*/
+               }
+       }
+
+       printf("\n-----\n\n");
+
+       //dump_table(stdout);
+       if (fopen_s(&file_out, "table_XOR." DISTANCE_STR ".txt", "w") == 0)
+       {
+               //dump_table(file_out);
+               fclose(file_out);
+       }
+
+       printf("\n-----\n\n");
+
+       for (uint_fast16_t i = 0; i < ROW_NUM; i++)
+       {
+               for (uint_fast16_t j = 0; j < ROW_NUM; j++)
+               {
+                       if (i == j)
+                       {
+                               continue; //i==j
+                       }
+                       if (get_distance_rows(g_table, i, j) < DISTANCE_MIN)
+                       {
+                               crit_exit("FATAL: Table verification has failed!");
+                       }
+               }
+       }
+
+       SEM_DESTROY(&stop_flag);
+       MUTEX_DESTROY(&stop_mutex);
+
+       printf("COMPLETED.\n\n");
+       system("shutdown /s /t 180");
+       return getchar();
+}