[轉(zhuǎn)載]Hacking Windows CE: 如何從線程ID獲取線程名稱(chēng)
Posted on 2013-03-14 10:38 宋鵬 閱讀(1173) 評(píng)論(1) 編輯 收藏 引用在一個(gè)線程出現(xiàn)異常行為時(shí),比如說(shuō)CPU占用率過(guò)高,拋出異常等,你一定想知道這個(gè)線程是由哪個(gè)模塊創(chuàng)建的。因此無(wú)論在哪個(gè)操作系統(tǒng)上,獲取線程名稱(chēng)是診斷線程相關(guān)問(wèn)題的重要一步。
在Windows CE上就沒(méi)這么幸運(yùn)了,沒(méi)有任何現(xiàn)成的API可用。官方Windows CE Base Team的blog對(duì)這個(gè)問(wèn)題的回答是可以用Remote Kernel Tracker,不過(guò)這需要你build一個(gè)特殊的kernel image,enable一些profiler功能-這在顯示的問(wèn)題診斷中顯然是不實(shí)際的。那么有沒(méi)有辦法不需要什么特殊的配置就可像Windows桌面操作系統(tǒng)那樣獲得入口地址呢?有是有的,不過(guò)需要一些hack手段。仔細(xì)研究CE下的Thread內(nèi)核數(shù)據(jù)結(jié)構(gòu),就會(huì)發(fā)現(xiàn)Thread結(jié)構(gòu)中有一項(xiàng)是記錄線程入口地址的。
而且,在Windows CE下,線程ID和其handle的值是一樣的!!因此我們可以寫(xiě)一個(gè)這樣的函數(shù)從線程ID拿到入口地址:
為了使用這些內(nèi)核數(shù)據(jù)結(jié)構(gòu),我們還需要另外一些輔助結(jié)構(gòu)和函數(shù),比較完整的代碼如下。當(dāng)然,官方肯定是不建議這么做的,但是重要的是解決問(wèn)題,你說(shuō)呢。
從線程ID獲取線程名稱(chēng)通常的方法是,先獲取該線程的入口地址,然后枚舉進(jìn)程內(nèi)所有已加載模塊,最后判斷線程入口地址落在哪個(gè)加載模塊范圍內(nèi)。枚舉進(jìn)程內(nèi)已加載模塊可用Win32標(biāo)準(zhǔn)的CreateToolhelp32Snapshot/Module32First/Module32Next系列ToolHelp API得到。獲取線程入口地址則沒(méi)有線程的Win32 API可用。不過(guò)在Windows NT based操作系統(tǒng)上(包括Windows NT 4.0/2000/XP/2003,等),有一個(gè)未公開(kāi)的Native API可用:NtQueryInformationThread。其聲明如下:
DWORD WINAPI NtQueryInformationThread(
HANDLE ThreadHandle,
THREAD_INFORMATION_CLASS ThreadInformationClass,
PVOID ThreadInformation,
ULONG ThreadInformationLength,
PULONG ReturnLength
);
HANDLE ThreadHandle,
THREAD_INFORMATION_CLASS ThreadInformationClass,
PVOID ThreadInformation,
ULONG ThreadInformationLength,
PULONG ReturnLength
);
獲取線程入口地址可用:
DWORD GetThreadStartAddress(DWORD dwThreadId)
{
HANDLE hThread = OpenThread(THREAD_ALL_ACCESS, FALSE, dwThreadId);
DWORD retaddr, len, error;
retaddr = len = 0;
error = NtQueryInformationThread( hThread, 9, &retaddr, sizeof(retaddr), &len );
CloseHandle(hThread);
if( error != 0 )
retaddr = 0;
return retaddr;
}
{
HANDLE hThread = OpenThread(THREAD_ALL_ACCESS, FALSE, dwThreadId);
DWORD retaddr, len, error;
retaddr = len = 0;
error = NtQueryInformationThread( hThread, 9, &retaddr, sizeof(retaddr), &len );
CloseHandle(hThread);
if( error != 0 )
retaddr = 0;
return retaddr;
}
在Windows CE上就沒(méi)這么幸運(yùn)了,沒(méi)有任何現(xiàn)成的API可用。官方Windows CE Base Team的blog對(duì)這個(gè)問(wèn)題的回答是可以用Remote Kernel Tracker,不過(guò)這需要你build一個(gè)特殊的kernel image,enable一些profiler功能-這在顯示的問(wèn)題診斷中顯然是不實(shí)際的。那么有沒(méi)有辦法不需要什么特殊的配置就可像Windows桌面操作系統(tǒng)那樣獲得入口地址呢?有是有的,不過(guò)需要一些hack手段。仔細(xì)研究CE下的Thread內(nèi)核數(shù)據(jù)結(jié)構(gòu),就會(huì)發(fā)現(xiàn)Thread結(jié)構(gòu)中有一項(xiàng)是記錄線程入口地址的。
typedef struct Thread {
DWORD _1[3];
PPROCESS pProc; /* 0C: pointer to current process */
PPROCESS pOwnerProc; /* 10: pointer to owner process */
DWORD _2[18];
DWORD dwStartAddr; /* 5c: thread PC at creation, used to get thread name */
DWORD _3[10];
}THREAD, *PTHREAD; /* Thread */
DWORD _1[3];
PPROCESS pProc; /* 0C: pointer to current process */
PPROCESS pOwnerProc; /* 10: pointer to owner process */
DWORD _2[18];
DWORD dwStartAddr; /* 5c: thread PC at creation, used to get thread name */
DWORD _3[10];
}THREAD, *PTHREAD; /* Thread */
因此要做的就是想辦法根據(jù)線程ID或handle得到這個(gè)數(shù)據(jù)。再研究,發(fā)現(xiàn)線程的Thread內(nèi)核數(shù)據(jù)結(jié)構(gòu)可通過(guò)句柄得到:
PTHREAD pTh = HandleToThread(ThreadHandle);
而且,在Windows CE下,線程ID和其handle的值是一樣的!!因此我們可以寫(xiě)一個(gè)這樣的函數(shù)從線程ID拿到入口地址:
DWORD GetThreadStartAddress(DWORD dwThreadId)
{
DWORD dwStartAddress = 0;
BOOL fOldMode = SetKMode(TRUE);
PTHREAD pTh = HandleToThread((HANDLE)dwThreadId);
if (pTh)
{
dwStartAddress = (DWORD)MapPtrToProcess((LPVOID)pTh->dwStartAddr, pTh->pOwnerProc->hProc);
}
return dwStartAddress;
}
{
DWORD dwStartAddress = 0;
BOOL fOldMode = SetKMode(TRUE);
PTHREAD pTh = HandleToThread((HANDLE)dwThreadId);
if (pTh)
{
dwStartAddress = (DWORD)MapPtrToProcess((LPVOID)pTh->dwStartAddr, pTh->pOwnerProc->hProc);
}
return dwStartAddress;
}
為了使用這些內(nèi)核數(shù)據(jù)結(jié)構(gòu),我們還需要另外一些輔助結(jié)構(gòu)和函數(shù),比較完整的代碼如下。當(dāng)然,官方肯定是不建議這么做的,但是重要的是解決問(wèn)題,你說(shuō)呢。
typedef struct Process {
DWORD _1[2];
HANDLE hProc; /* 08: handle for this process, needed only for SC_GetProcFromPtr */
}PROCESS, *PPROCESS;
typedef struct Thread {
DWORD _1[3];
PPROCESS pProc; /* 0C: pointer to current process */
PPROCESS pOwnerProc; /* 10: pointer to owner process */
DWORD _2[18];
DWORD dwStartAddr; /* 5c: thread PC at creation, used to get thread name */
DWORD _3[10];
}THREAD, *PTHREAD; /* Thread */
typedef struct cinfo {
char acName[4]; /* 00: object type ID string */
uchar disp; /* 04: type of dispatch */
uchar type; /* 05: api handle type */
ushort cMethods; /* 06: # of methods in dispatch table */
const PFNVOID *ppfnMethods;/* 08: ptr to array of methods (in server address space) */
const DWORD *pdwSig; /* 0C: ptr to array of method signatures */
PPROCESS pServer; /* 10: ptr to server process */
} CINFO; /* cinfo */
typedef CINFO *PCINFO;
typedef struct _HDATA HDATA, *PHDATA;
struct _HDATA {
DWORD _1[2]; /* 00: links for active handle list */
HANDLE hValue; /* 08: Current value of handle (nonce) */
DWORD lock; /* 0C: access information */
DWORD ref; /* 10: reference information */
const CINFO *pci; /* 14: ptr to object class description structure */
PVOID pvObj; /* 18: ptr to object */
DWORD dwInfo; /* 1C: extra handle info */
}; /* 20: sizeof(HDATA) */
#ifdef x86
struct KDataStruct {
LPDWORD lpvTls; /* 0x000 Current thread local storage pointer */
HANDLE ahSys[NUM_SYS_HANDLES]; /* 0x004 If this moves, change kapi.h */
DWORD _1[4];
ulong handleBase; /* 0x094 base address of handle table */
}; /* KDataStruct */
#endif
#ifdef ARM
struct KDataStruct {
LPDWORD lpvTls; /* 0x000 Current thread local storage pointer */
HANDLE ahSys[NUM_SYS_HANDLES]; /* 0x004 If this moves, change kapi.h */
DWORD _1[6];
ulong handleBase; /* 0x09c handle table base address */
}; /* KDataStruct */
#endif
#define HandleToThread(h) ((THREAD *)GetObjectPtrByType((h),SH_CURTHREAD))
#define HANDLE_ADDRESS_MASK 0x1ffffffc
void h2p(HANDLE h, PHDATA& phdRet)
{
if ((ulong)h < NUM_SYS_HANDLES+SYS_HANDLE_BASE && (ulong)h >= SYS_HANDLE_BASE)
h = ((KDataStruct*)PUserKData)->ahSys[(uint)h-SYS_HANDLE_BASE];
if (h)
{
phdRet = (PHDATA)(((ulong)h & HANDLE_ADDRESS_MASK) + ((KDataStruct*)PUserKData)->handleBase);
if (phdRet->hValue != h)
phdRet = 0;
}
else
phdRet = 0;
}
PVOID GetObjectPtrByType(HANDLE h, int type)
{
PHDATA phd;
h2p(h, phd);
return (phd && phd->pci && phd->pci->type==type) ? phd->pvObj : 0;
}
extern "C" LPVOID WINAPI MapPtrToProcess(LPVOID lpv, HANDLE hProc);
extern "C" BOOL WINAPI SetKMode(BOOL fMode);
DWORD GetThreadStartAddress(DWORD dwThreadId)
{
DWORD dwStartAddress = 0;
BOOL fOldMode = SetKMode(TRUE);
PTHREAD pTh = HandleToThread((HANDLE)dwThreadId);
if (pTh)
{
dwStartAddress = (DWORD)MapPtrToProcess((LPVOID)pTh->dwStartAddr, pTh->pOwnerProc->hProc);
}
return dwStartAddress;
}
DWORD _1[2];
HANDLE hProc; /* 08: handle for this process, needed only for SC_GetProcFromPtr */
}PROCESS, *PPROCESS;
typedef struct Thread {
DWORD _1[3];
PPROCESS pProc; /* 0C: pointer to current process */
PPROCESS pOwnerProc; /* 10: pointer to owner process */
DWORD _2[18];
DWORD dwStartAddr; /* 5c: thread PC at creation, used to get thread name */
DWORD _3[10];
}THREAD, *PTHREAD; /* Thread */
typedef struct cinfo {
char acName[4]; /* 00: object type ID string */
uchar disp; /* 04: type of dispatch */
uchar type; /* 05: api handle type */
ushort cMethods; /* 06: # of methods in dispatch table */
const PFNVOID *ppfnMethods;/* 08: ptr to array of methods (in server address space) */
const DWORD *pdwSig; /* 0C: ptr to array of method signatures */
PPROCESS pServer; /* 10: ptr to server process */
} CINFO; /* cinfo */
typedef CINFO *PCINFO;
typedef struct _HDATA HDATA, *PHDATA;
struct _HDATA {
DWORD _1[2]; /* 00: links for active handle list */
HANDLE hValue; /* 08: Current value of handle (nonce) */
DWORD lock; /* 0C: access information */
DWORD ref; /* 10: reference information */
const CINFO *pci; /* 14: ptr to object class description structure */
PVOID pvObj; /* 18: ptr to object */
DWORD dwInfo; /* 1C: extra handle info */
}; /* 20: sizeof(HDATA) */
#ifdef x86
struct KDataStruct {
LPDWORD lpvTls; /* 0x000 Current thread local storage pointer */
HANDLE ahSys[NUM_SYS_HANDLES]; /* 0x004 If this moves, change kapi.h */
DWORD _1[4];
ulong handleBase; /* 0x094 base address of handle table */
}; /* KDataStruct */
#endif
#ifdef ARM
struct KDataStruct {
LPDWORD lpvTls; /* 0x000 Current thread local storage pointer */
HANDLE ahSys[NUM_SYS_HANDLES]; /* 0x004 If this moves, change kapi.h */
DWORD _1[6];
ulong handleBase; /* 0x09c handle table base address */
}; /* KDataStruct */
#endif
#define HandleToThread(h) ((THREAD *)GetObjectPtrByType((h),SH_CURTHREAD))
#define HANDLE_ADDRESS_MASK 0x1ffffffc
void h2p(HANDLE h, PHDATA& phdRet)
{
if ((ulong)h < NUM_SYS_HANDLES+SYS_HANDLE_BASE && (ulong)h >= SYS_HANDLE_BASE)
h = ((KDataStruct*)PUserKData)->ahSys[(uint)h-SYS_HANDLE_BASE];
if (h)
{
phdRet = (PHDATA)(((ulong)h & HANDLE_ADDRESS_MASK) + ((KDataStruct*)PUserKData)->handleBase);
if (phdRet->hValue != h)
phdRet = 0;
}
else
phdRet = 0;
}
PVOID GetObjectPtrByType(HANDLE h, int type)
{
PHDATA phd;
h2p(h, phd);
return (phd && phd->pci && phd->pci->type==type) ? phd->pvObj : 0;
}
extern "C" LPVOID WINAPI MapPtrToProcess(LPVOID lpv, HANDLE hProc);
extern "C" BOOL WINAPI SetKMode(BOOL fMode);
DWORD GetThreadStartAddress(DWORD dwThreadId)
{
DWORD dwStartAddress = 0;
BOOL fOldMode = SetKMode(TRUE);
PTHREAD pTh = HandleToThread((HANDLE)dwThreadId);
if (pTh)
{
dwStartAddress = (DWORD)MapPtrToProcess((LPVOID)pTh->dwStartAddr, pTh->pOwnerProc->hProc);
}
return dwStartAddress;
}