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

posts - 33,  comments - 33,  trackbacks - 0

 編譯過的二進制代碼本身是一種數據結構,在代碼被載入內存執行的時候,由操作系統對這種數據結構進行操作,具體到Win32平臺,就是所謂的PE文件頭。
 對于Windows系統,源代碼是如何轉化為二進制代碼?全局變量存儲在什么地方,如何初始化?共享變量是如何工作?理解PE文件格式能更好地理解上面的問題.舉個例子,我們使用C++編寫源代碼,這些源代碼被編譯器翻譯成obj格式的目標文件,每一個目標文件都包含著全局變量、常量數據、資源、可執行代碼、用于鏈接的符號名以及調試信息。模塊的目標文件通過鏈接器與庫鏈接在一起,而庫自身是一種將目標文件組合在一起的格式。此外,鏈接器將所有已經初始化的全局變量合并到一個段中,將所有未初始化的全局變量合并到另一個段中,再將所有的執行代碼放入到另一個段中等。為什么要將目標文件的不同部分進行分組分段呢?主要是保護資源和優化資源的使用。例如常數和可執行代碼是只讀的,如果操作系統檢測到對這部分內存進行寫操作時,就會發出錯誤。又例如,所有的Win32程序都是用同一個gdi32.dll,這樣就優化了內存的使用,gdi32.dll的可執行代碼段自在內存中存儲一次。
 PE文件格式

PE格式保留了多個目錄,通常可以看到的目錄有導入、綁定導入、延遲導入、到處、重定位、資源和調試信息。將這些段和目錄組合在一起,

加上兩個頭,就成為了PE文件。
 -----------------
|IMAGE_DOS_HEADER | : 兼容DOS程序運行
|-----------------|
|DOS 存根程序     | : 小程序,用于顯示錯誤信息的軟件中斷然后退出,
|-----------------|
|PE Header        | : 真正的PE頭
|-----------------|
|Section Table    |  : 段表
|-----------------|
|Section 1        |
|-----------------|
|Section 2        |
|-----------------|
|Section ...      |
|-----------------|
|Section n        |
 --------------
對于DOS存根程序,由于存根程序長度不定,所以要DOS頭的e_lfanew確定偏移。
段的名稱和作用如下:
節名   作用
.arch  最初的構建信息(Alpha Architecture Information)
.bss   未經初始化的數據
.CRT   C運行期只讀數據
.data   已經初始化的數據
.debug   調試信息
.didata  延遲輸入文件名表
.edata  導出文件名表
.idata  導入文件名表
.pdata      異常信息(Exception Information)
.rdata  只讀的初始化數據
.reloc  重定位表信息
.rsrc  資源
.text   .exe或.dll文件的可執行代碼
.tls  線程的本地存儲器
.xdata  異常處理表

PE文件存儲的地址有些以虛擬地址的形式存儲,不過一旦其相應模塊被加載,它們就可以使用。下面編寫一個簡單的PEFile的C++類,來說明
PE文件的使用。
我們要實現一個小小的鉤子程序,將MessageBoxA改成我們自己設定的程序中去,由于程序比較簡單,不再贅述,詳見程序中的注釋。
PEFile.h

 

#ifndef PE_FILE_H
#define PE_FILE_H

class PEFile
{
public:
    PEFile(HMODULE _hModule);
    
const void* getDirectory(int _id);
    PIMAGE_IMPORT_DESCRIPTOR getImportDescriptor(LPCSTR _pDllName);
    
const unsigned * getFunctionPtr(PIMAGE_IMPORT_DESCRIPTOR _pImport,LPCSTR _pProcName);
    FARPROC setImportAddress(LPCSTR _pDllName,LPCTSTR _pProcName,FARPROC _pNewProc);
    FARPROC setExportAddress(LPCSTR _pProcName,FARPROC _pNewProc);
private:
    
const char* m_pModule;//模塊
    PIMAGE_DOS_HEADER m_pDOSHeader;//DOS頭
    PIMAGE_NT_HEADERS m_pNTHeader;//NT頭
}
;
#endif


PEFile.cpp

#include <Windows.h>
#include 
"PEFile.h"

