• <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>
            隨筆-91  評論-137  文章-0  trackbacks-0
            1.首先我們必須知道C語言的調用約定為__cdecl(即參數從右向左依次進棧,由調用者還原堆棧).
            2.一條push指令最多壓入4個字節,當不足4個字節時應補齊4個字節,超過4個字節時應該由低位到高位依次壓棧.
            3.pop指令也和push一樣一次只能彈出4個字節.
            4.我們需要一個CallStruct類型來儲存一個參數.
             1 class CallStruct
             2     {
             3     public:
             4         INT            Integer;
             5         BOOL        Bool;
             6         DOUBLE        Real;
             7         WCHAR        String[MAX_STRING];
             8         
             9         enum TYPE
            10         {
            11             ctInteger,
            12             ctString,
            13             ctBool,
            14             ctReal,
            15             ctVoid,
            16         }Type;
            17         
            18         CallStruct() : Integer(0),Bool(FALSE),Real(0)
            19         {
            20             String[0]    = 0;
            21         }
            22 
            23         CallStruct(const NAutoPtr<VirtualMachine::VarClass>& Var) : Integer(Var->Integer),Bool(Var->Bool),Real(Var->Real),Type((TYPE)Var->Type)
            24         {
            25             if(Type == ctString) wcscpy(String,Var->String);
            26         }
            27     };
            5.我們需要一個列表來存放參數和一個返回值對象來存放返回值.
            1     List<NAutoPtr<CallStruct>> VarList;
            2     NAutoPtr<CallStruct> Return;
            6.最后我們需要一個HMODULE和一個FARPROC分別存放dll的句柄和函數地址.
            1     HMODULE hModule;
            2     FARPROC FunctionPtr;
            7.然后我們添加幾個功能函數.
            1     BOOL AddVar(NAutoPtr<VirtualMachine::VarClass>& Var);
            2     BOOL SetReturnType(CallStruct::TYPE Type);
            3     BOOL SetLibName(LPTSTR Name);
            4     BOOL SetFunctionName(LPTSTR Name);
            注意:GetProcAddress第二個參數只接受LPCSTR類型的字符串,應此如果傳入的是Unicode編碼的字符必須將其轉換成ANSI的.
            8.我們添加一個函數Run用于調用函數.
             1 BOOL CallMacro::Run()
             2 {
             3     if(FunctionPtr == 0 || Return.Buffer() == 0return FALSE;
             4     union RealStruct
             5     {
             6         double Real;
             7         struct
             8         {
             9             int Head,Tail;
            10         };
            11     };
            12     NAutoPtr<CallStruct> cs;
            13     int Integer;
            14     BOOL Bool;
            15     RealStruct Real; // Push指令一次只能壓入4字節
            16     LPTSTR String;
            17 
            18     int iEsp;
            19     __asm mov int ptr[iEsp],esp; // 保存esp
            20     for(int i=0;i<VarList.Size();i++)
            21     {
            22         cs = VarList[i];
            23         Integer = cs->Integer;
            24         Bool = cs->Bool;
            25         Real.Real = cs->Real;
            26         String = cs->String;
            27         switch(cs->Type)
            28         {
            29         case CallStruct::ctInteger:
            30             __asm push Integer;
            31             break;
            32         case CallStruct::ctString:
            33             __asm push String;
            34             break;
            35         case CallStruct::ctBool:
            36             __asm push Bool;
            37             break;
            38         case CallStruct::ctReal:
            39             __asm push Real.Tail;
            40             __asm push Real.Head;
            41             break;
            42         }
            43     }
            44     FARPROC proc = FunctionPtr;
            45     int Head,Tail;
            46     __asm
            47     {
            48         call proc
            49         mov int ptr[Head],edx
            50         mov int ptr[Tail],eax
            51     }
            52     switch(Return->Type)
            53     {
            54     case CallStruct::ctInteger:
            55         Return->Integer = Tail;
            56         break;
            57     case CallStruct::ctString:
            58         wcscpy(Return->String,(LPCTSTR)Tail);
            59         break;
            60     case CallStruct::ctBool:
            61         Return->Bool = Tail;
            62         break;
            63     case CallStruct::ctReal:
            64         __asm fstp [Real.Real];
            65         Return->Real = Real.Real;
            66         break;
            67     }
            68     // __declspec調用約定,需要手工還原堆棧
            69     __asm mov esp,int ptr[iEsp];
            70     return TRUE;
            71 }
            Run函數首先檢查是否已經裝載了DLL并獲得了函數地址,以及返回值類型是否已經定義.
            然后根據類型依次將函數壓棧.
            然后調用call指令并保存返回值.(這里需要注意的是當返回值類型為double或float類型時必須使用fstp指令從FPU寄存器棧的棧頂的值取出來)
            最后還原堆棧.

            然后我們來測試一下.
            創建一個名為TestDLL的DLL工程并添加函數Test.
            1 TESTDLL_API double Test(double d,double* d1,WCHAR* lpBuffer)
            2 {
            3     if(d == 123.456789) MessageBox(0,lpBuffer,L"",0);
            4     *d1 = 789.654;
            5     return 77777;
            6 }

            然后創建一個測試工程,添加相關文件.
            在_tmain中添加以下代碼.
             1 int _tmain(int argc, _TCHAR* argv[])
             2 {
             3     double d;
             4     CallMacro cm;
             5     NAutoPtr<VirtualMachine::VarClass> Var;
             6 
             7     Var = new VirtualMachine::VarClass;
             8     Var->Type = VirtualMachine::VarClass::vtString;
             9     wcscpy(Var->String,L"aaaaa");
            10     cm.AddVar(Var);
            11 
            12     Var = new VirtualMachine::VarClass;
            13     Var->Type = VirtualMachine::VarClass::vtInteger;
            14     Var->Integer = (INT)&d;
            15     cm.AddVar(Var);
            16 
            17     Var = new VirtualMachine::VarClass;
            18     Var->Type = VirtualMachine::VarClass::vtReal;
            19     Var->Real = 123.456789;
            20     cm.AddVar(Var);
            21 
            22     cm.SetLibName(L"TestDll.dll");
            23     cm.SetFunctionName(L"Test");
            24     cm.SetReturnType(CallMacro::CallStruct::ctReal);
            25     cm.Run();
            26 
            27     wprintf(L"%f %f\n",d,cm.Return->Real);
            28     system("pause");
            29     return 0;
            30 }

            編譯并運行,可以看到Test函數調用成功,并成功輸出d的值和返回值.

            最后給出完整代碼.
            posted on 2011-02-06 22:21 lwch 閱讀(2645) 評論(0)  編輯 收藏 引用 所屬分類: NScript
            久久99精品久久久久久水蜜桃 | 青青草原综合久久大伊人精品| 欧美黑人又粗又大久久久| 欧美噜噜久久久XXX| 国产激情久久久久影院| 伊人久久亚洲综合影院| 国产精品青草久久久久婷婷 | 久久国产免费直播| 国产99久久精品一区二区| 久久精品亚洲乱码伦伦中文| 久久免费的精品国产V∧| 精品多毛少妇人妻AV免费久久| 超级97碰碰碰碰久久久久最新| 久久精品亚洲日本波多野结衣 | 久久亚洲精品无码AV红樱桃| 久久精品无码一区二区日韩AV| 国产精品九九九久久九九| 无码八A片人妻少妇久久| 国产免费久久精品丫丫| 久久久久99精品成人片直播| 久久精品中文字幕大胸| 日本道色综合久久影院| …久久精品99久久香蕉国产| 久久免费看黄a级毛片| 一本久久综合亚洲鲁鲁五月天| 9999国产精品欧美久久久久久| 色8久久人人97超碰香蕉987| 久久免费看黄a级毛片| 三级片免费观看久久| 国产精品伊人久久伊人电影 | 久久久久久久91精品免费观看| 91精品国产高清久久久久久91| 久久精品毛片免费观看| 亚洲日本va中文字幕久久| 亚洲午夜精品久久久久久app| 久久福利片| 久久久久国产亚洲AV麻豆| 久久亚洲国产精品123区| 久久国产免费直播| 武侠古典久久婷婷狼人伊人| 亚洲国产精品综合久久一线 |