青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

posts - 54, comments - 32, trackbacks - 0, articles - 0
在一個線程出現異常行為時,比如說CPU占用率過高,拋出異常等,你一定想知道這個線程是由哪個模塊創建的。因此無論在哪個操作系統上,獲取線程名稱是診斷線程相關問題的重要一步。

從線程ID獲取線程名稱通常的方法是,先獲取該線程的入口地址,然后枚舉進程內所有已加載模塊,最后判斷線程入口地址落在哪個加載模塊范圍內。枚舉進程內已加載模塊可用Win32標準的CreateToolhelp32Snapshot/Module32First/Module32Next系列ToolHelp API得到。獲取線程入口地址則沒有線程的Win32 API可用。不過在Windows NT based操作系統上(包括Windows NT 4.0/2000/XP/2003,等),有一個未公開的Native API可用:NtQueryInformationThread。其聲明如下:

DWORD WINAPI NtQueryInformationThread(
                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;
}

在Windows CE上就沒這么幸運了,沒有任何現成的API可用。官方Windows CE Base Team的blog這個問題的回答是可以用Remote Kernel Tracker,不過這需要你build一個特殊的kernel image,enable一些profiler功能-這在顯示的問題診斷中顯然是不實際的。那么有沒有辦法不需要什么特殊的配置就可像Windows桌面操作系統那樣獲得入口地址呢?有是有的,不過需要一些hack手段。仔細研究CE下的Thread內核數據結構,就會發現Thread結構中有一項是記錄線程入口地址的。

 

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 */

因此要做的就是想辦法根據線程ID或handle得到這個數據。再研究,發現線程的Thread內核數據結構可通過句柄得到:

 

PTHREAD pTh = HandleToThread(ThreadHandle);

而且,在Windows CE下,線程ID和其handle的值是一樣的!!因此我們可以寫一個這樣的函數從線程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;
}

為了使用這些內核數據結構,我們還需要另外一些輔助結構和函數,比較完整的代碼如下。當然,官方肯定是不建議這么做的,但是重要的是解決問題,你說呢。

 

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;
}

Feedback

# re: [轉載]Hacking Windows CE: 如何從線程ID獲取線程名稱   回復  更多評論   

