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

            雁過(guò)無(wú)痕

              弄著玩的。功能是簡(jiǎn)單的實(shí)現(xiàn)函數(shù)轉(zhuǎn)發(fā),即
                  調(diào)用CALL(func),轉(zhuǎn)為調(diào)用func(),
                  調(diào)用CALL(func, arg1, arg2) ,轉(zhuǎn)為調(diào)用func(arg1, arg2)

              代碼中,宏CALL/STDCALL分別用來(lái)調(diào)用  __cdecl/__stdcall 調(diào)用規(guī)定的函數(shù)
                         unsafe_call 兩者都可調(diào)用,但它不是多線(xiàn)程安全的。

              代碼只支持x86 32位, 除內(nèi)嵌匯編部分,盡量符合C++11標(biāo)準(zhǔn)。
                   
            原理:
                 剛進(jìn)入函數(shù)時(shí),
                 [esp]           函數(shù)返回地址
                 [esp + 4]     第一個(gè)參數(shù),即轉(zhuǎn)發(fā)函數(shù)的地址
                 [esp + 8]     第二個(gè)參數(shù),即轉(zhuǎn)發(fā)函數(shù)的的第一個(gè)參數(shù)
                 ... 
               
                  只要寫(xiě)三行匯編指令實(shí)現(xiàn)一個(gè)c_call函數(shù),就可調(diào)用轉(zhuǎn)發(fā)函數(shù)
                  pop eax                             ; eax為函數(shù)返回地址
                  xchg dword ptr[esp], eax     ; eax為轉(zhuǎn)發(fā)函數(shù)的地址,[esp]為函數(shù)返回地址
                  jmp eax
                 
                 當(dāng)轉(zhuǎn)發(fā)函數(shù)是__cdecl,即轉(zhuǎn)發(fā)函數(shù)不會(huì)調(diào)節(jié)棧,由于在c_call,pop eax,使esp多加了4,因而在調(diào)用完c_call后應(yīng)該手動(dòng)將esp值減4,保證棧平衡。
               
                當(dāng)轉(zhuǎn)發(fā)函數(shù)是__stdcall,轉(zhuǎn)發(fā)函數(shù)會(huì)調(diào)節(jié)棧,調(diào)用轉(zhuǎn)發(fā)函數(shù)完畢后,棧已經(jīng)保持平衡,因而調(diào)用c_call完畢,不應(yīng)該進(jìn)行棧指針調(diào)節(jié)。似乎將c_call的調(diào)用改為_(kāi)_stdcall即可,但實(shí)際上c_call有變長(zhǎng)參數(shù),改成__stdcall沒(méi)效果,每次調(diào)用編譯器還是會(huì)自動(dòng)生成調(diào)節(jié)棧指針代碼。因而只能每次調(diào)用完畢,編譯器給esp加了多少,就手動(dòng)減多少。(編譯器不一定會(huì)生成 call  xxxx; add esp, xx這樣的代碼,通過(guò)改函數(shù)返回地址,忽略后面的add esp, xx指令是很糟糕的做法。)
            call_redirect
            posted on 2012-08-05 21:02 flyinghearts 閱讀(1821) 評(píng)論(8)  編輯 收藏 引用 所屬分類(lèi): C++

            評(píng)論

            # re: 內(nèi)嵌匯編實(shí)現(xiàn)的函數(shù)轉(zhuǎn)發(fā) 2012-08-06 14:02 rix
            typedef int (*func_ptr)(...);
            int call_fun(func_ptr func, ...);
            #define PUSH __asm push 0
            int call_fun(func_ptr func, ...)
            {
            *((int*)(&func)-1) = (*((int*)(&func)-1))^(*((int*)(&func)));
            *((int*)(&func)) = (*((int*)(&func)-1))^(*((int*)(&func)));
            *((int*)(&func)-1) = (*((int*)(&func)-1))^(*((int*)(&func)));
            return (int)func;
            }

            #define START_CALL PUSH
            #define END_CALL

            int main (int argc, char * argv[])
            {
            START_CALL;
            call_fun((func_ptr)func_1, 1, 2);
            END_CALL;
            START_CALL;
            printf("func_2 return=%d\n", call_fun((func_ptr)func_2));
            END_CALL;
            START_CALL;
            call_fun((func_ptr)func_3, "printf this:%d, %s\n", 1, "ok");
            END_CALL;
            return 0;
            }  回復(fù)  更多評(píng)論
              

            # re: 內(nèi)嵌匯編實(shí)現(xiàn)的函數(shù)轉(zhuǎn)發(fā)[未登錄](méi) 2012-08-06 15:31 heroboy
            這樣?
            template<class T,class A1>
            T Call(T(*f)())
            {
            return f();
            }

            template<class T,class A1>
            T Call(T(*f)(A1),A1 a)
            {
            return f(a);
            }

            template<class T,class A1,class A2>
            T Call(T(*f)(A1),A1 a,A2 b)
            {
            return f(a,b);
            }
              回復(fù)  更多評(píng)論
              

            # re: 內(nèi)嵌匯編實(shí)現(xiàn)的函數(shù)轉(zhuǎn)發(fā) 2012-08-06 21:43 flyinghearts
            @rix
            這個(gè)代碼顯然是錯(cuò)的。
            call_fun只是簡(jiǎn)單返回 func的值而已。  回復(fù)  更多評(píng)論
              

            # re: 內(nèi)嵌匯編實(shí)現(xiàn)的函數(shù)轉(zhuǎn)發(fā) 2012-08-06 21:57 flyinghearts
            @heroboy
            你的模板寫(xiě)錯(cuò)了。

            用模板優(yōu)點(diǎn)不說(shuō)了,就說(shuō)說(shuō)缺點(diǎn)吧:
            如果轉(zhuǎn)發(fā)函數(shù)的參數(shù)是不定的,模板無(wú)法匹配到。
            另外,類(lèi)型匹配太嚴(yán)格了,若參數(shù)類(lèi)型是const char*, 傳入"123"字符串都要先轉(zhuǎn)下類(lèi)型,特別是參數(shù)是數(shù)字時(shí),非要查看下參數(shù)是int還是long,還是其它的。用起來(lái)太累人。



              回復(fù)  更多評(píng)論
              

            # re: 內(nèi)嵌匯編實(shí)現(xiàn)的函數(shù)轉(zhuǎn)發(fā) 2012-08-13 12:03 rix
            @flyinghearts
            自己調(diào)用下就知道了。
            *((int*)(&func)-1) = (*((int*)(&func)-1))^(*((int*)(&func)));
            *((int*)(&func)) = (*((int*)(&func)-1))^(*((int*)(&func)));
            *((int*)(&func)-1) = (*((int*)(&func)-1))^(*((int*)(&func)));
            關(guān)鍵在這個(gè)交換的地方,而不是返回值。  回復(fù)  更多評(píng)論
              

            # re: 內(nèi)嵌匯編實(shí)現(xiàn)的函數(shù)轉(zhuǎn)發(fā) 2012-08-14 21:10 flyinghearts
            @rix
            你的做法,還是避免不了要 內(nèi)嵌匯編。這個(gè)做法似乎可行,但卻存在很大問(wèn)題:過(guò)于依賴(lài)于編譯器怎么優(yōu)化。有些編譯器會(huì)認(rèn)為那段代碼在做無(wú)意義的事,直接優(yōu)化掉。你用gcc試試。

            另外,交換兩個(gè)數(shù),那樣寫(xiě)法,可讀性差,效率也差,。







              回復(fù)  更多評(píng)論
              

            # re: 內(nèi)嵌匯編實(shí)現(xiàn)的函數(shù)轉(zhuǎn)發(fā) 2012-08-17 21:25 flyinghearts
            @rix
            你那樣寫(xiě)不可取,至少要寫(xiě)成下面這樣,保證不被優(yōu)化掉:

            void* c_call(void* func, ...)
            {
            typedef void* type;
            const type tmp = func;
            volatile type& eip = func;
            volatile type& ret_addr = *(&func - 1);
            eip = ret_addr;
            ret_addr = tmp;
            return (void*)tmp;
            }

            但是這樣又存在一個(gè)大問(wèn)題:要保證調(diào)用這個(gè)函數(shù)時(shí),不會(huì)被內(nèi)聯(lián),一但被內(nèi)聯(lián),bug就來(lái)了。




              回復(fù)  更多評(píng)論
              

            九九久久99综合一区二区| 香蕉99久久国产综合精品宅男自 | 中文字幕人妻色偷偷久久| 久久久久女教师免费一区| 国产成人精品三上悠亚久久| 久久无码国产专区精品| 狠狠色综合网站久久久久久久| 久久r热这里有精品视频| 国产免费久久久久久无码| 久久毛片一区二区| 亚洲国产天堂久久综合网站| 久久综合成人网| 中文字幕久久亚洲一区| 麻豆精品久久久一区二区| 一级a性色生活片久久无少妇一级婬片免费放 | 久久精品成人国产午夜| 国产精品久久久天天影视香蕉| 精产国品久久一二三产区区别 | 精品无码久久久久久国产| 亚洲中文久久精品无码| 久久婷婷国产剧情内射白浆| 久久本道综合久久伊人| 久久精品一区二区| 国产精品美女久久久久av爽| 久久777国产线看观看精品| 亚洲欧洲久久av| 久久久久久精品久久久久| 色综合久久夜色精品国产| 久久精品国产99国产精品导航| 亚洲午夜精品久久久久久浪潮| 久久午夜综合久久| 精品多毛少妇人妻AV免费久久 | 精品久久久久久无码中文字幕| 久久综合综合久久97色| 香蕉aa三级久久毛片| 日韩人妻无码一区二区三区久久99| 免费精品久久天干天干| 99热成人精品热久久669| 久久一本综合| 久久精品国产72国产精福利| 午夜精品久久久久久中宇|