• <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>

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

            從線程ID獲取線程名稱通常的方法是,先獲取該線程的入口地址,然后枚舉進(jìn)程內(nèi)所有已加載模塊,最后判斷線程入口地址落在哪個加載模塊范圍內(nèi)。枚舉進(jìn)程內(nèi)已加載模塊可用Win32標(biāo)準(zhǔn)的CreateToolhelp32Snapshot/Module32First/Module32Next系列ToolHelp API得到。獲取線程入口地址則沒有線程的Win32 API可用。不過在Windows NT based操作系統(tǒng)上(包括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上就沒這么幸運了,沒有任何現(xiàn)成的API可用。官方Windows CE Base Team的blog這個問題的回答是可以用Remote Kernel Tracker,不過這需要你build一個特殊的kernel image,enable一些profiler功能-這在顯示的問題診斷中顯然是不實際的。那么有沒有辦法不需要什么特殊的配置就可像Windows桌面操作系統(tǒng)那樣獲得入口地址呢?有是有的,不過需要一些hack手段。仔細(xì)研究CE下的Thread內(nèi)核數(shù)據(jù)結(jié)構(gòu),就會發(fā)現(xiàn)Thread結(jié)構(gòu)中有一項是記錄線程入口地址的。

             

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

            因此要做的就是想辦法根據(jù)線程ID或handle得到這個數(shù)據(jù)。再研究,發(fā)現(xiàn)線程的Thread內(nèi)核數(shù)據(jù)結(jié)構(gòu)可通過句柄得到:

             

            PTHREAD pTh = HandleToThread(ThreadHandle);

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

            為了使用這些內(nèi)核數(shù)據(jù)結(jié)構(gòu),我們還需要另外一些輔助結(jié)構(gòu)和函數(shù),比較完整的代碼如下。當(dāng)然,官方肯定是不建議這么做的,但是重要的是解決問題,你說呢。

             

            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: [轉(zhuǎn)載]Hacking Windows CE: 如何從線程ID獲取線程名稱   回復(fù)  更多評論   

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

            只有注冊用戶登錄后才能發(fā)表評論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            国产91久久精品一区二区| 久久精品亚洲AV久久久无码| 99久久国产热无码精品免费 | 久久www免费人成看国产片| 久久久精品免费国产四虎| 久久福利青草精品资源站免费| 99久久精品国产一区二区三区 | 办公室久久精品| 久久久噜噜噜久久| 亚洲欧美伊人久久综合一区二区| 精品国产VA久久久久久久冰| 久久国产精品免费| 久久久久人妻精品一区 | 久久免费视频一区| 久久久久亚洲AV无码网站| 久久亚洲视频| 亚洲国产精品婷婷久久| 亚洲国产精品成人久久| 久久久久久噜噜精品免费直播| 久久99精品久久久久子伦| 欧美久久久久久| 久久精品国产99久久久香蕉| 色88久久久久高潮综合影院| 美女久久久久久| 久久综合综合久久97色| 亚洲AV乱码久久精品蜜桃| 久久精品夜色噜噜亚洲A∨| 97精品国产97久久久久久免费| 国内精品久久人妻互换| 中文字幕日本人妻久久久免费 | 91久久香蕉国产熟女线看| 久久精品无码专区免费东京热| 99久久国产综合精品女同图片| 久久久久久国产精品无码下载 | 四虎国产永久免费久久| 996久久国产精品线观看| 99久久99久久久精品齐齐| 久久无码人妻一区二区三区| 久久影院综合精品| 久久久久亚洲精品天堂| 国产∨亚洲V天堂无码久久久|