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

雁過無痕

  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理 ::
  弄著玩的。功能是簡單的實現函數轉發,即
      調用CALL(func),轉為調用func(),
      調用CALL(func, arg1, arg2) ,轉為調用func(arg1, arg2)

  代碼中,宏CALL/STDCALL分別用來調用  __cdecl/__stdcall 調用規定的函數
             unsafe_call 兩者都可調用,但它不是多線程安全的。

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

評論

# re: 內嵌匯編實現的函數轉發 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;
}  回復  更多評論
  

# re: 內嵌匯編實現的函數轉發[未登錄] 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);
}
  回復  更多評論
  

# re: 內嵌匯編實現的函數轉發 2012-08-06 21:43 flyinghearts
@rix
這個代碼顯然是錯的。
call_fun只是簡單返回 func的值而已。  回復  更多評論
  

# re: 內嵌匯編實現的函數轉發 2012-08-06 21:57 flyinghearts
@heroboy
你的模板寫錯了。

用模板優點不說了,就說說缺點吧:
如果轉發函數的參數是不定的,模板無法匹配到。
另外,類型匹配太嚴格了,若參數類型是const char*, 傳入"123"字符串都要先轉下類型,特別是參數是數字時,非要查看下參數是int還是long,還是其它的。用起來太累人。



  回復  更多評論
  

# re: 內嵌匯編實現的函數轉發 2012-08-13 12:03 rix
@flyinghearts
自己調用下就知道了。
*((int*)(&func)-1) = (*((int*)(&func)-1))^(*((int*)(&func)));
*((int*)(&func)) = (*((int*)(&func)-1))^(*((int*)(&func)));
*((int*)(&func)-1) = (*((int*)(&func)-1))^(*((int*)(&func)));
關鍵在這個交換的地方,而不是返回值。  回復  更多評論
  

# re: 內嵌匯編實現的函數轉發 2012-08-14 21:10 flyinghearts
@rix
你的做法,還是避免不了要 內嵌匯編。這個做法似乎可行,但卻存在很大問題:過于依賴于編譯器怎么優化。有些編譯器會認為那段代碼在做無意義的事,直接優化掉。你用gcc試試。

另外,交換兩個數,那樣寫法,可讀性差,效率也差,。







  回復  更多評論
  

# re: 內嵌匯編實現的函數轉發 2012-08-17 21:25 flyinghearts
@rix
你那樣寫不可取,至少要寫成下面這樣,保證不被優化掉:

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;
}

