mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
689 lines
19 KiB
689 lines
19 KiB
/***
|
|
*tidtable.c - Access thread data table
|
|
*
|
|
* Copyright (c) 1989-2001, Microsoft Corporation. All rights reserved.
|
|
*
|
|
*Purpose:
|
|
* This module contains the following routines for multi-thread
|
|
* data support:
|
|
*
|
|
* _mtinit = Initialize the mthread data
|
|
* _getptd = get the pointer to the per-thread data structure for
|
|
* the current thread
|
|
* _freeptd = free up a per-thread data structure and its
|
|
* subordinate structures
|
|
* __threadid = return thread ID for the current thread
|
|
* __threadhandle = return pseudo-handle for the current thread
|
|
*
|
|
*Revision History:
|
|
* 05-04-90 JCR Translated from ASM to C for portable 32-bit OS/2
|
|
* 06-04-90 GJF Changed error message interface.
|
|
* 07-02-90 GJF Changed __threadid() for DCR 1024/2012.
|
|
* 08-08-90 GJF Removed 32 from API names.
|
|
* 10-08-90 GJF New-style function declarators.
|
|
* 10-09-90 GJF Thread ids are of type unsigned long! Also, fixed a
|
|
* bug in __threadid().
|
|
* 10-22-90 GJF Another bug in __threadid().
|
|
* 12-04-90 SRW Changed to include <oscalls.h> instead of <doscalls.h>
|
|
* 12-06-90 SRW Added _CRUISER_ and _WIN32 conditionals.
|
|
* 05-31-91 GJF Win32 version [_WIN32_].
|
|
* 07-18-91 GJF Fixed many silly errors [_WIN32_].
|
|
* 09-29-91 GJF Conditionally added _getptd_lk/_getptd1_lk so that
|
|
* DEBUG version of mlock doesn't infinitely recurse
|
|
* the first time _THREADDATA_LOCK is asserted [_WIN32_].
|
|
* 01-30-92 GJF Must init. _pxcptacttab field to _XcptActTab.
|
|
* 02-25-92 GJF Initialize _holdrand field to 1.
|
|
* 02-13-93 GJF Revised to use TLS API. Also, purged Cruiser support.
|
|
* 03-26-93 GJF Initialize ptd->_holdrand to 1L (see thread.c).
|
|
* 04-06-93 SKS Replace _CRTAPI* with __cdecl
|
|
* 04-13-93 SKS Add _mtterm to do multi-thread termination
|
|
* Set freed __tlsindex to -1 again to prevent mis-use
|
|
* 04-26-93 SKS _mtinit now returns 0 or 1, no longer calls _amsg_exit
|
|
* 12-13-93 SKS Add _freeptd(), which frees up the per-thread data
|
|
* maintained by the C run-time library.
|
|
* 04-12-94 GJF Made definition of __tlsindex conditional on ndef
|
|
* DLL_FOR_WIN32S. Also, replaced MTHREAD with _MT.
|
|
* 01-10-95 CFW Debug CRT allocs.
|
|
* 04-12-95 DAK Added NT kernel support for C++ exceptions
|
|
* 04-18-95 SKS Add 5 MIPS per-thread variables.
|
|
* 04-25-95 DAK More Kernel EH support
|
|
* 05-02-95 SKS add _initptd() to do initialization of per-thread data
|
|
* 05-24-95 CFW Add _defnewh.
|
|
* 06-12-95 JWM _getptd() now preserves LastError.
|
|
* 01-17-97 GJF _freeptd() must free the thread's copy of the
|
|
* exception-action table.
|
|
* 09-26-97 BWT Fix NTSUBSET
|
|
* 02-03-98 GJF Changes for Win64: use uintptr_t type for anything with
|
|
* a HANDLE value.
|
|
* 04-27-98 GJF Added support for per-thread mbc information.
|
|
* 07-28-98 JWM Initialize __pceh (new per-thread data for comerr support).
|
|
* 09-03-98 GJF Added support for per-thread locale information.
|
|
* 12-04-98 JWM Pulled all comerr support.
|
|
* 12-08-98 GJF In _freeptd, fixed several errors in cleaning up the
|
|
* threadlocinfo.
|
|
* 12-18-98 GJF Fixed one more error in _freeptd.
|
|
* 01-18-99 GJF Take care not to free up __ptlocinfo when a thread
|
|
* exits.
|
|
* 03-16-99 GJF threadlocinfo incorporates more reference counters
|
|
* 04-24-99 PML Added __lconv_intl_refcount
|
|
* 04-28-99 PML Wrap __declspec(allocate()) in _CRTALLOC macro.
|
|
* 11-03-99 RDL Win64 _NTSUBSET_ warning fix.
|
|
* 06-08-00 PML No need to keep per-thread mbcinfo in circular linked
|
|
* list. Also, don't free mbcinfo if it's also the global
|
|
* info (vs7#118174).
|
|
* 02-20-01 PML vs7#172586 Avoid _RT_LOCK by preallocating all locks
|
|
* that will be required, and returning failure back on
|
|
* inability to allocate a lock.
|
|
* 06-12-01 BWT ntbug: 414059 - cleanup from mtinit failure
|
|
*
|
|
*******************************************************************************/
|
|
|
|
#if defined(_MT) || defined(_NTSUBSET_)
|
|
|
|
# if defined(_NTSUBSET_)
|
|
|
|
#include <nt.h>
|
|
#include <ntrtl.h>
|
|
#include <nturtl.h>
|
|
#include <ntstatus.h>
|
|
#include <ntos.h>
|
|
#include <fsrtl.h>
|
|
|
|
# endif /* _NTSUBSET_ */
|
|
|
|
#include <sect_attribs.h>
|
|
#include <cruntime.h>
|
|
#include <oscalls.h>
|
|
#include <internal.h>
|
|
#include <mtdll.h>
|
|
#include <memory.h>
|
|
#include <msdos.h>
|
|
#include <rterr.h>
|
|
#include <stdlib.h>
|
|
#include <stddef.h>
|
|
#include <dbgint.h>
|
|
|
|
#ifdef _MT
|
|
extern pthreadmbcinfo __ptmbcinfo;
|
|
|
|
extern threadlocinfo __initiallocinfo;
|
|
extern pthreadlocinfo __ptlocinfo;
|
|
|
|
void __cdecl __freetlocinfo(pthreadlocinfo);
|
|
#endif
|
|
|
|
unsigned long __tlsindex = 0xffffffff;
|
|
|
|
#if defined(_NTSUBSET_)
|
|
|
|
#define BEGIN_NONPAGED_CODE code_seg(".text")
|
|
#define END_NONPAGED_CODE code_seg("PAGE")
|
|
|
|
static _ptiddata pTheKernelExceptionContextPool;
|
|
static KSPIN_LOCK TheKernelExceptionContextPoolLock;
|
|
|
|
/****
|
|
*_initKernelExceptionContextPool() - creates kernel-mode exception context pool
|
|
*
|
|
*Purpose:
|
|
* (1) Initializes spin-lock used to serialize access to pool
|
|
*
|
|
*Entry:
|
|
* <NONE>
|
|
*Exit:
|
|
* <NONE>
|
|
*
|
|
*
|
|
*******************************************************************************/
|
|
#pragma BEGIN_NONPAGED_CODE
|
|
void __cdecl _initKernelExceptionContextPool(void)
|
|
{
|
|
KeInitializeSpinLock( &TheKernelExceptionContextPoolLock );
|
|
}
|
|
#pragma END_NONPAGED_CODE
|
|
|
|
|
|
/****
|
|
* Tell the linker to insert a call to this function into the CRT runtime
|
|
* startup.
|
|
*
|
|
* The segment .CRT$XCA is the first data segment bounding the array
|
|
* of CRT startup functions. We therefore insert our function between
|
|
* the first (.CRT$XCA) and the last (.CRT$XCZ).
|
|
*
|
|
*******************************************************************************/
|
|
#pragma data_seg(".CRT$XCAA")
|
|
typedef void (__cdecl *PFStartup)(void);
|
|
_CRTALLOC(".CRT$XCAA") PFStartup _KernelECPool = _initKernelExceptionContextPool;
|
|
#pragma data_seg(".CRT$XCZ")
|
|
|
|
#pragma data_seg()
|
|
|
|
|
|
/****
|
|
*_emptyKernelExceptionContextPool() - empties kernel-mode exception context pool
|
|
*
|
|
*Purpose:
|
|
* (1) Deletes all entries in the pool
|
|
*
|
|
*Entry:
|
|
* <NONE>
|
|
*Exit:
|
|
* <NONE>
|
|
*
|
|
*
|
|
*******************************************************************************/
|
|
#pragma BEGIN_NONPAGED_CODE
|
|
void _emptyKernelExceptionContextPool()
|
|
{
|
|
while ( pTheKernelExceptionContextPool != 0 ) {
|
|
_ptiddata ptd = pTheKernelExceptionContextPool;
|
|
pTheKernelExceptionContextPool = pTheKernelExceptionContextPool->_next;
|
|
|
|
ExFreePool( ptd );
|
|
}
|
|
}
|
|
#pragma END_NONPAGED_CODE
|
|
|
|
|
|
/****
|
|
*_newKernelTid() - Creates, initializes and sets the current thread's
|
|
* OfsChain to _tiddata;
|
|
*
|
|
*Purpose:
|
|
* (1) Creates a new kernel-mode _tiddata
|
|
* (2) Initializes fields in new _tiddata
|
|
* (3) Sets current thread's OfsChain to new _tiddata
|
|
*
|
|
* NOTES:
|
|
* (1) Can not fail; the memory allocation will cause a fatal system
|
|
* trap if memory can not be allocated
|
|
*
|
|
*Entry:
|
|
* <NONE>
|
|
*Exit:
|
|
* _ptiddata
|
|
*
|
|
*******************************************************************************/
|
|
#pragma BEGIN_NONPAGED_CODE
|
|
static _ptiddata _newKernelTid(void)
|
|
{
|
|
/*
|
|
* Pop an entry from the pool of _ptiddata's
|
|
*/
|
|
|
|
_ptiddata ptd;
|
|
PSINGLE_LIST_ENTRY entry;
|
|
|
|
entry = ExInterlockedPopEntryList((PSINGLE_LIST_ENTRY) & pTheKernelExceptionContextPool,
|
|
& TheKernelExceptionContextPoolLock );
|
|
|
|
/*
|
|
* If none are available, create a new one.
|
|
*/
|
|
|
|
if ( 0 == entry ) {
|
|
ptd = FsRtlAllocatePoolWithTag( NonPagedPoolMustSucceed,
|
|
sizeof(struct _tiddata),
|
|
' dtp' );
|
|
|
|
} else {
|
|
ptd = CONTAINING_RECORD( entry, struct _tiddata, _next );
|
|
}
|
|
|
|
ptd->_tid = (unsigned long) (uintptr_t) PsGetCurrentThread()->Cid.UniqueThread;
|
|
ptd->_terminate = 0;
|
|
ptd->_unexpected = 0;
|
|
ptd->_translator = SystemExceptionTranslator;
|
|
ptd->_curexception = 0;
|
|
ptd->_curcontext = 0;
|
|
|
|
PsGetCurrentThread()->OfsChain = ptd;
|
|
|
|
return ptd;
|
|
}
|
|
#pragma END_NONPAGED_CODE
|
|
|
|
|
|
/****
|
|
*_freeKernelTid() - appends the _tiddata to the pool
|
|
*
|
|
*Entry:
|
|
* _ptiddata
|
|
*Exit:
|
|
*
|
|
*******************************************************************************/
|
|
#pragma BEGIN_NONPAGED_CODE
|
|
static void _freeKernelTid(_ptiddata ptd)
|
|
{
|
|
/*
|
|
* Return the entry to the kernel pool
|
|
*/
|
|
ExInterlockedPushEntryList( (PSINGLE_LIST_ENTRY) &pTheKernelExceptionContextPool,
|
|
(PSINGLE_LIST_ENTRY) &ptd->_next,
|
|
&TheKernelExceptionContextPoolLock );
|
|
}
|
|
#pragma END_NONPAGED_CODE
|
|
#endif /* _NTSUBSET_ */
|
|
|
|
/****
|
|
*_mtinit() - Init multi-thread data bases
|
|
*
|
|
*Purpose:
|
|
* (1) Call _mtinitlocks to create/open all lock semaphores.
|
|
* (2) Allocate a TLS index to hold pointers to per-thread data
|
|
* structure.
|
|
*
|
|
* NOTES:
|
|
* (1) Only to be called ONCE at startup
|
|
* (2) Must be called BEFORE any mthread requests are made
|
|
*
|
|
*Entry:
|
|
* <NONE>
|
|
*Exit:
|
|
* returns FALSE on failure
|
|
*
|
|
*Uses:
|
|
* <any registers may be modified at init time>
|
|
*
|
|
*Exceptions:
|
|
*
|
|
*******************************************************************************/
|
|
#if defined(_NTSUBSET_)
|
|
#pragma BEGIN_NONPAGED_CODE
|
|
#endif
|
|
|
|
int __cdecl _mtinit (
|
|
void
|
|
)
|
|
{
|
|
_ptiddata ptd;
|
|
|
|
#if defined(_NTSUBSET_)
|
|
ptd = _newKernelTid();
|
|
#else
|
|
|
|
/*
|
|
* Initialize the mthread lock data base
|
|
*/
|
|
|
|
if ( !_mtinitlocks() ) {
|
|
_mtterm();
|
|
return FALSE; /* fail to load DLL */
|
|
}
|
|
|
|
/*
|
|
* Allocate a TLS index to maintain pointers to per-thread data
|
|
*/
|
|
if ( (__tlsindex = TlsAlloc()) == 0xffffffff ) {
|
|
_mtterm();
|
|
return FALSE; /* fail to load DLL */
|
|
}
|
|
|
|
|
|
/*
|
|
* Create a per-thread data structure for this (i.e., the startup)
|
|
* thread.
|
|
*/
|
|
if ( ((ptd = _calloc_crt(1, sizeof(struct _tiddata))) == NULL) ||
|
|
!TlsSetValue(__tlsindex, (LPVOID)ptd) )
|
|
{
|
|
_mtterm();
|
|
return FALSE; /* fail to load DLL */
|
|
}
|
|
|
|
/*
|
|
* Initialize the per-thread data
|
|
*/
|
|
|
|
_initptd(ptd);
|
|
|
|
ptd->_tid = GetCurrentThreadId();
|
|
ptd->_thandle = (uintptr_t)(-1);
|
|
|
|
#endif /* _NTSUBSET_ */
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/****
|
|
*_mtterm() - Clean-up multi-thread data bases
|
|
*
|
|
*Purpose:
|
|
* (1) Call _mtdeletelocks to free up all lock semaphores.
|
|
* (2) Free up the TLS index used to hold pointers to
|
|
* per-thread data structure.
|
|
*
|
|
* NOTES:
|
|
* (1) Only to be called ONCE at termination
|
|
* (2) Must be called AFTER all mthread requests are made
|
|
*
|
|
*Entry:
|
|
* <NONE>
|
|
*Exit:
|
|
* returns
|
|
*
|
|
*Uses:
|
|
*
|
|
*Exceptions:
|
|
*
|
|
*******************************************************************************/
|
|
|
|
void __cdecl _mtterm (
|
|
void
|
|
)
|
|
{
|
|
|
|
#if defined(_NTSUBSET_)
|
|
|
|
_freeKernelTid( PsGetCurrentThread()->OfsChain );
|
|
PsGetCurrentThread()->OfsChain = 0;
|
|
|
|
#else /* _NTSUBSET_ */
|
|
|
|
/*
|
|
* Clean up the mthread lock data base
|
|
*/
|
|
|
|
_mtdeletelocks();
|
|
|
|
/*
|
|
* Free up the TLS index
|
|
*
|
|
* (Set the variable __tlsindex back to the unused state (-1L).)
|
|
*/
|
|
|
|
if ( __tlsindex != 0xffffffff ) {
|
|
TlsFree(__tlsindex);
|
|
__tlsindex = 0xffffffff;
|
|
}
|
|
#endif /* _NTSUBSET_ */
|
|
}
|
|
|
|
|
|
|
|
#if !defined(_NTSUBSET_)
|
|
|
|
/***
|
|
*void _initptd(_ptiddata ptd) - initialize a per-thread data structure
|
|
*
|
|
*Purpose:
|
|
* This routine handles all of the per-thread initialization
|
|
* which is common to _beginthread, _beginthreadex, _mtinit
|
|
* and _getptd.
|
|
*
|
|
*Entry:
|
|
* pointer to a per-thread data block
|
|
*
|
|
*Exit:
|
|
* the common fields in that block are initialized
|
|
*
|
|
*Exceptions:
|
|
*
|
|
*******************************************************************************/
|
|
|
|
void __cdecl _initptd (
|
|
_ptiddata ptd
|
|
)
|
|
{
|
|
ptd->_pxcptacttab = (void *)_XcptActTab;
|
|
ptd->_holdrand = 1L;
|
|
#ifdef ANSI_NEW_HANDLER
|
|
ptd->_newh = _defnewh;
|
|
#endif /* ANSI_NEW_HANDLER */
|
|
|
|
#ifdef _M_MRX000
|
|
/*
|
|
* MIPS per-thread data
|
|
*/
|
|
ptd->_MipsPtdDelta =
|
|
ptd->_MipsPtdEpsilon = -1L ;
|
|
#endif
|
|
}
|
|
|
|
#endif /* _NTSUBSET_ */
|
|
|
|
|
|
/***
|
|
*_ptiddata _getptd(void) - get per-thread data structure for the current thread
|
|
*
|
|
*Purpose:
|
|
*
|
|
*Entry:
|
|
* unsigned long tid
|
|
*
|
|
*Exit:
|
|
* success = pointer to _tiddata structure for the thread
|
|
* failure = fatal runtime exit
|
|
*
|
|
*Exceptions:
|
|
*
|
|
*******************************************************************************/
|
|
|
|
_ptiddata __cdecl _getptd (
|
|
void
|
|
)
|
|
{
|
|
_ptiddata ptd;
|
|
DWORD TL_LastError;
|
|
|
|
#if defined(_NTSUBSET_)
|
|
ptd = PsGetCurrentThread()->OfsChain;
|
|
|
|
if ( ptd == 0 ) {
|
|
ptd = _newKernelTid();
|
|
}
|
|
#else /* _NTSUBSET_ */
|
|
|
|
TL_LastError = GetLastError();
|
|
if ( (ptd = TlsGetValue(__tlsindex)) == NULL ) {
|
|
/*
|
|
* no per-thread data structure for this thread. try to create
|
|
* one.
|
|
*/
|
|
if ( ((ptd = _calloc_crt(1, sizeof(struct _tiddata))) != NULL) &&
|
|
TlsSetValue(__tlsindex, (LPVOID)ptd) ) {
|
|
|
|
/*
|
|
* Initialize of per-thread data
|
|
*/
|
|
|
|
_initptd(ptd);
|
|
|
|
ptd->_tid = GetCurrentThreadId();
|
|
ptd->_thandle = (uintptr_t)(-1);
|
|
}
|
|
else
|
|
_amsg_exit(_RT_THREAD); /* write message and die */
|
|
}
|
|
|
|
SetLastError(TL_LastError);
|
|
|
|
#endif /* _NTSUBSET_ */
|
|
|
|
return(ptd);
|
|
}
|
|
|
|
|
|
/***
|
|
*void _freeptd(_ptiddata) - free up a per-thread data structure
|
|
*
|
|
*Purpose:
|
|
* Called from _endthread and from a DLL thread detach handler,
|
|
* this routine frees up the per-thread buffer associated with a
|
|
* thread that is going away. The tiddata structure itself is
|
|
* freed, but not until its subordinate buffers are freed.
|
|
*
|
|
*Entry:
|
|
* pointer to a per-thread data block (malloc-ed memory)
|
|
* If NULL, the pointer for the current thread is fetched.
|
|
*
|
|
*Exit:
|
|
*
|
|
*Exceptions:
|
|
*
|
|
*******************************************************************************/
|
|
|
|
void __cdecl _freeptd (
|
|
_ptiddata ptd
|
|
)
|
|
{
|
|
|
|
#if defined(_NTSUBSET_)
|
|
|
|
if ( 0 == ptd ) {
|
|
ptd = PsGetCurrentThread()->OfsChain;
|
|
} else {
|
|
ASSERT( ptd == PsGetCurrentThread()->OfsChain );
|
|
}
|
|
|
|
_freeKernelTid( ptd );
|
|
PsGetCurrentThread()->OfsChain = 0;
|
|
|
|
#else /* _NTSUBSET_ */
|
|
|
|
pthreadmbcinfo ptmbci;
|
|
pthreadlocinfo ptloci;
|
|
|
|
/*
|
|
* Do nothing unless per-thread data has been allocated for this module!
|
|
*/
|
|
|
|
if ( __tlsindex != 0xFFFFFFFF ) {
|
|
/*
|
|
* if parameter "ptd" is NULL, get the per-thread data pointer
|
|
* Must NOT call _getptd because it will allocate one if none exists!
|
|
*/
|
|
|
|
if ( ! ptd )
|
|
ptd = TlsGetValue(__tlsindex );
|
|
|
|
/*
|
|
* Free up the _tiddata structure & its malloc-ed buffers.
|
|
*/
|
|
|
|
if ( ptd ) {
|
|
if(ptd->_errmsg)
|
|
_free_crt((void *)ptd->_errmsg);
|
|
|
|
if(ptd->_namebuf0)
|
|
_free_crt((void *)ptd->_namebuf0);
|
|
|
|
if(ptd->_namebuf1)
|
|
_free_crt((void *)ptd->_namebuf1);
|
|
|
|
if(ptd->_asctimebuf)
|
|
_free_crt((void *)ptd->_asctimebuf);
|
|
|
|
if(ptd->_gmtimebuf)
|
|
_free_crt((void *)ptd->_gmtimebuf);
|
|
|
|
if(ptd->_cvtbuf)
|
|
_free_crt((void *)ptd->_cvtbuf);
|
|
|
|
if (ptd->_pxcptacttab != _XcptActTab)
|
|
_free_crt((void *)ptd->_pxcptacttab);
|
|
|
|
_mlock(_MB_CP_LOCK);
|
|
__try {
|
|
if ( ((ptmbci = ptd->ptmbcinfo) != NULL) &&
|
|
(--(ptmbci->refcount) == 0) &&
|
|
(ptmbci != __ptmbcinfo) )
|
|
_free_crt(ptmbci);
|
|
}
|
|
__finally {
|
|
_munlock(_MB_CP_LOCK);
|
|
}
|
|
|
|
_mlock(_SETLOCALE_LOCK);
|
|
|
|
__try {
|
|
if ( (ptloci = ptd->ptlocinfo) != NULL )
|
|
{
|
|
(ptloci->refcount)--;
|
|
|
|
if ( ptloci->lconv_intl_refcount != NULL )
|
|
(*(ptloci->lconv_intl_refcount))--;
|
|
|
|
if ( ptloci->lconv_mon_refcount != NULL )
|
|
(*(ptloci->lconv_mon_refcount))--;
|
|
|
|
if ( ptloci->lconv_num_refcount != NULL )
|
|
(*(ptloci->lconv_num_refcount))--;
|
|
|
|
if ( ptloci->ctype1_refcount != NULL )
|
|
(*(ptloci->ctype1_refcount))--;
|
|
|
|
(ptloci->lc_time_curr->refcount)--;
|
|
|
|
if ( (ptloci != __ptlocinfo) &&
|
|
(ptloci != &__initiallocinfo) &&
|
|
(ptloci->refcount == 0) )
|
|
__freetlocinfo(ptloci);
|
|
}
|
|
}
|
|
__finally {
|
|
_munlock(_SETLOCALE_LOCK);
|
|
}
|
|
|
|
_free_crt((void *)ptd);
|
|
}
|
|
|
|
/*
|
|
* Zero out the one pointer to the per-thread data block
|
|
*/
|
|
|
|
TlsSetValue(__tlsindex, (LPVOID)0);
|
|
}
|
|
|
|
#endif /* _NTSUBSET_ */
|
|
|
|
}
|
|
|
|
#if defined(_NTSUBSET_)
|
|
#pragma END_NONPAGED_CODE
|
|
#endif
|
|
|
|
|
|
/***
|
|
*__threadid() - Returns current thread ID
|
|
*__threadhandle() - Returns "pseudo-handle" for current thread
|
|
*
|
|
*Purpose:
|
|
* The two function are simply do-nothing wrappers for the corresponding
|
|
* Win32 APIs (GetCurrentThreadId and GetCurrentThread, respectively).
|
|
*
|
|
*Entry:
|
|
* void
|
|
*
|
|
*Exit:
|
|
* thread ID value
|
|
*
|
|
*Exceptions:
|
|
*
|
|
*******************************************************************************/
|
|
|
|
_CRTIMP unsigned long __cdecl __threadid (
|
|
void
|
|
)
|
|
{
|
|
#if defined(_NTSUBSET_)
|
|
return( (unsigned long) (uintptr_t) PsGetCurrentThread()->Cid.UniqueThread );
|
|
#else
|
|
return( GetCurrentThreadId() );
|
|
#endif
|
|
}
|
|
|
|
_CRTIMP uintptr_t __cdecl __threadhandle(
|
|
void
|
|
)
|
|
{
|
|
#if defined(_NTSUBSET_)
|
|
return( (uintptr_t) PsGetCurrentThread()->Cid.UniqueThread );
|
|
#else
|
|
return( (uintptr_t)GetCurrentThread() );
|
|
#endif
|
|
}
|
|
|
|
#endif
|