0ad/source/lib/sysdep/win/winit.cpp
janwas db189468a9 (update-workspace required)
found another means of doing startup/shutdown that doesn't require
hooking and allows init callbacks to use CRT functions (avoiding
problems similar to the recent rash of pre-libc bugs). also, callback
registration no longer needs ugly #pragma syntax.

remove 'Detours' (evil and no longer needed)

This was SVN commit r5137.
2007-06-04 00:00:57 +00:00

66 lines
2 KiB
C++

/**
* =========================================================================
* File : winit.cpp
* Project : 0 A.D.
* Description : windows-specific module init and shutdown mechanism
* =========================================================================
*/
// license: GPL; see lib/license.txt
#include "precompiled.h"
#include "winit.h"
#include "win.h" // GetTickCount
// see http://blogs.msdn.com/larryosterman/archive/2004/09/27/234840.aspx
typedef LibError (*PfnLibErrorVoid)(void);
// pointers to start and end of function tables.
// notes:
// - COFF tosses out empty segments, so we have to put in one value
// (zero, because CallFunctionPointers has to ignore entries =0 anyway).
// - ASCII '$' and 'Z' come before resp. after '0'..'9', so use that to
// bound the section names.
__declspec(allocate("WINIT$I$")) PfnLibErrorVoid initBegin = 0;
__declspec(allocate("WINIT$IZ")) PfnLibErrorVoid initEnd = 0;
__declspec(allocate("WINIT$S$")) PfnLibErrorVoid shutdownBegin = 0;
__declspec(allocate("WINIT$SZ")) PfnLibErrorVoid shutdownEnd = 0;
// note: #pragma comment(linker, "/include") is not necessary since
// these are referenced below.
/**
* call into a range of function pointers.
* @param [begin, end): STL-style range
*
* note: pointers = 0 are ignored. this is because the above placeholders
* are initialized to 0 and because the range may be larger than
* expected due to COFF section padding (with zeroes).
**/
static void CallFunctionPointers(PfnLibErrorVoid* begin, PfnLibErrorVoid* end)
{
const DWORD t0 = GetTickCount();
for(PfnLibErrorVoid* ppfunc = begin; ppfunc < end; ppfunc++)
{
if(*ppfunc)
(*ppfunc)();
}
const DWORD t1 = GetTickCount();
debug_printf("WINIT| total elapsed time in callbacks %d ms (+-10)\n", t1-t0);
}
void winit_CallInitFunctions()
{
CallFunctionPointers(&initBegin, &initEnd);
}
void winit_CallShutdownFunctions()
{
CallFunctionPointers(&shutdownBegin, &shutdownEnd);
}