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

            Xiao.Zhu C++

            Xiao.Zhu C++

              C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              29 隨筆 :: 14 文章 :: 17 評論 :: 0 Trackbacks
            截獲API是個很有用的東西,比如你想分析一下別人的程序是怎樣工作的。這里我介紹一下一種我自己試驗通過的方法。  
                首先,我們必須設(shè)法把自己的代碼放到目標程序的進程空間里去。Windows Hook可以幫我們實現(xiàn)這一點。SetWindowsHookEx的聲明如下:  
            HHOOK SetWindowsHookEx(  
            int idHook, // hook type  
            HOOKPROC lpfn, // hook procedure  
            HINSTANCE hMod, // handle to application instance  
            DWORD dwThreadId // thread identifier  
            );  
                具體的參數(shù)含義可以翻閱msdn,沒有msdn可謂寸步難行。  
                這里Hook本身的功能并不重要,我們使用它的目的僅僅只是為了能夠讓W(xué)indows把我們的代碼植入別的進程里去。hook Type我們?nèi)芜x一種即可,只要保證是目標程序肯定會調(diào)用到就行,這里我用的是WH_CALLWNDPROC。lpfn和hMod分別指向我們的鉤子代碼及其所在的dll,dwThreadId設(shè)為0,表示對所有系統(tǒng)內(nèi)的線程都掛上這樣一個hook,這樣我們才能把代碼放到別的進程里去。  

                之后,我們的代碼就已經(jīng)進入了系統(tǒng)內(nèi)的所有進程空間了。必須注意的是,我們只需要截獲我們所關(guān)心的目標程序的調(diào)用,因此還必須區(qū)分一下進程號。我們自己的鉤子函數(shù)中,第一次運行將進行最重要的API重定向的工作。也就是通過將所需要截獲的API的開頭幾個字節(jié)改為一個跳轉(zhuǎn)指令,使其跳轉(zhuǎn)到我們的API中來。這是最關(guān)鍵的部分。這里我想截三個調(diào)用,ws2_32.dll中的send和recv、user32.dll中的GetMessageA。  

            DWORD dwCurrentPID = 0;  
            HHOOK hOldHook = NULL;  
            DWORD pSend = 0;  
            DWORD pRecv = 0;  
            GETMESSAGE pGetMessage = NULL;  

            BYTE btNewBytes[8] = { 0x0B8, 0x0, 0x0, 0x40, 0x0, 0x0FF, 0x0E0, 0 };  
            DWORD dwOldBytes[3][2];  

            HANDLE hDebug = INVALID_HANDLE_value;  

            LRESULT CALLBACK CallWndProc( int nCode, WPARAM wParam, LPARAM lParam )  
            {  
            DWORD dwSize;  
            DWORD dwPIDWatched;  
            HMODULE hLib;  

            if( dwCurrentPID == 0 )  
            {  
            dwCurrentPID = GetCurrentProcessId();  
            HWND hwndMainHook;  
            hwndMainHook = ::FindWindow( 0, "MainHook" );  
            dwPIDWatched = ::SendMessage( hwndMainHook, (WM_USER+100), 0, 0 );  
            hOldHook = (HHOOK)::SendMessage( hwndMainHook, (WM_USER+101), 0, 0 );  

            if( dwCurrentPID == dwPIDWatched )  
            {  
            hLib = LoadLibrary( "ws2_32.dll" );  
            pSend = (DWORD)GetProcAddress( hLib, "send" );  
            pRecv = (DWORD)GetProcAddress( hLib, "recv" );  

            ::ReadProcessMemory( INVALID_HANDLE_value, (void *)pSend, (void *)dwOldBytes[0], sizeof(DWORD)*2, &dwSize );  
            *(DWORD *)( btNewBytes + 1 ) = (DWORD)new_send;  
            ::WriteProcessMemory( INVALID_HANDLE_value, (void *)pSend, (void *)btNewBytes, sizeof(DWORD)*2, &dwSize );  

            ::ReadProcessMemory( INVALID_HANDLE_value, (void *)pRecv, (void *)dwOldBytes[1], sizeof(DWORD)*2, &dwSize );  
            *(DWORD *)( btNewBytes + 1 ) = (DWORD)new_recv;  
            ::WriteProcessMemory( INVALID_HANDLE_value, (void *)pRecv, (void *)btNewBytes, sizeof(DWORD)*2, &dwSize );  

            hLib = LoadLibrary( "user32.dll" );  
            pGetMessage = (GETMESSAGE)GetProcAddress( hLib, "GetMessageA" );  
            ::ReadProcessMemory( INVALID_HANDLE_value, (void *)pGetMessage, (void *)dwOldBytes[2], sizeof(DWORD)*2, &dwSize );  
            *(DWORD *)( btNewBytes + 1 ) = (DWORD)new_GetMessage;  
            ::WriteProcessMemory( INVALID_HANDLE_value, (void *)pGetMessage, (void *)btNewBytes, sizeof(DWORD)*2, &dwSize );  

            hDebug = ::CreateFile( "C:\\Trace.log", GENERIC_WRITE, 0, 0, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0 );  
            }  
            }  

            if( hOldHook != NULL )  
            {  
            return CallNextHookEx( hOldHook, nCode, wParam, lParam );  
            }  

            return 0;  
            }  

                上面的鉤子函數(shù),只有第一次運行時有用,就是把三個函數(shù)的首8字節(jié)修改一下(實際上只需要7個)。btNewBytes中的指令實際就是  
            mov eax, 0x400000  
            jmp eax  
            這里的0x400000就是新的函數(shù)的地址,比如new_recv/new_send/new_GetMessage,此時,偷梁換柱已經(jīng)完成。再看看我們的函數(shù)中都干了些什么。以GetMessageA為例:  

            BOOL _stdcall new_GetMessage( LPMSG lpMsg, HWND hWnd, UINT wMsgFilterMin, UINT wMsgFilterMax )  
            {  
            DWORD dwSize;  
            char szTemp[256];  
            BOOL r = false;  

            //Watch here before it’s executed.  
            sprintf( szTemp, "Before GetMessage : HWND 0x%8.8X, msgMin 0x%8.8X, msgMax 0x%8.8x \r\n", hWnd, wMsgFilterMin, wMsgFilterMax );  
            ::WriteFile( hDebug, szTemp, strlen(szTemp), &dwSize, 0 );  
            //Watch over  

            // restore it at first  
            ::WriteProcessMemory( INVALID_HANDLE_value, (void *)pGetMessage, (void *)dwOldBytes[2], sizeof(DWORD)*2, &dwSize );  

            // execute it  
            r = pGetMessage( lpMsg, hWnd, wMsgFilterMin, wMsgFilterMax );  

            // hook it again  
            *(DWORD *)( btNewBytes + 1 ) = (DWORD)new_GetMessage;  
            ::WriteProcessMemory( INVALID_HANDLE_value, (void *)pGetMessage, (void *)btNewBytes, sizeof(DWORD)*2, &dwSize );  

            //Watch here after it’s executed  
            sprintf( szTemp, "Result of GetMessage is %d.\r\n", r );  
            ::WriteFile( hDebug, szTemp, strlen( szTemp ), &dwSize, 0 );  
            if( r )  
            {  
            sprintf( szTemp, "Msg : HWND 0x%8.8X, MSG 0x%8.8x, wParam 0x%8.8X, lParam 0x%8.8X\r\nTime 0x%8.8X, X %d, Y %d\r\n",  
            lpMsg->hwnd, lpMsg->message,  
            lpMsg->wParam, lpMsg->lParam, lpMsg->time,  
            lpMsg->pt.x, lpMsg->pt.y );  
            ::WriteFile( hDebug, szTemp, strlen( szTemp ), &dwSize, 0 );  
            }  
            strcpy( szTemp, "\r\n" );  
            ::WriteFile( hDebug, szTemp, strlen( szTemp ), &dwSize, 0 );  

            //Watch over  

            return r;  
            }  

                先將截獲下來的參數(shù),寫入到一個log文件中,以便分析。然后恢復(fù)原先保留下來的GetMessageA的首8字節(jié),然后執(zhí)行真正的GetMessageA調(diào)用,完畢后再將執(zhí)行結(jié)果也寫入log文件,然后將GetMessageA的執(zhí)行結(jié)果返回給調(diào)用者。  
            整個截獲的過程就是這樣。你可以把其中的寫log部分改成你自己想要的操作。這里有個不足的地方是,截獲動作是不能夠并發(fā)進行的,如果目標進程是多線程的,就會有問題。解決辦法是,可以在每次new_GetMessage中加入一個CriticalSection的鎖和解鎖,以使調(diào)用變?yōu)榇羞M行,但這個我沒有試驗過。
            posted on 2007-04-28 16:00 Xiao.Zhu 閱讀(809) 評論(0)  編輯 收藏 引用

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


            伊人久久大香线焦综合四虎| 99久久精品免费看国产免费| 国内精品免费久久影院| 国产精品久久久福利| 99久久精品影院老鸭窝| 青青青国产精品国产精品久久久久| 久久精品中文騷妇女内射| 丰满少妇高潮惨叫久久久| 久久久久久狠狠丁香| 国内精品久久久久久久久电影网| 国产精品久久久久乳精品爆| 久久99精品久久久久久水蜜桃 | 国产精品成人无码久久久久久| 情人伊人久久综合亚洲| 精品免费久久久久国产一区 | 日韩欧美亚洲综合久久| 亚洲国产成人久久综合区| 一本色道久久综合狠狠躁| 久久国产精品久久久| 一本久久免费视频| 欧美激情精品久久久久| 久久人人爽人人人人爽AV| 日本三级久久网| 午夜人妻久久久久久久久| 精品国产婷婷久久久| 国产成人精品久久免费动漫| 青青热久久国产久精品| 国产精品久久久久国产A级| 亚洲精品国产综合久久一线| 久久精品夜夜夜夜夜久久| 日韩美女18网站久久精品| 欧美激情精品久久久久| 色综合久久久久无码专区| 性高湖久久久久久久久AAAAA| 国产精品久久久久无码av| 一本久道久久综合狠狠爱| 亚洲欧美日韩精品久久亚洲区| 99久久精品国产一区二区三区 | 成人国内精品久久久久影院| 中文字幕久久精品| 人人狠狠综合久久亚洲高清|