mirror of
https://github.com/alliedmodders/hl2sdk.git
synced 2025-01-10 10:59:39 +08:00
1176 lines
36 KiB
C++
1176 lines
36 KiB
C++
//===== Copyright © 1996-2005, Valve Corporation, All rights reserved. ======//
|
|
//
|
|
// Purpose:
|
|
//
|
|
// $NoKeywords: $
|
|
//
|
|
//===========================================================================//
|
|
|
|
#ifndef PLATFORM_H
|
|
#define PLATFORM_H
|
|
|
|
#if defined( _X360 )
|
|
#define NO_STEAM
|
|
#define NO_VOICE
|
|
// for the 360, the ppc platform and the rtos are tightly coupled
|
|
// setup the 360 environment here !once! for much less leaf module include wackiness
|
|
// these are critical order and purposely appear *before* anything else
|
|
#define _XBOX
|
|
#include <xtl.h>
|
|
#include <xaudio.h>
|
|
#include <xbdm.h>
|
|
#include <Xgraphics.h>
|
|
#include <xui.h>
|
|
#include <pmcpbsetup.h>
|
|
#include <XMAHardwareAbstraction.h>
|
|
#undef _XBOX
|
|
#endif
|
|
|
|
#include "wchartypes.h"
|
|
#include "basetypes.h"
|
|
#include "tier0/valve_off.h"
|
|
|
|
#ifdef _WIN32
|
|
#pragma once
|
|
#endif
|
|
|
|
// feature enables
|
|
#define NEW_SOFTWARE_LIGHTING
|
|
|
|
#ifdef _LINUX
|
|
// need this for _alloca
|
|
#include <alloca.h>
|
|
#endif // _LINUX
|
|
|
|
#if defined __APPLE__
|
|
#include <stdlib.h>
|
|
#else
|
|
#include <malloc.h>
|
|
#endif
|
|
#ifdef _MSC_VER
|
|
#include <new.h>
|
|
#elif defined __GNUC__
|
|
#include <new>
|
|
#endif
|
|
|
|
// need this for memset
|
|
#include <string.h>
|
|
|
|
#include "tier0/valve_minmax_on.h" // GCC 4.2.2 headers screw up our min/max defs.
|
|
|
|
#ifdef _RETAIL
|
|
#define IsRetail() true
|
|
#else
|
|
#define IsRetail() false
|
|
#endif
|
|
|
|
#ifdef _DEBUG
|
|
#define IsRelease() false
|
|
#define IsDebug() true
|
|
#else
|
|
#define IsRelease() true
|
|
#define IsDebug() false
|
|
#endif
|
|
|
|
// Deprecating, infavor of IsX360() which will revert to IsXbox()
|
|
// after confidence of xbox 1 code flush
|
|
#define IsXbox() false
|
|
|
|
#ifdef _WIN32
|
|
#define IsLinux() false
|
|
#ifndef _X360
|
|
#define IsPC() true
|
|
#define IsConsole() false
|
|
#define IsX360() false
|
|
#define IsPS3() false
|
|
#define IS_WINDOWS_PC
|
|
#else
|
|
#ifndef _CONSOLE
|
|
#define _CONSOLE
|
|
#endif
|
|
#define IsPC() false
|
|
#define IsConsole() true
|
|
#define IsX360() true
|
|
#define IsPS3() false
|
|
#endif
|
|
#elif defined(_LINUX)
|
|
#define IsPC() true
|
|
#define IsConsole() false
|
|
#define IsX360() false
|
|
#define IsPS3() false
|
|
#define IsLinux() true
|
|
#else
|
|
#error
|
|
#endif
|
|
|
|
typedef unsigned char uint8;
|
|
typedef signed char int8;
|
|
|
|
#ifdef __x86_64__
|
|
#define X64BITS
|
|
#endif
|
|
|
|
#if defined( _WIN32 )
|
|
|
|
typedef __int16 int16;
|
|
typedef unsigned __int16 uint16;
|
|
typedef __int32 int32;
|
|
typedef unsigned __int32 uint32;
|
|
typedef __int64 int64;
|
|
typedef unsigned __int64 uint64;
|
|
|
|
#ifdef X64BITS
|
|
typedef __int64 intp; // intp is an integer that can accomodate a pointer
|
|
typedef unsigned __int64 uintp; // (ie, sizeof(intp) >= sizeof(int) && sizeof(intp) >= sizeof(void *)
|
|
#else
|
|
typedef __int32 intp;
|
|
typedef unsigned __int32 uintp;
|
|
#endif
|
|
|
|
#if defined( _X360 )
|
|
#ifdef __m128
|
|
#undef __m128
|
|
#endif
|
|
#define __m128 __vector4
|
|
#endif
|
|
|
|
#else // _WIN32
|
|
|
|
typedef short int16;
|
|
typedef unsigned short uint16;
|
|
typedef int int32;
|
|
typedef unsigned int uint32;
|
|
typedef long long int64;
|
|
typedef unsigned long long uint64;
|
|
#ifdef X64BITS
|
|
typedef long long intp;
|
|
typedef unsigned long long uintp;
|
|
#else
|
|
typedef int intp;
|
|
typedef unsigned int uintp;
|
|
#endif
|
|
|
|
#endif // else _WIN32
|
|
|
|
|
|
typedef float float32;
|
|
typedef double float64;
|
|
|
|
// for when we don't care about how many bits we use
|
|
typedef unsigned int uint;
|
|
|
|
// This can be used to ensure the size of pointers to members when declaring
|
|
// a pointer type for a class that has only been forward declared
|
|
#ifdef _MSC_VER
|
|
#define SINGLE_INHERITANCE __single_inheritance
|
|
#define MULTIPLE_INHERITANCE __multiple_inheritance
|
|
#else
|
|
#define SINGLE_INHERITANCE
|
|
#define MULTIPLE_INHERITANCE
|
|
#endif
|
|
|
|
#ifdef _MSC_VER
|
|
#define NO_VTABLE __declspec( novtable )
|
|
#else
|
|
#define NO_VTABLE
|
|
#endif
|
|
|
|
// This can be used to declare an abstract (interface only) class.
|
|
// Classes marked abstract should not be instantiated. If they are, and access violation will occur.
|
|
//
|
|
// Example of use:
|
|
//
|
|
// abstract_class CFoo
|
|
// {
|
|
// ...
|
|
// }
|
|
//
|
|
// MSDN __declspec(novtable) documentation: http://msdn.microsoft.com/library/default.asp?url=/library/en-us/vclang/html/_langref_novtable.asp
|
|
//
|
|
// Note: NJS: This is not enabled for regular PC, due to not knowing the implications of exporting a class with no no vtable.
|
|
// It's probable that this shouldn't be an issue, but an experiment should be done to verify this.
|
|
//
|
|
#ifndef _X360
|
|
#define abstract_class class
|
|
#else
|
|
#define abstract_class class NO_VTABLE
|
|
#endif
|
|
|
|
/*
|
|
FIXME: Enable this when we no longer fear change =)
|
|
|
|
// need these for the limits
|
|
#include <limits.h>
|
|
#include <float.h>
|
|
|
|
// Maximum and minimum representable values
|
|
#define INT8_MAX SCHAR_MAX
|
|
#define INT16_MAX SHRT_MAX
|
|
#define INT32_MAX LONG_MAX
|
|
#define INT64_MAX (((int64)~0) >> 1)
|
|
|
|
#define INT8_MIN SCHAR_MIN
|
|
#define INT16_MIN SHRT_MIN
|
|
#define INT32_MIN LONG_MIN
|
|
#define INT64_MIN (((int64)1) << 63)
|
|
|
|
#define UINT8_MAX ((uint8)~0)
|
|
#define UINT16_MAX ((uint16)~0)
|
|
#define UINT32_MAX ((uint32)~0)
|
|
#define UINT64_MAX ((uint64)~0)
|
|
|
|
#define UINT8_MIN 0
|
|
#define UINT16_MIN 0
|
|
#define UINT32_MIN 0
|
|
#define UINT64_MIN 0
|
|
|
|
#ifndef UINT_MIN
|
|
#define UINT_MIN UINT32_MIN
|
|
#endif
|
|
|
|
#define FLOAT32_MAX FLT_MAX
|
|
#define FLOAT64_MAX DBL_MAX
|
|
|
|
#define FLOAT32_MIN FLT_MIN
|
|
#define FLOAT64_MIN DBL_MIN
|
|
*/
|
|
|
|
// portability / compiler settings
|
|
#if defined(_WIN32) && !defined(WINDED)
|
|
|
|
#if defined(_M_IX86)
|
|
#define __i386__ 1
|
|
#endif
|
|
|
|
#elif _LINUX
|
|
typedef unsigned int DWORD;
|
|
typedef unsigned short WORD;
|
|
typedef void * HINSTANCE;
|
|
#define _MAX_PATH PATH_MAX
|
|
#endif // defined(_WIN32) && !defined(WINDED)
|
|
|
|
|
|
// Defines MAX_PATH
|
|
#ifndef MAX_PATH
|
|
#define MAX_PATH 260
|
|
#endif
|
|
|
|
#define ALIGN_VALUE( val, alignment ) ( ( val + alignment - 1 ) & ~( alignment - 1 ) ) // need macro for constant expression
|
|
|
|
// Used to step into the debugger
|
|
#if defined( _WIN32 ) && !defined( _X360 )
|
|
#define DebuggerBreak() __asm { int 3 }
|
|
#elif defined( _X360 )
|
|
#define DebuggerBreak() DebugBreak()
|
|
#else
|
|
#define DebuggerBreak() {}
|
|
#endif
|
|
#define DebuggerBreakIfDebugging() if ( !Plat_IsInDebugSession() ) ; else DebuggerBreak()
|
|
|
|
// C functions for external declarations that call the appropriate C++ methods
|
|
#ifndef EXPORT
|
|
#ifdef _WIN32
|
|
#define EXPORT _declspec( dllexport )
|
|
#else
|
|
#define EXPORT /* */
|
|
#endif
|
|
#endif
|
|
|
|
#if defined __i386__ && !defined __linux__
|
|
#define id386 1
|
|
#else
|
|
#define id386 0
|
|
#endif // __i386__
|
|
|
|
// decls for aligning data
|
|
#ifdef _WIN32
|
|
#define DECL_ALIGN(x) __declspec(align(x))
|
|
|
|
#elif _LINUX
|
|
#define DECL_ALIGN(x) __attribute__((aligned(x)))
|
|
#else
|
|
#define DECL_ALIGN(x) /* */
|
|
#endif
|
|
|
|
#define ALIGN8 DECL_ALIGN(8)
|
|
#define ALIGN16 DECL_ALIGN(16)
|
|
#define ALIGN32 DECL_ALIGN(32)
|
|
#define ALIGN128 DECL_ALIGN(128)
|
|
|
|
|
|
// Linux had a few areas where it didn't construct objects in the same order that Windows does.
|
|
// So when CVProfile::CVProfile() would access g_pMemAlloc, it would crash because the allocator wasn't initalized yet.
|
|
#ifdef _LINUX
|
|
#define CONSTRUCT_EARLY __attribute__((init_priority(101)))
|
|
#else
|
|
#define CONSTRUCT_EARLY
|
|
#endif
|
|
|
|
#ifdef _WIN32
|
|
#define SELECTANY __declspec(selectany)
|
|
#elif _LINUX
|
|
#define SELECTANY __attribute__((weak))
|
|
#else
|
|
#define SELECTANY static
|
|
#endif
|
|
|
|
#if defined( _WIN32 )
|
|
|
|
// Used for dll exporting and importing
|
|
#define DLL_EXPORT extern "C" __declspec( dllexport )
|
|
#define DLL_IMPORT extern "C" __declspec( dllimport )
|
|
|
|
// Can't use extern "C" when DLL exporting a class
|
|
#define DLL_CLASS_EXPORT __declspec( dllexport )
|
|
#define DLL_CLASS_IMPORT __declspec( dllimport )
|
|
|
|
// Can't use extern "C" when DLL exporting a global
|
|
#define DLL_GLOBAL_EXPORT extern __declspec( dllexport )
|
|
#define DLL_GLOBAL_IMPORT extern __declspec( dllimport )
|
|
|
|
#elif defined _LINUX
|
|
// Used for dll exporting and importing
|
|
#define DLL_EXPORT extern "C" __attribute__ ((visibility("default")))
|
|
#define DLL_IMPORT extern "C"
|
|
|
|
// Can't use extern "C" when DLL exporting a class
|
|
#define DLL_CLASS_EXPORT __attribute__ ((visibility("default")))
|
|
#define DLL_CLASS_IMPORT
|
|
|
|
// Can't use extern "C" when DLL exporting a global
|
|
#define DLL_GLOBAL_EXPORT extern __attribute ((visibility("default")))
|
|
#define DLL_GLOBAL_IMPORT extern
|
|
|
|
#else
|
|
#error "Unsupported Platform."
|
|
#endif
|
|
|
|
// Used for standard calling conventions
|
|
#if defined( _WIN32 ) && !defined( _X360 )
|
|
#define STDCALL __stdcall
|
|
#define FASTCALL __fastcall
|
|
#define FORCEINLINE __forceinline
|
|
// GCC 3.4.1 has a bug in supporting forced inline of templated functions
|
|
// this macro lets us not force inlining in that case
|
|
#define FORCEINLINE_TEMPLATE __forceinline
|
|
#elif defined( _X360 )
|
|
#define STDCALL __stdcall
|
|
#ifdef FORCEINLINE
|
|
#undef FORCEINLINE
|
|
#endif
|
|
#define FORCEINLINE __forceinline
|
|
#define FORCEINLINE_TEMPLATE __forceinline
|
|
#else
|
|
#define STDCALL
|
|
#define FASTCALL
|
|
#ifdef _LINUX_DEBUGGABLE
|
|
#define FORCEINLINE
|
|
#else
|
|
#define FORCEINLINE inline
|
|
#endif
|
|
// GCC 3.4.1 has a bug in supporting forced inline of templated functions
|
|
// this macro lets us not force inlining in that case
|
|
#define FORCEINLINE_TEMPLATE inline
|
|
#define __stdcall __attribute__ ((__stdcall__))
|
|
#endif
|
|
|
|
// Force a function call site -not- to inlined. (useful for profiling)
|
|
#define DONT_INLINE(a) (((int)(a)+1)?(a):(a))
|
|
|
|
// Pass hints to the compiler to prevent it from generating unnessecary / stupid code
|
|
// in certain situations. Several compilers other than MSVC also have an equivilent
|
|
// construct.
|
|
//
|
|
// Essentially the 'Hint' is that the condition specified is assumed to be true at
|
|
// that point in the compilation. If '0' is passed, then the compiler assumes that
|
|
// any subsequent code in the same 'basic block' is unreachable, and thus usually
|
|
// removed.
|
|
#ifdef _MSC_VER
|
|
#define HINT(THE_HINT) __assume((THE_HINT))
|
|
#else
|
|
#define HINT(THE_HINT) 0
|
|
#endif
|
|
|
|
// Marks the codepath from here until the next branch entry point as unreachable,
|
|
// and asserts if any attempt is made to execute it.
|
|
#define UNREACHABLE() { Assert(0); HINT(0); }
|
|
|
|
// In cases where no default is present or appropriate, this causes MSVC to generate
|
|
// as little code as possible, and throw an assertion in debug.
|
|
#define NO_DEFAULT default: UNREACHABLE();
|
|
|
|
#ifdef _WIN32
|
|
// Alloca defined for this platform
|
|
#define stackalloc( _size ) _alloca( ALIGN_VALUE( _size, 16 ) )
|
|
#define stackfree( _p )
|
|
#elif _LINUX
|
|
// Alloca defined for this platform
|
|
#define stackalloc( _size ) _alloca( ALIGN_VALUE( _size, 16 ) )
|
|
#define stackfree( _p )
|
|
#endif
|
|
|
|
#ifdef _WIN32
|
|
#define RESTRICT __restrict
|
|
#define RESTRICT_FUNC __declspec(restrict)
|
|
#else
|
|
#define RESTRICT
|
|
#define RESTRICT_FUNC
|
|
#endif
|
|
|
|
|
|
#ifdef _WIN32
|
|
// Remove warnings from warning level 4.
|
|
#pragma warning(disable : 4514) // warning C4514: 'acosl' : unreferenced inline function has been removed
|
|
#pragma warning(disable : 4100) // warning C4100: 'hwnd' : unreferenced formal parameter
|
|
#pragma warning(disable : 4127) // warning C4127: conditional expression is constant
|
|
#pragma warning(disable : 4512) // warning C4512: 'InFileRIFF' : assignment operator could not be generated
|
|
#pragma warning(disable : 4611) // warning C4611: interaction between '_setjmp' and C++ object destruction is non-portable
|
|
#pragma warning(disable : 4710) // warning C4710: function 'x' not inlined
|
|
#pragma warning(disable : 4702) // warning C4702: unreachable code
|
|
#pragma warning(disable : 4505) // unreferenced local function has been removed
|
|
#pragma warning(disable : 4239) // nonstandard extension used : 'argument' ( conversion from class Vector to class Vector& )
|
|
#pragma warning(disable : 4097) // typedef-name 'BaseClass' used as synonym for class-name 'CFlexCycler::CBaseFlex'
|
|
#pragma warning(disable : 4324) // Padding was added at the end of a structure
|
|
#pragma warning(disable : 4244) // type conversion warning.
|
|
#pragma warning(disable : 4305) // truncation from 'const double ' to 'float '
|
|
#pragma warning(disable : 4786) // Disable warnings about long symbol names
|
|
#pragma warning(disable : 4250) // 'X' : inherits 'Y::Z' via dominance
|
|
#pragma warning(disable : 4201) // nonstandard extension used : nameless struct/union
|
|
|
|
#if _MSC_VER >= 1300
|
|
#pragma warning(disable : 4511) // Disable warnings about private copy constructors
|
|
#pragma warning(disable : 4121) // warning C4121: 'symbol' : alignment of a member was sensitive to packing
|
|
#pragma warning(disable : 4530) // warning C4530: C++ exception handler used, but unwind semantics are not enabled. Specify /EHsc (disabled due to std headers having exception syntax)
|
|
#endif
|
|
|
|
#if _MSC_VER >= 1400
|
|
#pragma warning(disable : 4996) // functions declared deprecated
|
|
#endif
|
|
|
|
|
|
#endif
|
|
|
|
// When we port to 64 bit, we'll have to resolve the int, ptr vs size_t 32/64 bit problems...
|
|
#if !defined( _WIN64 ) && defined _MSC_VER
|
|
#pragma warning( disable : 4267 ) // conversion from 'size_t' to 'int', possible loss of data
|
|
#pragma warning( disable : 4311 ) // pointer truncation from 'char *' to 'int'
|
|
#pragma warning( disable : 4312 ) // conversion from 'unsigned int' to 'memhandle_t' of greater size
|
|
#endif
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// fsel
|
|
//-----------------------------------------------------------------------------
|
|
#ifndef _X360
|
|
|
|
static FORCEINLINE float fsel(float fComparand, float fValGE, float fLT)
|
|
{
|
|
return fComparand >= 0 ? fValGE : fLT;
|
|
}
|
|
static FORCEINLINE double fsel(double fComparand, double fValGE, double fLT)
|
|
{
|
|
return fComparand >= 0 ? fValGE : fLT;
|
|
}
|
|
|
|
#else
|
|
|
|
// __fsel(double fComparand, double fValGE, double fLT) == fComparand >= 0 ? fValGE : fLT
|
|
// this is much faster than if ( aFloat > 0 ) { x = .. }
|
|
#define fsel __fsel
|
|
|
|
#endif
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// FP exception handling
|
|
//-----------------------------------------------------------------------------
|
|
//#define CHECK_FLOAT_EXCEPTIONS 1
|
|
|
|
#if !defined( _X360 )
|
|
#if defined( _MSC_VER )
|
|
|
|
inline void SetupFPUControlWordForceExceptions()
|
|
{
|
|
// use local to get and store control word
|
|
uint16 tmpCtrlW;
|
|
__asm
|
|
{
|
|
fnclex /* clear all current exceptions */
|
|
fnstcw word ptr [tmpCtrlW] /* get current control word */
|
|
and [tmpCtrlW], 0FCC0h /* Keep infinity control + rounding control */
|
|
or [tmpCtrlW], 0230h /* set to 53-bit, mask only inexact, underflow */
|
|
fldcw word ptr [tmpCtrlW] /* put new control word in FPU */
|
|
}
|
|
}
|
|
|
|
#ifdef CHECK_FLOAT_EXCEPTIONS
|
|
|
|
inline void SetupFPUControlWord()
|
|
{
|
|
SetupFPUControlWordForceExceptions();
|
|
}
|
|
|
|
#else
|
|
|
|
inline void SetupFPUControlWord()
|
|
{
|
|
// use local to get and store control word
|
|
uint16 tmpCtrlW;
|
|
__asm
|
|
{
|
|
fnstcw word ptr [tmpCtrlW] /* get current control word */
|
|
and [tmpCtrlW], 0FCC0h /* Keep infinity control + rounding control */
|
|
or [tmpCtrlW], 023Fh /* set to 53-bit, mask only inexact, underflow */
|
|
fldcw word ptr [tmpCtrlW] /* put new control word in FPU */
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
inline void SetupFPUControlWord()
|
|
{
|
|
__volatile unsigned short int __cw;
|
|
__asm __volatile ("fnstcw %0" : "=m" (__cw));
|
|
__cw = __cw & 0x0FCC0; // keep infinity control, keep rounding mode
|
|
__cw = __cw | 0x023F; // set 53-bit, no exceptions
|
|
__asm __volatile ("fldcw %0" : : "m" (__cw));
|
|
}
|
|
|
|
#endif // _MSC_VER
|
|
|
|
#else
|
|
|
|
#ifdef _DEBUG
|
|
FORCEINLINE bool IsFPUControlWordSet()
|
|
{
|
|
float f = 0.996f;
|
|
union
|
|
{
|
|
double flResult;
|
|
int pResult[2];
|
|
};
|
|
flResult = __fctiw( f );
|
|
return ( pResult[1] == 1 );
|
|
}
|
|
#endif
|
|
|
|
inline void SetupFPUControlWord()
|
|
{
|
|
// Set round-to-nearest in FPSCR
|
|
// (cannot assemble, must use op-code form)
|
|
__emit( 0xFF80010C ); // mtfsfi 7,0
|
|
|
|
// Favour compatibility over speed (make sure the VPU set to Java-compliant mode)
|
|
// NOTE: the VPU *always* uses round-to-nearest
|
|
__vector4 a = { 0.0f, 0.0f, 0.0f, 0.0f };
|
|
a; // Avoid compiler warning
|
|
__asm
|
|
{
|
|
mtvscr a; // Clear the Vector Status & Control Register to zero
|
|
}
|
|
}
|
|
|
|
#endif // _X360
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Standard functions for handling endian-ness
|
|
//-----------------------------------------------------------------------------
|
|
|
|
//-------------------------------------
|
|
// Basic swaps
|
|
//-------------------------------------
|
|
|
|
template <typename T>
|
|
inline T WordSwapC( T w )
|
|
{
|
|
uint16 temp;
|
|
|
|
temp = ((*((uint16 *)&w) & 0xff00) >> 8);
|
|
temp |= ((*((uint16 *)&w) & 0x00ff) << 8);
|
|
|
|
return *((T*)&temp);
|
|
}
|
|
|
|
template <typename T>
|
|
inline T DWordSwapC( T dw )
|
|
{
|
|
uint32 temp;
|
|
|
|
temp = *((uint32 *)&dw) >> 24;
|
|
temp |= ((*((uint32 *)&dw) & 0x00FF0000) >> 8);
|
|
temp |= ((*((uint32 *)&dw) & 0x0000FF00) << 8);
|
|
temp |= ((*((uint32 *)&dw) & 0x000000FF) << 24);
|
|
|
|
return *((T*)&temp);
|
|
}
|
|
|
|
//-------------------------------------
|
|
// Fast swaps
|
|
//-------------------------------------
|
|
|
|
#if defined( _X360 )
|
|
|
|
#define WordSwap WordSwap360Intr
|
|
#define DWordSwap DWordSwap360Intr
|
|
|
|
template <typename T>
|
|
inline T WordSwap360Intr( T w )
|
|
{
|
|
T output;
|
|
__storeshortbytereverse( w, 0, &output );
|
|
return output;
|
|
}
|
|
|
|
template <typename T>
|
|
inline T DWordSwap360Intr( T dw )
|
|
{
|
|
T output;
|
|
__storewordbytereverse( dw, 0, &output );
|
|
return output;
|
|
}
|
|
|
|
#elif defined( _MSC_VER )
|
|
|
|
#define WordSwap WordSwapAsm
|
|
#define DWordSwap DWordSwapAsm
|
|
|
|
#pragma warning(push)
|
|
#pragma warning (disable:4035) // no return value
|
|
|
|
template <typename T>
|
|
inline T WordSwapAsm( T w )
|
|
{
|
|
__asm
|
|
{
|
|
mov ax, w
|
|
xchg al, ah
|
|
}
|
|
}
|
|
|
|
template <typename T>
|
|
inline T DWordSwapAsm( T dw )
|
|
{
|
|
__asm
|
|
{
|
|
mov eax, dw
|
|
bswap eax
|
|
}
|
|
}
|
|
|
|
#pragma warning(pop)
|
|
|
|
#else
|
|
|
|
#define WordSwap WordSwapC
|
|
#define DWordSwap DWordSwapC
|
|
|
|
#endif
|
|
|
|
//-------------------------------------
|
|
// The typically used methods.
|
|
//-------------------------------------
|
|
|
|
#if defined(__i386__) && !defined LITTLE_ENDIAN
|
|
#define LITTLE_ENDIAN 1
|
|
#endif
|
|
|
|
#if (defined( _SGI_SOURCE ) || defined( _X360 )) && !defined BIG_ENDIAN
|
|
#define BIG_ENDIAN 1
|
|
#endif
|
|
|
|
// If a swapped float passes through the fpu, the bytes may get changed.
|
|
// Prevent this by swapping floats as DWORDs.
|
|
#define SafeSwapFloat( pOut, pIn ) (*((uint*)pOut) = DWordSwap( *((uint*)pIn) ))
|
|
|
|
#if defined(LITTLE_ENDIAN)
|
|
|
|
#define BigShort( val ) WordSwap( val )
|
|
#define BigWord( val ) WordSwap( val )
|
|
#define BigLong( val ) DWordSwap( val )
|
|
#define BigDWord( val ) DWordSwap( val )
|
|
#define LittleShort( val ) ( val )
|
|
#define LittleWord( val ) ( val )
|
|
#define LittleLong( val ) ( val )
|
|
#define LittleDWord( val ) ( val )
|
|
#define SwapShort( val ) BigShort( val )
|
|
#define SwapWord( val ) BigWord( val )
|
|
#define SwapLong( val ) BigLong( val )
|
|
#define SwapDWord( val ) BigDWord( val )
|
|
|
|
// Pass floats by pointer for swapping to avoid truncation in the fpu
|
|
#define BigFloat( pOut, pIn ) SafeSwapFloat( pOut, pIn )
|
|
#define LittleFloat( pOut, pIn ) ( *pOut = *pIn )
|
|
#define SwapFloat( pOut, pIn ) BigFloat( pOut, pIn )
|
|
|
|
#elif defined(BIG_ENDIAN)
|
|
|
|
#define BigShort( val ) ( val )
|
|
#define BigWord( val ) ( val )
|
|
#define BigLong( val ) ( val )
|
|
#define BigDWord( val ) ( val )
|
|
#define LittleShort( val ) WordSwap( val )
|
|
#define LittleWord( val ) WordSwap( val )
|
|
#define LittleLong( val ) DWordSwap( val )
|
|
#define LittleDWord( val ) DWordSwap( val )
|
|
#define SwapShort( val ) LittleShort( val )
|
|
#define SwapWord( val ) LittleWord( val )
|
|
#define SwapLong( val ) LittleLong( val )
|
|
#define SwapDWord( val ) LittleDWord( val )
|
|
|
|
// Pass floats by pointer for swapping to avoid truncation in the fpu
|
|
#define BigFloat( pOut, pIn ) ( *pOut = *pIn )
|
|
#define LittleFloat( pOut, pIn ) SafeSwapFloat( pOut, pIn )
|
|
#define SwapFloat( pOut, pIn ) LittleFloat( pOut, pIn )
|
|
|
|
#else
|
|
|
|
// @Note (toml 05-02-02): this technique expects the compiler to
|
|
// optimize the expression and eliminate the other path. On any new
|
|
// platform/compiler this should be tested.
|
|
inline short BigShort( short val ) { int test = 1; return ( *(char *)&test == 1 ) ? WordSwap( val ) : val; }
|
|
inline uint16 BigWord( uint16 val ) { int test = 1; return ( *(char *)&test == 1 ) ? WordSwap( val ) : val; }
|
|
inline long BigLong( long val ) { int test = 1; return ( *(char *)&test == 1 ) ? DWordSwap( val ) : val; }
|
|
inline uint32 BigDWord( uint32 val ) { int test = 1; return ( *(char *)&test == 1 ) ? DWordSwap( val ) : val; }
|
|
inline short LittleShort( short val ) { int test = 1; return ( *(char *)&test == 1 ) ? val : WordSwap( val ); }
|
|
inline uint16 LittleWord( uint16 val ) { int test = 1; return ( *(char *)&test == 1 ) ? val : WordSwap( val ); }
|
|
inline long LittleLong( long val ) { int test = 1; return ( *(char *)&test == 1 ) ? val : DWordSwap( val ); }
|
|
inline uint32 LittleDWord( uint32 val ) { int test = 1; return ( *(char *)&test == 1 ) ? val : DWordSwap( val ); }
|
|
inline short SwapShort( short val ) { return WordSwap( val ); }
|
|
inline uint16 SwapWord( uint16 val ) { return WordSwap( val ); }
|
|
inline long SwapLong( long val ) { return DWordSwap( val ); }
|
|
inline uint32 SwapDWord( uint32 val ) { return DWordSwap( val ); }
|
|
|
|
// Pass floats by pointer for swapping to avoid truncation in the fpu
|
|
inline void BigFloat( float *pOut, const float *pIn ) { int test = 1; ( *(char *)&test == 1 ) ? SafeSwapFloat( pOut, pIn ) : ( *pOut = *pIn ); }
|
|
inline void LittleFloat( float *pOut, const float *pIn ) { int test = 1; ( *(char *)&test == 1 ) ? ( *pOut = *pIn ) : SafeSwapFloat( pOut, pIn ); }
|
|
inline void SwapFloat( float *pOut, const float *pIn ) { SafeSwapFloat( pOut, pIn ); }
|
|
|
|
#endif
|
|
|
|
#if _X360
|
|
inline unsigned long LoadLittleDWord( unsigned long *base, unsigned int dwordIndex )
|
|
{
|
|
return __loadwordbytereverse( dwordIndex<<2, base );
|
|
}
|
|
|
|
inline void StoreLittleDWord( unsigned long *base, unsigned int dwordIndex, unsigned long dword )
|
|
{
|
|
__storewordbytereverse( dword, dwordIndex<<2, base );
|
|
}
|
|
#else
|
|
inline unsigned long LoadLittleDWord( unsigned long *base, unsigned int dwordIndex )
|
|
{
|
|
return LittleDWord( base[dwordIndex] );
|
|
}
|
|
|
|
inline void StoreLittleDWord( unsigned long *base, unsigned int dwordIndex, unsigned long dword )
|
|
{
|
|
base[dwordIndex] = LittleDWord(dword);
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifndef STATIC_TIER0
|
|
|
|
#ifdef TIER0_DLL_EXPORT
|
|
#define PLATFORM_INTERFACE DLL_EXPORT
|
|
#define PLATFORM_OVERLOAD DLL_GLOBAL_EXPORT
|
|
#else
|
|
#define PLATFORM_INTERFACE DLL_IMPORT
|
|
#define PLATFORM_OVERLOAD DLL_GLOBAL_IMPORT
|
|
#endif
|
|
|
|
#else // BUILD_AS_DLL
|
|
|
|
#define PLATFORM_INTERFACE extern
|
|
#define PLATFORM_OVERLOAD
|
|
|
|
#endif // BUILD_AS_DLL
|
|
|
|
|
|
// When in benchmark mode, the timer returns a simple incremented value each time you call it.
|
|
//
|
|
// It should not be changed after startup unless you really know what you're doing. The only place
|
|
// that should do this is the benchmark code itself so it can output a legit duration.
|
|
PLATFORM_INTERFACE void Plat_SetBenchmarkMode( bool bBenchmarkMode );
|
|
PLATFORM_INTERFACE bool Plat_IsInBenchmarkMode();
|
|
|
|
|
|
PLATFORM_INTERFACE double Plat_FloatTime(); // Returns time in seconds since the module was loaded.
|
|
PLATFORM_INTERFACE unsigned long Plat_MSTime(); // Time in milliseconds.
|
|
|
|
// b/w compatibility
|
|
#define Sys_FloatTime Plat_FloatTime
|
|
|
|
|
|
// Processor Information:
|
|
struct CPUInformation
|
|
{
|
|
int m_Size; // Size of this structure, for forward compatability.
|
|
|
|
bool m_bRDTSC : 1, // Is RDTSC supported?
|
|
m_bCMOV : 1, // Is CMOV supported?
|
|
m_bFCMOV : 1, // Is FCMOV supported?
|
|
m_bSSE : 1, // Is SSE supported?
|
|
m_bSSE2 : 1, // Is SSE2 Supported?
|
|
m_b3DNow : 1, // Is 3DNow! Supported?
|
|
m_bMMX : 1, // Is MMX supported?
|
|
m_bHT : 1; // Is HyperThreading supported?
|
|
|
|
uint8 m_nLogicalProcessors; // Number op logical processors.
|
|
uint8 m_nPhysicalProcessors; // Number of physical processors
|
|
|
|
int64 m_Speed; // In cycles per second.
|
|
|
|
tchar* m_szProcessorID; // Processor vendor Identification.
|
|
};
|
|
|
|
PLATFORM_INTERFACE const CPUInformation& GetCPUInformation();
|
|
|
|
PLATFORM_INTERFACE void GetCurrentDate( int *pDay, int *pMonth, int *pYear );
|
|
|
|
// ---------------------------------------------------------------------------------- //
|
|
// Performance Monitoring Events - L2 stats etc...
|
|
// ---------------------------------------------------------------------------------- //
|
|
PLATFORM_INTERFACE void InitPME();
|
|
PLATFORM_INTERFACE void ShutdownPME();
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Thread related functions
|
|
//-----------------------------------------------------------------------------
|
|
// Registers the current thread with Tier0's thread management system.
|
|
// This should be called on every thread created in the game.
|
|
PLATFORM_INTERFACE unsigned long Plat_RegisterThread( const tchar *pName = _T("Source Thread"));
|
|
|
|
// Registers the current thread as the primary thread.
|
|
PLATFORM_INTERFACE unsigned long Plat_RegisterPrimaryThread();
|
|
|
|
// VC-specific. Sets the thread's name so it has a friendly name in the debugger.
|
|
// This should generally only be handled by Plat_RegisterThread and Plat_RegisterPrimaryThread
|
|
PLATFORM_INTERFACE void Plat_SetThreadName( unsigned long dwThreadID, const tchar *pName );
|
|
|
|
// These would be private if it were possible to export private variables from a .DLL.
|
|
// They need to be variables because they are checked by inline functions at performance
|
|
// critical places.
|
|
PLATFORM_INTERFACE unsigned long Plat_PrimaryThreadID;
|
|
|
|
// Returns the ID of the currently executing thread.
|
|
PLATFORM_INTERFACE unsigned long Plat_GetCurrentThreadID();
|
|
|
|
// Returns the ID of the primary thread.
|
|
inline unsigned long Plat_GetPrimaryThreadID()
|
|
{
|
|
return Plat_PrimaryThreadID;
|
|
}
|
|
|
|
// Returns true if the current thread is the primary thread.
|
|
inline bool Plat_IsPrimaryThread()
|
|
{
|
|
//return true;
|
|
return (Plat_GetPrimaryThreadID() == Plat_GetCurrentThreadID() );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Process related functions
|
|
//-----------------------------------------------------------------------------
|
|
PLATFORM_INTERFACE const tchar *Plat_GetCommandLine();
|
|
#ifndef _WIN32
|
|
// helper function for OS's that don't have a ::GetCommandLine() call
|
|
PLATFORM_INTERFACE void Plat_SetCommandLine( const char *cmdLine );
|
|
#endif
|
|
PLATFORM_INTERFACE const char *Plat_GetCommandLineA();
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Security related functions
|
|
//-----------------------------------------------------------------------------
|
|
// Ensure that the hardware key's drivers have been installed.
|
|
PLATFORM_INTERFACE bool Plat_VerifyHardwareKeyDriver();
|
|
|
|
// Ok, so this isn't a very secure way to verify the hardware key for now. It
|
|
// is primarially depending on the fact that all the binaries have been wrapped
|
|
// with the secure wrapper provided by the hardware keys vendor.
|
|
PLATFORM_INTERFACE bool Plat_VerifyHardwareKey();
|
|
|
|
// The same as above, but notifies user with a message box when the key isn't in
|
|
// and gives him an opportunity to correct the situation.
|
|
PLATFORM_INTERFACE bool Plat_VerifyHardwareKeyPrompt();
|
|
|
|
// Can be called in real time, doesn't perform the verify every frame. Mainly just
|
|
// here to allow the game to drop out quickly when the key is removed, rather than
|
|
// allowing the wrapper to pop up it's own blocking dialog, which the engine doesn't
|
|
// like much.
|
|
PLATFORM_INTERFACE bool Plat_FastVerifyHardwareKey();
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Just logs file and line to simple.log
|
|
//-----------------------------------------------------------------------------
|
|
PLATFORM_INTERFACE void* Plat_SimpleLog( const tchar* file, int line );
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Returns true if debugger attached, false otherwise
|
|
//-----------------------------------------------------------------------------
|
|
#if defined(_WIN32)
|
|
PLATFORM_INTERFACE bool Plat_IsInDebugSession();
|
|
PLATFORM_INTERFACE void Plat_DebugString( const char * );
|
|
#else
|
|
#define Plat_IsInDebugSession() (false)
|
|
#define Plat_DebugString(s) ((void)0)
|
|
#endif
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// XBOX Components valid in PC compilation space
|
|
//-----------------------------------------------------------------------------
|
|
|
|
#define XBOX_DVD_SECTORSIZE 2048
|
|
#define XBOX_DVD_ECC_SIZE 32768 // driver reads in quantum ECC blocks
|
|
#define XBOX_HDD_SECTORSIZE 512
|
|
|
|
// Custom windows messages for Xbox input
|
|
#define WM_XREMOTECOMMAND (WM_USER + 100)
|
|
#define WM_XCONTROLLER_KEY (WM_USER + 101)
|
|
#define WM_SYS_UI (WM_USER + 102)
|
|
#define WM_SYS_SIGNINCHANGED (WM_USER + 103)
|
|
#define WM_SYS_STORAGEDEVICESCHANGED (WM_USER + 104)
|
|
#define WM_SYS_PROFILESETTINGCHANGED (WM_USER + 105)
|
|
#define WM_SYS_MUTELISTCHANGED (WM_USER + 106)
|
|
#define WM_SYS_INPUTDEVICESCHANGED (WM_USER + 107)
|
|
#define WM_SYS_INPUTDEVICECONFIGCHANGED (WM_USER + 108)
|
|
#define WM_LIVE_CONNECTIONCHANGED (WM_USER + 109)
|
|
#define WM_LIVE_INVITE_ACCEPTED (WM_USER + 110)
|
|
#define WM_LIVE_LINK_STATE_CHANGED (WM_USER + 111)
|
|
#define WM_LIVE_CONTENT_INSTALLED (WM_USER + 112)
|
|
#define WM_LIVE_MEMBERSHIP_PURCHASED (WM_USER + 113)
|
|
#define WM_LIVE_VOICECHAT_AWAY (WM_USER + 114)
|
|
#define WM_LIVE_PRESENCE_CHANGED (WM_USER + 115)
|
|
#define WM_FRIENDS_PRESENCE_CHANGED (WM_USER + 116)
|
|
#define WM_FRIENDS_FRIEND_ADDED (WM_USER + 117)
|
|
#define WM_FRIENDS_FRIEND_REMOVED (WM_USER + 118)
|
|
#define WM_CUSTOM_GAMEBANNERPRESSED (WM_USER + 119)
|
|
#define WM_CUSTOM_ACTIONPRESSED (WM_USER + 120)
|
|
#define WM_XMP_STATECHANGED (WM_USER + 121)
|
|
#define WM_XMP_PLAYBACKBEHAVIORCHANGED (WM_USER + 122)
|
|
#define WM_XMP_PLAYBACKCONTROLLERCHANGED (WM_USER + 123)
|
|
|
|
inline const char *GetPlatformExt( void )
|
|
{
|
|
return IsX360() ? ".360" : "";
|
|
}
|
|
|
|
// flat view, 6 hw threads
|
|
#define XBOX_PROCESSOR_0 ( 1<<0 )
|
|
#define XBOX_PROCESSOR_1 ( 1<<1 )
|
|
#define XBOX_PROCESSOR_2 ( 1<<2 )
|
|
#define XBOX_PROCESSOR_3 ( 1<<3 )
|
|
#define XBOX_PROCESSOR_4 ( 1<<4 )
|
|
#define XBOX_PROCESSOR_5 ( 1<<5 )
|
|
|
|
// core view, 3 cores with 2 hw threads each
|
|
#define XBOX_CORE_0_HWTHREAD_0 XBOX_PROCESSOR_0
|
|
#define XBOX_CORE_0_HWTHREAD_1 XBOX_PROCESSOR_1
|
|
#define XBOX_CORE_1_HWTHREAD_0 XBOX_PROCESSOR_2
|
|
#define XBOX_CORE_1_HWTHREAD_1 XBOX_PROCESSOR_3
|
|
#define XBOX_CORE_2_HWTHREAD_0 XBOX_PROCESSOR_4
|
|
#define XBOX_CORE_2_HWTHREAD_1 XBOX_PROCESSOR_5
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Include additional dependant header components.
|
|
//-----------------------------------------------------------------------------
|
|
#include "tier0/fasttimer.h"
|
|
|
|
#if defined( _X360 )
|
|
#include "xbox/xbox_core.h"
|
|
#endif
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Methods to invoke the constructor, copy constructor, and destructor
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <class T>
|
|
inline void Construct( T* pMemory )
|
|
{
|
|
::new( pMemory ) T;
|
|
}
|
|
|
|
template <class T>
|
|
inline void CopyConstruct( T* pMemory, T const& src )
|
|
{
|
|
::new( pMemory ) T(src);
|
|
}
|
|
|
|
template <class T>
|
|
inline void Destruct( T* pMemory )
|
|
{
|
|
pMemory->~T();
|
|
|
|
#ifdef _DEBUG
|
|
memset( pMemory, 0xDD, sizeof(T) );
|
|
#endif
|
|
}
|
|
|
|
|
|
//
|
|
// GET_OUTER()
|
|
//
|
|
// A platform-independent way for a contained class to get a pointer to its
|
|
// owner. If you know a class is exclusively used in the context of some
|
|
// "outer" class, this is a much more space efficient way to get at the outer
|
|
// class than having the inner class store a pointer to it.
|
|
//
|
|
// class COuter
|
|
// {
|
|
// class CInner // Note: this does not need to be a nested class to work
|
|
// {
|
|
// void PrintAddressOfOuter()
|
|
// {
|
|
// printf( "Outer is at 0x%x\n", GET_OUTER( COuter, m_Inner ) );
|
|
// }
|
|
// };
|
|
//
|
|
// CInner m_Inner;
|
|
// friend class CInner;
|
|
// };
|
|
|
|
#define GET_OUTER( OuterType, OuterMember ) \
|
|
( ( OuterType * ) ( (uint8 *)this - offsetof( OuterType, OuterMember ) ) )
|
|
|
|
|
|
/* TEMPLATE_FUNCTION_TABLE()
|
|
|
|
(Note added to platform.h so platforms that correctly support templated
|
|
functions can handle portions as templated functions rather than wrapped
|
|
functions)
|
|
|
|
Helps automate the process of creating an array of function
|
|
templates that are all specialized by a single integer.
|
|
This sort of thing is often useful in optimization work.
|
|
|
|
For example, using TEMPLATE_FUNCTION_TABLE, this:
|
|
|
|
TEMPLATE_FUNCTION_TABLE(int, Function, ( int blah, int blah ), 10)
|
|
{
|
|
return argument * argument;
|
|
}
|
|
|
|
is equivilent to the following:
|
|
|
|
(NOTE: the function has to be wrapped in a class due to code
|
|
generation bugs involved with directly specializing a function
|
|
based on a constant.)
|
|
|
|
template<int argument>
|
|
class FunctionWrapper
|
|
{
|
|
public:
|
|
int Function( int blah, int blah )
|
|
{
|
|
return argument*argument;
|
|
}
|
|
}
|
|
|
|
typedef int (*FunctionType)( int blah, int blah );
|
|
|
|
class FunctionName
|
|
{
|
|
public:
|
|
enum { count = 10 };
|
|
FunctionType functions[10];
|
|
};
|
|
|
|
FunctionType FunctionName::functions[] =
|
|
{
|
|
FunctionWrapper<0>::Function,
|
|
FunctionWrapper<1>::Function,
|
|
FunctionWrapper<2>::Function,
|
|
FunctionWrapper<3>::Function,
|
|
FunctionWrapper<4>::Function,
|
|
FunctionWrapper<5>::Function,
|
|
FunctionWrapper<6>::Function,
|
|
FunctionWrapper<7>::Function,
|
|
FunctionWrapper<8>::Function,
|
|
FunctionWrapper<9>::Function
|
|
};
|
|
*/
|
|
|
|
PLATFORM_INTERFACE bool vtune( bool resume );
|
|
|
|
|
|
#define TEMPLATE_FUNCTION_TABLE(RETURN_TYPE, NAME, ARGS, COUNT) \
|
|
\
|
|
typedef RETURN_TYPE (FASTCALL *__Type_##NAME) ARGS; \
|
|
\
|
|
template<const int nArgument> \
|
|
struct __Function_##NAME \
|
|
{ \
|
|
static RETURN_TYPE FASTCALL Run ARGS; \
|
|
}; \
|
|
\
|
|
template <const int i> \
|
|
struct __MetaLooper_##NAME : __MetaLooper_##NAME<i-1> \
|
|
{ \
|
|
__Type_##NAME func; \
|
|
inline __MetaLooper_##NAME() { func = __Function_##NAME<i>::Run; } \
|
|
}; \
|
|
\
|
|
template<> \
|
|
struct __MetaLooper_##NAME<0> \
|
|
{ \
|
|
__Type_##NAME func; \
|
|
inline __MetaLooper_##NAME() { func = __Function_##NAME<0>::Run; } \
|
|
}; \
|
|
\
|
|
class NAME \
|
|
{ \
|
|
private: \
|
|
static const __MetaLooper_##NAME<COUNT> m; \
|
|
public: \
|
|
enum { count = COUNT }; \
|
|
static const __Type_##NAME* functions; \
|
|
}; \
|
|
const __MetaLooper_##NAME<COUNT> NAME::m; \
|
|
const __Type_##NAME* NAME::functions = (__Type_##NAME*)&m; \
|
|
template<const int nArgument> \
|
|
RETURN_TYPE FASTCALL __Function_##NAME<nArgument>::Run ARGS
|
|
|
|
|
|
#define LOOP_INTERCHANGE(BOOLEAN, CODE)\
|
|
if( (BOOLEAN) )\
|
|
{\
|
|
CODE;\
|
|
} else\
|
|
{\
|
|
CODE;\
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
#if defined(_INC_WINDOWS) && defined(_WIN32)
|
|
template <typename FUNCPTR_TYPE>
|
|
class CDynamicFunction
|
|
{
|
|
public:
|
|
CDynamicFunction( const char *pszModule, const char *pszName, FUNCPTR_TYPE pfnFallback = NULL )
|
|
{
|
|
m_pfn = pfnFallback;
|
|
|
|
HMODULE hModule = ::LoadLibrary( pszModule );
|
|
if ( hModule )
|
|
m_pfn = (FUNCPTR_TYPE)::GetProcAddress( hModule, pszName );
|
|
}
|
|
|
|
operator bool() { return m_pfn != NULL; }
|
|
bool operator !() { return !m_pfn; }
|
|
operator FUNCPTR_TYPE() { return m_pfn; }
|
|
|
|
private:
|
|
FUNCPTR_TYPE m_pfn;
|
|
};
|
|
#endif
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
#include "tier0/valve_on.h"
|
|
|
|
#endif /* PLATFORM_H */
|