我是个在学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;
}
`
你的思路没问题,但有些细节没注意
mprotect
函数的返回值,参考man mprotect
mprotect
修改的内存地址必须以页大小对齐,且需要写属性举个例子