• <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獲取線程名稱通常的方法是,先獲取該線程的入口地址,然后枚舉進程內(nèi)所有已加載模塊,最后判斷線程入口地址落在哪個加載模塊范圍內(nèi)。枚舉進程內(nèi)已加載模塊可用Win32標準的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手段。仔細研究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ù),比較完整的代碼如下。當然,官方肯定是不建議這么做的,但是重要的是解決問題,你說呢。

             

            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獲取線程名稱   回復  更多評論   

            2014-12-17 10:30 by 獵頭招聘
            這篇文章我找了很久,很實用,收下了。
            日本免费一区二区久久人人澡| 人妻无码αv中文字幕久久琪琪布 人妻无码精品久久亚瑟影视 | 久久精品国产色蜜蜜麻豆| 青青草国产精品久久| 久久伊人五月天论坛| 久久久青草久久久青草| 亚州日韩精品专区久久久| 精品熟女少妇av免费久久| 欧美久久久久久| 久久99精品国产99久久| 亚洲欧美精品一区久久中文字幕| 久久久人妻精品无码一区| 久久精品天天中文字幕人妻| 久久av高潮av无码av喷吹| 精品精品国产自在久久高清| 午夜精品久久久久久久久| 久久久免费观成人影院| 亚洲AV无码久久| 精品国产乱码久久久久久浪潮| 国内精品久久久久| 久久www免费人成看片| 亚洲av成人无码久久精品| 人妻无码久久精品| 久久亚洲国产中v天仙www| 久久久久人妻一区精品性色av| 77777亚洲午夜久久多人| 亚洲国产欧美国产综合久久| 香蕉99久久国产综合精品宅男自| 国内精品人妻无码久久久影院 | 欧美伊人久久大香线蕉综合69| 精品熟女少妇av免费久久| 久久精品国产只有精品2020| 1000部精品久久久久久久久| 久久精品国产久精国产思思| 国内精品伊人久久久久777| 久久人人爽人人爽人人片AV东京热| 久久国产精品二国产精品| 久久中文字幕无码专区| 一本色综合久久| 色老头网站久久网| 久久久久人妻一区精品色|