如果有疑问,欢迎加入我的QQ交流群:887683536

由于我的虚拟机是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!

《程序员的自我修养——链接,装载与库》内联汇编实现printf在终端打印字符串_64位系统


以上的使用是未使用链接脚本的,所以他的可执行文件的.text .data .rodata 段是分离的

《程序员的自我修养——链接,装载与库》内联汇编实现printf在终端打印字符串_链接脚本_02

现在我们使用链接脚本来执行上述过程,先看代码

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!

然后我们来看一看这两个可执行文件有何不同

《程序员的自我修养——链接,装载与库》内联汇编实现printf在终端打印字符串_链接脚本_03


可以发现,.text,.data,.rodata消失了,出现了一个tinytext段。以上各种参数信息,在《程序员的自我修养——链接,装载与库》书中静态链接部分都能找到答案。附上参考文章​​参考文章​