Current Stockfish on Windows 98?!
Published on the
Topics: retro, windows-9x
A few weeks ago I had this weird idea: would it be possible to make an up-to-date version of Stockfish run on Windows 98? There's an immediate problem with that: Stockfish is written in C++17, which is the still-current, at the time of writing, revision of the C++ language; surely there are no C++17 capable compilers that can still target something as old as Windows 98?
Disclaimer: in this article, by "Windows 98" I mean Windows 98 second edition with the main files of the unofficial service pack installed, which is a convenient bundle of official and unofficial patches. I don't care about Windows 98 first edition, nor the second edition without those updates installed. Note that KernelEx is not needed.
I'm fully aware this has no practical value, but it was a fun thing to do.
As it turns out, the GNU toolchain, comprised of GNU binutils and GCC, the GNU Compiler Collection, doesn't particularly care about all the different Windows versions and will happily output a binary that works on any 32-bit Windows.
The compiler itself, assembler and linker are only a part of a functional compiler package, though. There's one part we're missing: the runtime library, which provides the system-specific program initialization code that runs before the main
function. Since we're using GCC, the obvious choice is the mingw-w64 runtime. It also provides Windows API headers, bindings to (pretty much all the extant versions of) the Microsoft Visual C++ standard C library, and a POSIX threading API compatibility layer, winpthreads, which is required for C++ standard threading library support in GCC.
Fortunately for us, a minimal Windows program compiled with GCC with mingw-w64 appears to work fine when executed under Windows 98:
C++
#include <windows.h>
int WINAPI WinMain(HINSTANCE, HINSTANCE, char*, int)
{
MessageBoxA(nullptr, "Hello world!", "A test", MB_OK);
return 0;
}
So maybe all we have to do is to compile Stockfish and it will just work on Windows 98…?
Naïve attempt: don't do anything, just try it!
Alas, as can be expected, it's not that easy. If we don't do anything apart from just compiling Stockfish as-is with a stock GCC + mingw-w64 toolchain, trying to run the resulting binary on Windows 98 will just give us errors about unresolved imports.
There are three symbols the binary requires that are not present under Windows 98, all imported from kernel32.dll
: AddVectoredExceptionHandler
, RemoveVectoredExceptionHandler
and GetTickCount64
. Looking for these identifiers in Stockfish source code reveals nothing, so where are they coming from?
Looking at the disassembly, all three of these functions are used inside the aforementioned winpthreads library. Can we do anything about that?
Rebuilding winpthreads
None of those three functions are used for anything critical. The vectored exception handling APIs, introduced in Windows XP, are used only to allow threads to be named under a debugger. GetTickCount64
, added in Vista, is used for internal time tracking, but there exists a fallback path for older systems in the source code. The obvious solution would be to just patch them out and rebuild winpthreads:
Unified diff
diff --git a/mingw-w64-libraries/winpthreads/src/misc.c b/mingw-w64-libraries/winpthreads/src/misc.c
--- a/mingw-w64-libraries/winpthreads/src/misc.c
+++ b/mingw-w64-libraries/winpthreads/src/misc.c
@@ -55,7 +55,7 @@ unsigned long long _pthread_rel_time_in_ms(const struct timespec *ts)
static unsigned long long
_pthread_get_tick_count (long long *frequency)
{
-#if defined (_WIN32_WINNT) && (_WIN32_WINNT >= _WIN32_WINNT_VISTA)
+#if 0 && defined (_WIN32_WINNT) && (_WIN32_WINNT >= _WIN32_WINNT_VISTA)
(void) frequency; /* unused */
return GetTickCount64 ();
#else
diff --git a/mingw-w64-libraries/winpthreads/src/thread.c b/mingw-w64-libraries/winpthreads/src/thread.c
--- a/mingw-w64-libraries/winpthreads/src/thread.c
+++ b/mingw-w64-libraries/winpthreads/src/thread.c
@@ -57,7 +57,7 @@ static pthread_t idListNextId = 0;
#if !defined(_MSC_VER)
#define USE_VEH_FOR_MSC_SETTHREADNAME
#endif
-#if !WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
+#if 1 || !WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
/* forbidden RemoveVectoredExceptionHandler/AddVectoredExceptionHandler APIs */
#undef USE_VEH_FOR_MSC_SETTHREADNAME
#endif
@@ -109,7 +109,7 @@ SetThreadName (DWORD dwThreadID, LPCSTR szThreadName)
/* Without a debugger we *must* have an exception handler,
* otherwise raising an exception will crash the process.
*/
-#if WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
+#if 0 && WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
if ((!IsDebuggerPresent ()) && (SetThreadName_VEH_handle == NULL))
#else
if (!IsDebuggerPresent ())
After rebuilding winpthreads and re-linking Stockfish, the resulting binary now launches under Windows 98!
Deadlock on startup
Yes, the binary launches, but it doesn't work. It prints the version banner, but then hangs forever. Damn.
There are actually two issues that cause the hang. One is relatively easy to fix: winpthreads uses GetHandleInformation
for testing if the Win32 object handles are valid. This function exists on Windows 98 for limited NT compatibility, but it doesn't actually do anything, always returning 0 and setting the last error code to ERROR_CALL_NOT_IMPLEMENTED
. Since this is mostly a debugging feature, it can safely be removed with another trivial patch:
Unified diff
diff --git a/mingw-w64-libraries/winpthreads/src/misc.h b/mingw-w64-libraries/winpthreads/src/misc.h
--- a/mingw-w64-libraries/winpthreads/src/misc.h
+++ b/mingw-w64-libraries/winpthreads/src/misc.h
@@ -62,7 +62,7 @@ typedef long long LONGBAG;
typedef long LONGBAG;
#endif
-#if !WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
+#if 1 || !WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
#undef GetHandleInformation
#define GetHandleInformation(h,f) (1)
#endif
The other issue is much more severe: there is another function like GetHandleInformation
that "exists" on Windows 98, but is not actually implemented and always returns with an error. It's TryEnterCriticalSection
, used several times in the implementation of condition variables. The code is written under the assumption that TryEnterCriticalSection
actually works as described: if the critical section is not currently locked, it expects the call to take ownership of the critical section and succeed, but on Windows 98, this will never happen.
Critical sections are the Win32 name for regular user-mode mutexes. Win32 also offers kernel-mode mutexes, which are heavy and slow, as they require system calls for every operation, so they're usually only used for inter-process communication.
It's not possible to work around that problem by patching away TryEnterCriticalSection
calls without a significant re-architecture of winpthreads's condition variables. Clearly, a different approach is needed this time.
Implementing that one missing function ourselves
TryEnterCriticalSection
being present, but not actually implemented on Windows 98 is the only obstacle left, albeit a significant one. The algorithm itself is simple: find out where the atomic indicating whether the section is locked is, and use an operation like atomic compare-and-swap to change it, returning true if the section wasn't locked before. Unfortunately, the structure of critical sections is completely undocumented, and there aren't many good sources that are still alive on what critical sections looked like in that old operating system, so finding out where the atomic variable is requires some work.
From what I can gather from old debug symbols and looking at disassembly, the CRITICAL_SECTION
structure looks similar to this in Windows 98:
C++
struct critical_section {
std::uint8_t type; // Always 4.
struct critical_section_impl* impl;
// Padding for NT compatibility, where the structure is completely
// different.
std::uint32_t reserved[4];
};
static_assert(sizeof(critical_section) == 24);
That's not very useful, it's essentially just a pointer to something else. So what does critical_section_impl
look like? That's where our atomic actually is, together with a recursion counter, current owner and some uninteresting to us stuff:
C++
struct critical_section_impl {
std::uint8_t type;
int recursion_count;
void* owner_thread;
std::uint32_t reserved;
std::atomic<int> lock_count; // Starts at 1 when not owned, decreasing.
void* internal_pointers[3];
};
static_assert(sizeof(critical_section_impl) == 32);
There is another complication waiting for us here: the owner thread is represented as a pointer to its TDBX, which is the kernel-mode data structure representing a thread, internal to vwin32.vxd
, the process management part of the Windows 9x kernel.
Fortunately, the (undocumented) thread information block does contain a pointer to the thread's TDBX. Unfortunately, its location varies between system versions: it's at offset 0x50
from the start of the TIB in Windows 95 (according to Windows 95 System Programming Secrets) and 98, but at offset 0x80
in Windows Me (according to KernelEx sources). While I don't care about Windows 95 at all, Windows Me would be nice to support if only because it's the successor of Windows 98.
With all that in mind, we can implement our version of TryEnterCriticalSection
for Windows 98 and Me, using a C++ template to parameterize the code over the TDBX pointer offset:
C++
constexpr std::uint8_t critical_section_type = 4;
template <unsigned TdbxOffset>
void* get_current_tdbx()
{
const char* tib;
asm("mov {%%fs:0x18, %0|%0, fs:[0x18]}" : "=r"(tib));
return *reinterpret_cast<void* const*>(tib + TdbxOffset);
}
template <unsigned TdbxOffset>
bool try_enter_9x_impl(critical_section_impl* cs)
{
const auto current_tdbx = get_current_tdbx<TdbxOffset>();
int actual_lock_count = 1;
if(cs->lock_count.compare_exchange_strong(actual_lock_count, 0)) {
cs->owner_thread = current_tdbx;
++cs->recursion_count;
return true;
} else if(cs->owner_thread == current_tdbx) {
cs->lock_count.fetch_sub(1, std::memory_order_relaxed);
++cs->recursion_count;
return true;
} else {
return false;
}
}
template <unsigned TdbxOffset>
[[gnu::stdcall]] int try_enter_9x(CRITICAL_SECTION* cs)
{
const auto actual = reinterpret_cast<critical_section*>(cs);
if(actual->type != critical_section_type) {
[[unlikely]] RaiseException(EXCEPTION_ACCESS_VIOLATION, 0, 0, 0);
return 0;
}
return try_enter_9x_impl<TdbxOffset>(actual->impl);
}
Dynamic dispatch
We could implement TryEnterCriticalSection
by just always calling try_enter_9x
, but that would break compatibility with Windows NT. Having one binary that can run on anything from Windows 98 to Windows 10 is just cleaner in my opinion, so let's choose the implementation at runtime, based on the operating system version.
To do that, we can use the GetVersion
Windows API function, which has the nice guarantee of returning a value with the highest bit clear if and only if the operating system is some version of Windows NT, together with the usual LoadLibraryA
and GetProcAddress
calls to retrieve the address of the actual TryEnterCriticalSection
implementation on Windows NT:
C++
constexpr unsigned tdbx_offset_98 = 0x50;
constexpr unsigned tdbx_offset_me = 0x80;
constexpr std::uint32_t windows_9x_mask = 0x80000000;
constexpr std::uint32_t windows_me_version = 0xC0005A04;
[[gnu::stdcall]] int try_enter_dispatch(CRITICAL_SECTION* cs);
using try_enter_type = decltype(&TryEnterCriticalSection);
std::atomic<try_enter_type> implementation = &try_enter_dispatch;
[[gnu::stdcall]] int try_enter_dispatch(CRITICAL_SECTION* cs)
{
const auto system_version = GetVersion();
if((system_version & windows_9x_mask) == 0) {
const auto kernel32 = LoadLibraryA("kernel32.dll");
const auto impl = reinterpret_cast<try_enter_type>(
GetProcAddress(kernel32, "TryEnterCriticalSection"));
implementation.store(impl, std::memory_order_relaxed);
} else {
const auto impl = system_version == windows_me_version
? &try_enter_9x<tdbx_offset_me>
: &try_enter_9x<tdbx_offset_98>;
implementation.store(impl, std::memory_order_relaxed);
}
return implementation.load(std::memory_order_relaxed)(cs);
}
extern "C"
[[gnu::stdcall]] int TryEnterCriticalSection_compat(CRITICAL_SECTION* cs)
{
return implementation.load(std::memory_order_relaxed)(cs);
}
The complete, compilable source code of the fallback implementation of TryEnterCriticalSection
is available under the CC0 license.
All that's left now is to force winpthreads to use our function instead of the system one by adding a few lines to misc.h
and rebuilding the winpthreads static library:
C++
#ifndef TryEnterCriticalSection
int WINAPI TryEnterCriticalSection_compat(CRITICAL_SECTION* cs);
#define TryEnterCriticalSection TryEnterCriticalSection_compat
#endif
Now we can just build Stockfish using our modified winpthreads, remembering to link our fallback implementation, and…
Damn it, Stockfish
When I first attempted this ordeal, at the beginning of September, that was indeed everything that's needed to get Stockfish running on Windows 98. A recent change to Stockfish broke it however: now the binary will fail to run, because the import GetLargePageMinimum
is unresolved. Not a big deal, it's used in exactly one place in Stockfish code: large page allocation code. The call can be replaced with a constant 0 to allow Windows 98 to run it, at the cost of the binary no longer supporting large pages. A slight speed loss on modern versions of Windows, but the point here is to run it on Windows 98.
After this one last trivial change…
It finally works!
It really does! Of course, since it's just a chess engine, it's only a console program:
Since it's just an engine, we should be able to use any Universal Chess Interface-capable graphical program with it, and indeed, the Arena chess interface version 1.1, from 2004, has no problems with using this build of Stockfish as an engine:
Now I can watch Stockfish demolish the old engines shipped with a 16-year-old chess program, running under a 22-year-old operating system, on 18-year-old hardware.