用汇编的眼光看C++(之特殊函数)

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接: https://blog.csdn.net/feixiaoxing/article/details/6768054

【 声明:版权所有,欢迎转载,请勿用于商业用途。  联系信箱:feixiaoxing @163.com】  

    这里说的函数主要指的是inline函数、static函数。inline函数比较特殊,它既具有宏的性质,同时也能让编译器对它进行函数检查。static函数同样也比较特殊,它只可以被同文件的函数使用。如果static函数在include文件中,那么这个头文件只要被使用一次,那么这个函数就要在exec文件中重新出现一次。现在大家可能理解起来有点困难,但是请大家稍微等待一下,下面我们将会用示例进行说明。最后,我们用一个替换的技巧对函数指针进行修改,让你调用的函数发生修改,这样给大家都函数的定义加深一下印象。


    (1)内联函数


 
 
  1. inline int add(int a, int b)
  2. {
  3. return a + b;
  4. }
    那么这个函数在应用的时候,会怎么编译呢,可以看一下?


 
 
  1. 0040114 A mov eax,1
  2. 0040114 F add eax,2
  3. 00401152 mov dword ptr [ebp-4], eax

    inline函数是一种特殊的函数。在进行函数编译的时候,编译器会对内联函数这段代码按照函数的要求进行格式检查。但是编译生成执行代码的过程中,编译器会把这段代码按照宏的性质复制到call的函数当中。所以在call函数中,我们发现这段调用代码并不是call的形式,而是直接按照语句的形式。但是这种inline函数中的代码行数不能过多,因为我们内联的目的就是就是减少call的机会。

注意:

    a) 函数在编译的时候需要打开INLINE优化开关,【PROJECT】->【setting】->【C/C++】->【optimizations】,在内联扩展中选择第二项

    b)编译的时候会生成错误,那么删除编译指令/ZI即可,结果是源码无法单步调试,只能汇编级单步调试


     (2) static函数是什么属性


 
 
  1. static int add(int a, int b)
  2. {
  3. return a + b;
  4. }
    a) 如果在不同的源文件都有这样一个add函数呢 ?

    如果在不同的文件里面函数声明为static函数,那么没有关系,各个static函数只为各个文件使用,不存在multi definition的问题。

    b)如果头文件有这样一个static函数声明和定义?

    头文件中有一个static函数的话,那么调用这个函数的每个文件都为这个static函数重新编译一下。结果和a)的结果是一样的,大家可以自己试试看一下,对static函数地址打印一下,看看是不是add函数的地址是一样的。

    

    (3)一个修改函数地址的范例


 
 
  1. #include <windows.h>
  2. int add(int a, int b)
  3. {
  4. return a + b;
  5. }
  6. int sub(int a, int b)
  7. {
  8. return a - b;
  9. }
  10. void set()
  11. {
  12. HANDLE hProcess = GetCurrentProcess();
  13. DWORD pOldFlag = 0;
  14. BOOL result = 0;
  15. result = VirtualProtectEx(hProcess, (LPVOID) add, 0x10, PAGE_EXECUTE_READWRITE, &pOldFlag);
  16. if(result != 0)
  17. {
  18. printf( "%d\n", GetLastError());
  19. }
  20. }
  21. void process()
  22. {
  23. char* n = ( char*) add;
  24. char* t = ( char*) sub;
  25. *n = 0xFF;
  26. *(n+ 1) = 0x25;
  27. *( int*)(n + 2) = ( int)&t;
  28. int data = add( 3, 2);
  29. assert( 1 == data);
  30. return;
  31. }
    简单介绍一下,上面的代码包括四个函数,add函数和sub函数主要为了替换测试使用,set函数是修改代码段访问属性的一段代码,而process函数就是我们测试使用的一段代码。其实这段代码的意思不难,目的在于你在call add函数,发现实际上在call的是sub函数。那么我们是怎么做到的呢,关键在两个方面:(1)修改add 函数代码段的访问属性;(2)修改add函数第一个字节的内容,那么我们需要把函数add处地内容修改为jmp sub,那么就要先修改属性,后修改内容。

【预告: 下面一片博客主要涉及class内存分布问题】

  

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值