c - 如何在 C 或 Assembly 中修改 Stack 上的返回地址

标签 c assembly stack program-counter

如您所知,当子程序调用时,当前 PC(程序计数器)值存储在堆栈中。我想在子程序中修改它,如下所示。我想使用 gcc 编译器在 Intel Core-i7 3632QM 上执行此操作。

void main()
{
     foo();
}
void foo()
{
     pop return address from stack;
     modify return address;
     push it to stack;
}

最佳答案

这几乎可以肯定是一个 XY 问题,你没有说出你真正想做什么。 不管怎样,这里是修改返回地址的示例代码:

#include <stdio.h>
#include <stdlib.h>

void bar()
{
    puts("entered the bar ;)");
    exit(0);
}

void** search(void** addr, void* value) __attribute__((noinline));
void** search(void** addr, void* value)
{
    while(*addr != value) addr++;
    return addr;
}

void foo() __attribute__((noinline));
void foo()
{
    void** p = search((void**)&p, __builtin_return_address(0));
    *p = bar;
}

int main()
{
    foo();
    return 0;
}

See it in action .

显然 foo 不能被内联,因为它甚至有一个返回地址,我不得不将 search 拆分到它自己的函数中,以解决一些模糊的优化问题,从而否则编译器会删除对返回地址的写入。与仅硬编码局部变量的某些特定偏移量相比,像这样搜索返回地址可以更容忍堆栈布局差异。

关于c - 如何在 C 或 Assembly 中修改 Stack 上的返回地址,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27213382/

相关文章:

c - 在 C 中使用 Net-SNMP 的简单 snmp 陷阱生成器程序?

c++ - 使用 SIMD 指令执行任意 128/256/512 位排列的最快方法是什么?

java - 找到不匹配的括号的索引

c - GCC 计算 goto 和堆栈指针的值

c - http2 请求返回错误请求

c - 如何帮助程序员在 C 中编写安全和正确的 printf 调用?

使用 setrlimit 更改堆栈大小后,C++ 仍然出现段错误(核心转储)错误

linux - 中断发生时,或者进程调度时,是否需要保存标志寄存器?

assembly - 如何让用户的十六进制输入在 x86 程序集(16 位 DOS)中打印为十进制?

python - 遍历一个 Spaghetti Stack 并返回到一棵树