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

如果有疑问,欢迎加入我的QQ交流群:887683536
《程序员的自我修养——链接,装载与库》内联汇编实现printf在终端打印字符串_第1张图片

由于我的虚拟机是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 段是分离的
《程序员的自我修养——链接,装载与库》内联汇编实现printf在终端打印字符串_第2张图片

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

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在终端打印字符串_第3张图片
可以发现,.text,.data,.rodata消失了,出现了一个tinytext段。以上各种参数信息,在《程序员的自我修养——链接,装载与库》书中静态链接部分都能找到答案。附上参考文章参考文章

你可能感兴趣的:(操作系统,汇编语言,计组)