由于我的虚拟机是64位Ubunt20.04,所以和书中所述环境不一致,系统调用号,不一致,所以我看了一遍知乎上的内容,了解了关于64位系统下的环境,然后实现这个过程。 先看代码:
nomain.c
char* str = "Hello World!\n";
void exit() {
asm("movq $42,%rdi \n\t"
"movq $60,%rax \n\t"
"syscall \n\t");
}
void printf() {
asm("movq $13, %%rdx \n\t"
"movq %0, %%rsi \n\t"
"movq $1, %%rdi \n\t"
"movq $1, %%rax \n\t"
"syscall \n\t" ::"r"(str));
}
void nomain() {
printf();
exit();
}
我们设定程序的入口函数为nomain 执行命令:gcc -c -fno-builtin nomain.c 生成 nomain.o 可重定位目标文件 再执行命令:ld -static -e nomain -o progs nomain.o 生成可执行文件 progs 然后执行./progs 输出 Hello World! 以上的使用是未使用链接脚本的,所以他的可执行文件的.text .data .rodata 段是分离的
现在我们使用链接脚本来执行上述过程,先看代码
TinyHello.lds
ENTRY(nomain)
SECTIONS
{
. = 0x08048000 + SIZEOF_HEADERS;
tinytext : {*(.text) *(.data) *(.rodata)}
/DISCARD/ : {*(.comment)}
}
然后执行命令:ld -static -T TinyHello.lds -e nomain -o progs nomain.o 执行./progs,在终端输出 Hello World! 然后我们来看一看这两个可执行文件有何不同 可以发现,.text,.data,.rodata消失了,出现了一个tinytext段。以上各种参数信息,在《程序员的自我修养——链接,装载与库》书中静态链接部分都能找到答案。附上参考文章参考文章
|