PEFile::PEFile(HMODULE _hModule)
{
    m_pModule 
= (const char*)_hModule;
    
if (IsBadReadPtr(m_pModule,sizeof(IMAGE_DOS_HEADER)))
    
{
        m_pDOSHeader 
= 0;
        m_pNTHeader 
= 0;
    }

    
else
    
{
        m_pDOSHeader 
= (PIMAGE_DOS_HEADER)m_pModule;
        
if (IsBadReadPtr(m_pModule + m_pDOSHeader->e_lfanew,sizeof(IMAGE_NT_HEADERS)))
        
{
            m_pNTHeader 
= 0;
        }

        
else
        
{
            m_pNTHeader 
= (PIMAGE_NT_HEADERS)(m_pModule + m_pDOSHeader->e_lfanew);
        }

    }

}


const void* PEFile::getDirectory(int _id)
{
    
return m_pModule + (m_pNTHeader->OptionalHeader.DataDirectory[_id].VirtualAddress);
}


//返回引入的模塊
PIMAGE_IMPORT_DESCRIPTOR PEFile::getImportDescriptor(LPCSTR _pDllName)
{
    PIMAGE_IMPORT_DESCRIPTOR pImport 
= (PIMAGE_IMPORT_DESCRIPTOR)getDirectory(IMAGE_DIRECTORY_ENTRY_IMPORT);
    
if (pImport == 0)
    
{
        
return 0;
    }

    
//一個一個查找
    while(pImport->FirstThunk)
    
{
        
if (stricmp(_pDllName,m_pModule + pImport->Name) == 0)
        
{
            
return pImport;
        }

        
++pImport;
    }

    
return 0;
}


