C语言 有没有可能调用一个指向字符串的函数指针?
我是个在学C的萌新,一天突发奇想,指令和数据只是对人来说才有意义,
一段二进制串对CPU来说既可是数据,也可是指令,IP指向哪里就当作指令执行。那这样的话是不是意味着可以在C中执行字符串呢?
可,在探索中我发现没那么简单。
Program received signal SIGSEGV, Segmentation fault
我不确定这背后具体原因,我想可能是系统的保护机制。想到内存可执行可写等属性会不会有影响,于是加上mprotect
。但,还是不行。或许除代码段外的其它段都是不可执行的?
所以调用一个字符串可行吗?
如果不可行有具体的原因吗?
如果您抽空回答,我十分感谢!
Ps:如果觉得我上面说的和下面的代码很幼稚,那还望多多指教。谢谢 >_<
#include <stdio.h>
#include <string.h>
#include <sys/mman.h>
#define NOP 0x90
#define M_SIZE 128
/*
在Ubuntu19.04 gcc version 8.3.0 (Ubuntu 8.3.0-6ubuntu1) 中编译通过
*/
static char justAJust[] = "Hello world!";
//这是个我自己构造的字符串,其功能是同justAFun一样的
static char justAString[M_SIZE] = {0x55,0x48,0x89,0xe5,
0x48,0x83,0xec,0x10,
0x48,0x89,0x7d,0xf8,
0x48,0x8b,0x45,0xf8,
0x48,0x89,0xc7,0xe8,
0xe8,0xcf,0xff,0xff,
0x90,0xc9,0xc3,0x90};
void justAFun(char *justAJust){
puts(justAJust);
}
int main(int args,char *argv[]){
int i = 0;
char *justACharPtr = NULL;
void (*justAPtr)(char *) = NULL;
justACharPtr = (char *) justAFun;
memset(&justAString[strlen(justAString)],NOP,M_SIZE-strlen(justAString));
mprotect(justAString,24,PROT_EXEC | PROT_READ); //尝试去赋予justString可执行属性,但不管用
puts("justAFun函数16进制表示: ");
do
{
printf("%#x",justACharPtr[i] & 0xFF); //显示justAFun的'内容'
} while((justACharPtr[i++] & 0xFF) != 0xc3 && putchar(','));
putchar('\n');
justAFun(justAJust); //非常普通的调用justAFun
justAPtr = justAFun; //使用函数指针调用justAFun
justAPtr(justAJust);
justAPtr = (void (*)(char *))justAString; //尝试调用一个字符串
justAPtr(justAJust);
return 0;
}
`
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(3)
你的思路没问题,但有些细节没注意
mprotect
函数的返回值,参考man mprotect
mprotect
修改的内存地址必须以页大小对齐,且需要写属性举个例子
可行,题主开始关心指令执行是个非常好的开始 :-D 执行不了只是因为你尝试去把data segment的内存修改为可执行,是不可以的,但是我们可以mmap一段可执行内存来存放函数指令,而运行时生成指令你还需要关心,调用约定、平台指令等问题,我举了个例子:
运行时生成
void method(char *str){puts(str);}
main.c
gcc main.c && ./a.out
操作系统不会允许你这么做的,病毒喜欢这种方式,现代cpu都有内存权限控制,内存块有各种权限设置,比如指令执行权限,只有代码执行权限的内存cpu才能执行其中代码,否则会出错,而权限由操作系统分配,操作系统在加载你的程序的时候,会把指令部分加载到专门的内存区域,并且设置可以执行指令的权限,而常字符串、各种全局数据、堆栈还有heap,操作系统在分配这些内存的时候都不会给指令执行权限,所以你想执行保存在字符串中的指令是不可能的。