2014-12-17 10:30 by 獵頭招聘
這篇文章我找了很久,很實用,收下了。

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            99国产精品久久久久久久久久 | 国产欧美午夜| 欧美91大片| 国产一区二区av| 午夜精品区一区二区三| 久久国内精品视频| 激情综合色综合久久| 久久精品亚洲一区二区| 欧美大片18| 亚洲免费在线播放| 国模大胆一区二区三区| 欧美二区在线播放| a4yy欧美一区二区三区| 在线中文字幕不卡| 老司机午夜免费精品视频| 亚洲美女毛片| 国产精品久久久久久久浪潮网站| 亚洲免费在线看| 欧美成人精品在线| 亚洲欧洲日产国产网站| 亚洲伊人网站| 免费看的黄色欧美网站| 欧美一二三区精品| 亚洲级视频在线观看免费1级| 欧美日韩在线看| 久久人人爽爽爽人久久久| 亚洲美女视频在线观看| 久久久噜噜噜久噜久久 | 亚洲欧美日产图| 久久亚洲影音av资源网| 久久国产福利国产秒拍| 亚洲一区二区三区国产| 亚洲第一网站| 国产欧美日韩亚州综合| 亚洲国产成人久久| 在线成人激情视频| 一区二区三区无毛| 亚洲天堂免费在线观看视频| 最新日韩欧美| 亚洲高清不卡av| 午夜精品久久久久久久白皮肤 | 欧美国产日韩一区二区三区| 欧美午夜在线| 欧美日韩二区三区| 免费不卡在线观看av| 久久成人18免费网站| 亚洲欧美一区二区在线观看| 在线午夜精品自拍| 欧美不卡高清| 欧美日韩理论| 亚洲国产小视频| 久久精品亚洲| 亚洲一区二区三区视频播放| 欧美精品一区二区三| 欧美日韩mv| 亚洲精品日韩在线观看| 亚洲美女精品成人在线视频| 另类欧美日韩国产在线| 先锋a资源在线看亚洲| 欧美日韩一区二区在线| 99re热这里只有精品视频| 在线一区亚洲| 香港成人在线视频| 亚洲深夜福利视频| 欧美系列亚洲系列| 亚洲一级黄色片| 亚洲素人在线| 免费看的黄色欧美网站| 欧美特黄a级高清免费大片a级| 国产精品国产三级国产普通话三级| 国产精品人成在线观看免费| 国产一区亚洲一区| 久久久久久自在自线| 欧美一区二区福利在线| 欧美高清在线观看| 日韩视频一区二区三区在线播放免费观看 | 亚洲欧美日本国产专区一区| 在线亚洲激情| 国产美女精品视频| 91久久精品一区二区别| 亚洲成色www8888| 亚洲精品欧美日韩| 亚洲色诱最新| 国产精品久久久久秋霞鲁丝 | 在线播放一区| 亚洲国产成人午夜在线一区| 亚洲一区日韩| 狠狠色综合播放一区二区| 亚洲免费久久| 亚洲特色特黄| 亚洲国产精品va在线看黑人| 亚洲精品在线一区二区| 国产目拍亚洲精品99久久精品| 久久精品中文字幕免费mv| 艳女tv在线观看国产一区| 国产免费观看久久| 亚洲福利久久| 中文有码久久| 亚洲成人在线| 在线综合欧美| 亚洲国产黄色| 亚洲欧美经典视频| 亚洲精品国产视频| 久久riav二区三区| 国产三级欧美三级日产三级99| 亚洲国产精品电影| 亚洲图片激情小说| 亚洲国产欧美一区二区三区久久 | 日韩午夜在线观看视频| 狠狠色综合色区| 亚洲手机视频| 一区二区三区日韩| 免费欧美在线| 久久精品国产96久久久香蕉| 欧美精品日韩一区| 99re6热只有精品免费观看| 亚洲伊人网站| 亚洲一区二区精品在线| 免费视频久久| 久久午夜精品| 国产欧美在线播放| 亚洲特色特黄| 午夜精品美女久久久久av福利| 欧美激情性爽国产精品17p| 老司机免费视频一区二区| 国产美女诱惑一区二区| 国产精品99久久久久久久久| av成人天堂| 亚洲一区999| 99精品国产一区二区青青牛奶| 久久精品成人| 久久亚洲精选| 伊人久久婷婷色综合98网| 久久国产黑丝| 久久综合狠狠综合久久综合88 | 亚洲欧美综合精品久久成人 | 国产精品久久一级| 99国内精品| 亚洲无线视频| 欧美日韩在线不卡一区| 日韩视频国产视频| 一区二区日韩伦理片| 欧美啪啪成人vr| 一区二区三区日韩欧美| 国产精品一区二区三区乱码| 一二三四社区欧美黄| 亚洲欧美一区二区三区极速播放| 欧美日韩在线电影| 宅男噜噜噜66国产日韩在线观看| 亚洲一区高清| 久久精品免费| 久久视频在线视频| 亚洲第一精品福利| 欧美国产日韩一区二区三区| 亚洲第一精品电影| 亚洲精选一区二区| 国产精品海角社区在线观看| 久久久亚洲精品一区二区三区| 伊人成人网在线看| 欧美黑人在线播放| 亚洲精品久久久久久久久久久久| 国产精品第2页| 亚洲一区二区视频| 久久久综合免费视频| 亚洲激情av在线| 欧美日韩伦理在线| 午夜久久99| 亚洲国产高潮在线观看| 亚洲手机视频| 黄色av一区| 欧美日韩免费区域视频在线观看| 亚洲欧美精品一区| 亚洲国产第一| 欧美在线三区| 亚洲精品孕妇| 国产一区二区三区在线观看免费视频 | 夜色激情一区二区| 国产精品久久久久影院色老大 | 欧美一区二区视频97| 欧美激情va永久在线播放| 亚洲靠逼com| 久久久久高清| 亚洲图中文字幕| 亚洲国产成人在线| 国产精品入口尤物| 欧美激情第8页| 久久精品国产精品亚洲精品| 日韩视频欧美视频| 免费亚洲电影在线观看| 亚洲免费伊人电影在线观看av| 亚洲电影免费观看高清| 国产欧美日韩一区| 欧美日韩一区二区视频在线观看| 久久永久免费| 免费短视频成人日韩| 亚洲一卡久久| 99国产精品99久久久久久| 国内精品久久国产| 国产精品入口福利| 欧美日韩一区二区欧美激情|