//返回引入的函數
const unsigned * PEFile::getFunctionPtr(PIMAGE_IMPORT_DESCRIPTOR _pImport,LPCSTR _pProcName)
{
    PIMAGE_THUNK_DATA pThunk 
= (PIMAGE_THUNK_DATA)(m_pModule + _pImport->OriginalFirstThunk);
    
//一個一個查找
    for (int i = 0; pThunk->u1.Function ; ++i)
    
{
        
bool isMatch = false;
        
if (pThunk->u1.Ordinal & 0x80000000)
        
{
            isMatch 
= (pThunk->u1.Ordinal & 0xFFFF== ((DWORD)_pProcName);
        }

        
else
        
{
            isMatch 
= stricmp(_pProcName,(m_pModule + ((unsigned)pThunk->u1.AddressOfData +2 ))) == 0;
        }


        
if (isMatch)
        
{
            
return ((unsigned *)(m_pModule + _pImport->FirstThunk)) + i;
        }

        
++pThunk;
    }

    
return 0;
}


//設定導入的函數地址
FARPROC PEFile::setImportAddress(LPCSTR _pDllName,LPCTSTR _pProcName,FARPROC _pNewProc)
{
    PIMAGE_IMPORT_DESCRIPTOR pImport 
= getImportDescriptor(_pDllName);
    
if (pImport)
    
{
        
//拿到某模塊的某函數地址
        const unsigned* pfn = getFunctionPtr(pImport,_pProcName);
        
if (IsBadReadPtr(pfn,sizeof(DWORD)))
        
{
            
return 0;
        }

        FARPROC oldproc 
= (FARPROC)*pfn;

        
//寫入自定的函數地址
        DWORD dwWritten;
        WriteProcessMemory(GetCurrentProcess(),(
void*)pfn,&_pNewProc,sizeof(DWORD),&dwWritten);
        
return oldproc;
    }

    
return 0;
}


//導出目錄
FARPROC PEFile::setExportAddress(LPCSTR _pProcName,FARPROC _pNewProc)
{
    PIMAGE_EXPORT_DIRECTORY pExport 
= (PIMAGE_EXPORT_DIRECTORY)getDirectory(IMAGE_DIRECTORY_ENTRY_EXPORT);
    
if (pExport == 0)
    
{
        
return 0;
    }

    unsigned ord 
= 0;
    
if ((unsigned)_pProcName < 0xFFFF)
    
{
        ord 
= (unsigned)_pProcName;
    }

    
else
    
{
        
const DWORD* pNames = (const DWORD*)(m_pModule + pExport->AddressOfNames);
        
const WORD* pOrds = (const WORD*)(m_pModule + pExport->AddressOfNameOrdinals);
        
for (unsigned i = 0; i < pExport->AddressOfNames; ++i)
        
{
            
if (stricmp(_pProcName,m_pModule + pNames[i]) == 0)
            
{
                ord 
= pExport->Base + pOrds[i];
                
break;
            }

        }

    }

    
if ( (ord < pExport->Base) || (ord > pExport->NumberOfFunctions))
    
{
        
return 0;
    }

    DWORD 
*pRVA = ((DWORD*)(m_pModule + pExport->AddressOfFunctions)) + ord - pExport->Base;
    DWORD rslt 
= *pRVA;
    DWORD dwWritten 
= 0;
    DWORD newRVA 
= (DWORD)_pNewProc - (DWORD)m_pModule;
    WriteProcessMemory(::GetCurrentProcess(),pRVA,
&newRVA,sizeof(DWORD),&dwWritten);
    
return (FARPROC)(m_pModule + rslt);
}


測試代碼:

#include <Windows.h>
#include 
"PEFile.h"


int WINAPI MyMessageBoxA(HWND hWnd,LPCSTR pText,LPCSTR pCaption,UINT uType)
{
    
char buf1[128];
    
char buf2[128];
    strcpy(buf1,pCaption);
    strcat(buf1,
"  -  catch!");

    strcpy(buf2,pText);
    strcat(buf2,
"  -  catch!");

    
return MessageBoxExA(hWnd,buf2,buf1,uType,0);
}


int WINAPI WinMain( __in HINSTANCE hInstance, __in_opt HINSTANCE hPrevInstance, __in_opt LPSTR lpCmdLine, __in int nShowCmd )
{
    
{
        PEFile pe(hInstance);
        pe.setImportAddress(
"user32.dll","MessageBoxA",(FARPROC)MyMessageBoxA);
        MessageBoxA(
0,"Test","SetImportAddresss",MB_OK);
    }

    HMODULE hUser 
= GetModuleHandle("user32.dll");
    PEFile user32(hUser);
    FARPROC oldproc 
= GetProcAddress(hUser,"MessageBoxA");
    user32.setExportAddress(
"MessageBoxA",(FARPROC)MyMessageBoxA);
    FARPROC newproc 
= GetProcAddress(hUser,"MessageBoxA");
    
char temp[64];
    wsprintf(temp, 
"GetProcAddress(MessageBoxA)\n"
        
"changes from %x to %x", oldproc, newproc);
    MessageBoxA(NULL, temp, 
"SetExportAddress", MB_OK);
    
return 0;
}


運行一下可以發現,MessageBoxA已經變成自定義的函數,并且使用導出目錄的方法導出函數,GetProcAddress將返回新函數的地址,將來DLL載入此進程時也會和新函數鏈接。



 

posted on 2011-09-02 20:18 bennycen 閱讀(2392) 評論(2)  編輯 收藏 引用 所屬分類: Windows Programming

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   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>
            欧美在线一区二区三区| 一区福利视频| 欧美成人午夜视频| 国产欧美日韩视频在线观看| 亚洲国产视频a| 狠狠综合久久av一区二区老牛| 亚洲全部视频| 亚洲国产精品久久久久婷婷老年| 性欧美暴力猛交69hd| 亚洲综合国产精品| 欧美日韩在线播放| 亚洲精品一品区二品区三品区| 永久久久久久| 欧美在线观看一区二区| 午夜精品一区二区在线观看| 欧美精品粉嫩高潮一区二区| 亚洲国产一区二区三区青草影视| 伊人影院久久| 久久久久国产精品午夜一区| 久久香蕉精品| 亚洲高清视频在线观看| 久久亚洲综合色| 麻豆91精品| 亚洲国产精品99久久久久久久久| 久久久久久久综合| 欧美成人a视频| 亚洲国产成人精品久久| 欧美a级一区二区| 最新国产の精品合集bt伙计| 最新高清无码专区| 欧美日韩伦理在线免费| 99国产精品国产精品久久| 亚洲一区二区三区涩| 国产精品美女久久福利网站| 午夜精品福利在线观看| 久久久久国产成人精品亚洲午夜| 国内精品久久久久影院优| 久久精品综合一区| 欧美激情区在线播放| 日韩午夜激情| 国产精品久久久久7777婷婷| 午夜精品一区二区在线观看| 久久综合一区二区| 日韩一区二区精品视频| 国产精品v日韩精品| 香港久久久电影| 欧美成人激情在线| av成人免费在线| 国产精品色网| 久久久国际精品| 亚洲人午夜精品| 欧美在线1区| 亚洲国产精品va| 国产精品多人| 久久精品国产一区二区三 | 一区在线观看| 欧美69视频| 中文亚洲欧美| 欧美不卡在线| 亚洲免费综合| 亚洲成人影音| 欧美午夜精彩| 欧美 日韩 国产 一区| 中文精品视频| 亚洲大片在线| 久久精品欧洲| av不卡在线观看| 在线免费高清一区二区三区| 欧美日韩在线播放三区| 久久久午夜电影| 亚洲午夜久久久久久久久电影院| 欧美成人亚洲成人| 久久岛国电影| 亚洲一区在线免费| 亚洲人成网站色ww在线| 国产日韩在线一区| 欧美色欧美亚洲另类二区| 久久久久久久波多野高潮日日| 一区二区三区av| 亚洲人成人一区二区在线观看| 久久九九全国免费精品观看| 亚洲一区二区免费视频| 日韩亚洲欧美在线观看| 伊人精品视频| 国内一区二区在线视频观看| 欧美日韩亚洲国产一区| 另类亚洲自拍| 久久精品国产精品亚洲综合| 亚洲欧美中文另类| 在线一区欧美| 亚洲九九九在线观看| 亚洲国产色一区| 亚洲电影av在线| 免费欧美日韩| 欧美jizzhd精品欧美巨大免费| 久久久久久久一区二区三区| 午夜精品福利一区二区三区av | 99精品欧美一区二区三区综合在线 | 亚洲电影在线播放| 欧美国产日韩一区二区在线观看| 久久久久高清| 久久婷婷一区| 久久综合福利| 欧美成人午夜激情视频| 美女亚洲精品| 亚洲春色另类小说| 亚洲高清二区| 91久久午夜| 亚洲欧洲一区二区三区| 亚洲精品视频在线观看网站| 亚洲人成高清| 一区二区精品国产| 亚洲女人av| 久久精品99无色码中文字幕| 久久xxxx| 久久午夜羞羞影院免费观看| 久久综合久久88| 欧美成人在线免费观看| 欧美精品久久天天躁| 欧美日韩国产在线播放| 国产精品久久久久aaaa| 国产日韩欧美高清免费| 国产综合香蕉五月婷在线| 1000部精品久久久久久久久| 在线视频国产日韩| 99视频一区二区三区| 亚洲欧美国内爽妇网| 久久久www| 欧美好骚综合网| 99精品视频网| 欧美呦呦网站| 欧美激情一区在线| 国产精品拍天天在线| 国内一区二区三区| 亚洲三级免费| 欧美资源在线观看| 欧美激情1区2区| 亚洲网在线观看| 久久夜色精品一区| 欧美日韩国产丝袜另类| 国产一区二区三区四区三区四| 亚洲欧洲在线观看| 亚洲欧美日韩成人| 美日韩精品免费| 在线午夜精品| 牛夜精品久久久久久久99黑人 | 韩国精品久久久999| 日韩视频免费在线观看| 久久爱www.| 日韩亚洲精品电影| 久久精品人人做人人爽电影蜜月 | 国产一二三精品| aa级大片欧美三级| 免费影视亚洲| 午夜精品电影| 欧美日韩一区二区三区四区五区 | 亚洲综合色网站| 欧美激情综合在线| 国内综合精品午夜久久资源| 亚洲午夜精品视频| 亚洲国产专区校园欧美| 久久电影一区| 国产精品一区一区三区| 99re在线精品| 亚洲成人自拍视频| 久久精品国产欧美激情| 国产精品福利片| 99ri日韩精品视频| 亚洲电影免费观看高清| 久久精品免费播放| 国产亚洲欧美日韩一区二区| 亚洲视频高清| 亚洲理论电影网| 欧美成人日韩| 亚洲大片av| 美女亚洲精品| 久久亚洲精品网站| 国内精品一区二区三区| 久久精品国产综合| 午夜精品一区二区三区在线| 欧美午夜www高清视频| 夜夜狂射影院欧美极品| 亚洲国产欧美一区二区三区同亚洲| 久久久天天操| 1024国产精品| 欧美激情精品久久久久久蜜臀 | 极品少妇一区二区三区| 久久精品国产精品亚洲精品| 亚洲一区二区三区高清不卡| 国产精品久久久久久久久久免费看| 国产精品99久久久久久www| 亚洲精品日韩精品| 欧美日韩国产一区精品一区 | 亚洲精品免费电影| 欧美国产三区| 欧美精品午夜| 亚洲一区在线播放| 亚洲一二三区精品| 国产亚洲二区| 久久手机精品视频|