• <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
            青青草国产成人久久91网| 久久精品成人免费观看97| 久久婷婷五月综合国产尤物app | 婷婷综合久久狠狠色99h| 久久99热精品| 国产精品成人久久久| 国产精品久久久久AV福利动漫| 亚洲国产精品热久久| 一本色道久久综合狠狠躁| 99久久精品国产免看国产一区| 久久免费国产精品| 国产精品美女久久久久| 一本色道久久综合| 欧美日韩精品久久免费| 青青草原精品99久久精品66| 久久国产三级无码一区二区| 国产精品禁18久久久夂久| 亚洲日本久久久午夜精品| 色噜噜狠狠先锋影音久久| 亚洲国产精品18久久久久久| 久久久网中文字幕| 99久久精品免费观看国产| 久久久久久毛片免费播放| 亚洲精品国精品久久99热| 99久久国产综合精品五月天喷水| 久久天堂AV综合合色蜜桃网| 四虎国产精品成人免费久久| 久久精品?ⅴ无码中文字幕| 国产精品久久久99| 麻豆成人久久精品二区三区免费| 久久伊人精品青青草原日本| 伊人久久免费视频| 久久久久久综合一区中文字幕| 久久久久亚洲Av无码专| 亚洲愉拍99热成人精品热久久| yy6080久久| 久久99精品久久久大学生| 欧美一区二区久久精品| 国产精品久久久久久久app | 国产精品久久久久久搜索| 亚洲国产精品无码久久SM |