但是這樣又存在一個大問題:要保證調用這個函數時,不會被內聯,一但被內聯,bug就來了。




  回復  更多評論
  

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲欧美日韩中文播放| 久久精品国产一区二区三区免费看| 久久久国产午夜精品| 亚洲欧洲一区| 亚洲欧美网站| 亚洲伦理久久| 另类亚洲自拍| 欧美一区二区黄色| 久久久国产精品一区二区三区| 在线电影国产精品| 久久综合给合| 米奇777超碰欧美日韩亚洲| 亚洲美女性视频| 亚洲主播在线| 亚洲国产成人久久综合一区| 最近看过的日韩成人| 亚洲伊人久久综合| 亚洲欧美综合精品久久成人 | 另类欧美日韩国产在线| 欧美va天堂va视频va在线| 亚洲影院在线观看| 久久午夜视频| 性久久久久久| 欧美电影免费观看大全| 亚洲欧美一区二区原创| 久久综合色影院| 亚洲影院色无极综合| 久久一区欧美| 先锋资源久久| 欧美欧美天天天天操| 美女国内精品自产拍在线播放| 国产精品电影观看| 亚洲东热激情| 亚洲人成亚洲人成在线观看图片| 国产精品免费久久久久久| 老司机免费视频一区二区| 欧美日韩第一区| 久久这里只有| 国产精一区二区三区| 亚洲人成小说网站色在线| 国内外成人免费视频| 宅男噜噜噜66一区二区| 亚洲乱码精品一二三四区日韩在线 | 亚洲影视九九影院在线观看| 91久久国产综合久久91精品网站| 午夜久久tv| 欧美日韩精品久久| 欧美三级欧美一级| 欧美黄污视频| 韩国av一区二区三区| 中文亚洲视频在线| 99精品国产在热久久| 久久综合九色综合欧美狠狠| 久久电影一区| 国产精品三级久久久久久电影| 亚洲精品乱码久久久久久久久 | 国产精品综合视频| 亚洲午夜精品17c| 中日韩美女免费视频网址在线观看| 麻豆国产精品一区二区三区| 久热这里只精品99re8久| 国产一区二区三区奇米久涩| 午夜免费电影一区在线观看| 午夜精品区一区二区三| 国产精品国产精品| 亚洲影视在线播放| 久久精品九九| 狠狠久久婷婷| 久久综合久久久| 亚洲第一页中文字幕| 欧美激情女人20p| 欧美高清日韩| 亚洲激情视频在线| 99国产一区| 欧美日韩精品国产| 亚洲一二三区精品| 久久精品国产69国产精品亚洲| 国产精品一区二区男女羞羞无遮挡 | 午夜在线精品偷拍| 国产亚洲精品bv在线观看| 欧美在线3区| 免费在线一区二区| 一区二区久久久久久| 欧美视频网址| 亚洲欧美制服中文字幕| 老司机精品视频一区二区三区| 亚洲激情国产| 欧美午夜精品久久久久久久| 午夜精品理论片| 欧美高清免费| 亚洲欧美综合网| 国内精品久久久久影院色| 欧美大尺度在线| 在线综合亚洲欧美在线视频| 久久久亚洲精品一区二区三区| 亚洲精品综合| 国产视频在线观看一区二区| 欧美aaaaaaaa牛牛影院| 亚洲美女啪啪| 久久久综合视频| 亚洲无限乱码一二三四麻| 国产日韩亚洲欧美精品| 国户精品久久久久久久久久久不卡| 免费一级欧美片在线播放| 亚洲第一页中文字幕| 国产精品对白刺激久久久| 香蕉久久夜色| 亚洲精品久久久久久久久久久 | 国产一区二区三区免费在线观看| 欧美成人第一页| 亚洲欧美日韩国产综合精品二区| 欧美电影免费观看高清| 亚洲手机成人高清视频| 亚洲高清激情| 国产午夜精品麻豆| 欧美日韩精品免费在线观看视频| 久久九九久精品国产免费直播| 亚洲精品久久久久久久久久久| 另类图片国产| 欧美一级大片在线免费观看| 999亚洲国产精| 国产一区二三区| 国产精品久久久久aaaa| 欧美电影免费观看| 久久视频在线视频| 久久成人免费电影| 亚洲欧美日韩精品久久奇米色影视| 亚洲国产高清在线| 欧美xxx成人| 久久欧美肥婆一二区| 欧美伊人久久| 午夜欧美视频| 午夜宅男久久久| 亚洲一区视频在线| 在线综合亚洲| 99国产精品一区| 99国产精品久久久久老师| 亚洲美女福利视频网站| 亚洲精品免费在线| 国产一区二区在线观看免费| 亚洲午夜伦理| 99视频一区二区三区| 亚洲精品乱码久久久久久蜜桃91| 狠狠色丁香久久婷婷综合丁香| 国产欧美va欧美不卡在线| 国产精品女主播一区二区三区| 欧美三级免费| 国产精品萝li| 国产亚洲人成a一在线v站| 国产免费观看久久| 国产日韩欧美制服另类| 国产欧美日韩不卡免费| 国产欧美一区二区精品秋霞影院| 国产日韩欧美一区二区三区四区 | 免费黄网站欧美| 免费视频一区| 欧美精品日日鲁夜夜添| 欧美日韩在线视频一区| 国产精品美女黄网| 国产欧美日韩不卡| 亚洲福利视频网| 亚洲区欧美区| 亚洲一级影院| 久久国产精品黑丝| 欧美激情国产日韩| 亚洲精品中文字幕在线观看| av成人国产| 午夜精品久久久| 久久影视三级福利片| 欧美日韩另类一区| 国产一区二区久久| 亚洲欧洲免费视频| 亚洲视频播放| 久久国内精品自在自线400部| 久久综合久久综合久久| 亚洲精品久久久蜜桃| 香蕉久久夜色精品| 能在线观看的日韩av| 国产精品久久久久久亚洲毛片| 国产一区二区三区直播精品电影| 亚洲精品久久久久久久久| 亚洲女人天堂av| 欧美18av| 亚洲尤物在线视频观看| 美日韩免费视频| 国产精品嫩草久久久久| 亚洲精品乱码久久久久久| 欧美在线高清视频| 亚洲黄色影片| 欧美专区18| 欧美涩涩网站| 91久久精品日日躁夜夜躁欧美 | 91久久精品视频| 欧美在线不卡| 亚洲欧洲日本在线| 久久久精品网| 国产农村妇女毛片精品久久麻豆 | 国产精品丝袜91| 亚洲精品一区二区网址| 久久综合久久88|