• <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
            在一個(gè)線程出現(xiàn)異常行為時(shí),比如說(shuō)CPU占用率過(guò)高,拋出異常等,你一定想知道這個(gè)線程是由哪個(gè)模塊創(chuàng)建的。因此無(wú)論在哪個(gè)操作系統(tǒng)上,獲取線程名稱是診斷線程相關(guān)問(wèn)題的重要一步。

            從線程ID獲取線程名稱通常的方法是,先獲取該線程的入口地址,然后枚舉進(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
                            );

            獲取線程入口地址可用:

             

            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上就沒(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 */

            因此要做的就是想辦法根據(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的值是一樣的!!因此我們可以寫一個(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;
            }

            為了使用這些內(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;
            }

            Feedback

            # re: [轉(zhuǎn)載]Hacking Windows CE: 如何從線程ID獲取線程名稱   回復(fù)  更多評(píng)論   

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

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


            亚洲欧美久久久久9999| 色噜噜狠狠先锋影音久久| 一级女性全黄久久生活片免费| 久久精品综合网| 久久国产精品一国产精品金尊| 亚洲精品高清久久| 欧美精品丝袜久久久中文字幕| 亚洲精品无码久久久久去q| 国产精品免费看久久久| 国产精品九九久久免费视频| 狠狠色丁香久久婷婷综合图片| 久久精品毛片免费观看| 四虎国产精品成人免费久久| 伊人久久精品无码二区麻豆| 99热都是精品久久久久久| 亚洲精品乱码久久久久久蜜桃图片| 青青草原综合久久| 久久综合久久自在自线精品自| 久久免费99精品国产自在现线 | 99久久精品国产一区二区| 怡红院日本一道日本久久| 亚洲级αV无码毛片久久精品| 久久精品一区二区三区中文字幕| 亚洲av伊人久久综合密臀性色| 国产精品伊人久久伊人电影| 久久精品九九亚洲精品| 囯产精品久久久久久久久蜜桃| 蜜臀久久99精品久久久久久| 精品99久久aaa一级毛片| 国内精品人妻无码久久久影院| 久久人人爽人人爽人人片AV高清 | 色婷婷综合久久久久中文| 亚洲国产精品成人久久蜜臀| 亚洲乱亚洲乱淫久久| 99久久精品国内| 好久久免费视频高清| 久久国产色AV免费看| 99久久国语露脸精品国产| 国产精品一区二区久久精品| 99久久这里只有精品| 国产婷婷成人久久Av免费高清 |