[TakeLionOS汇编与C语言自学笔记-第五课]
版权所有 2021年 江志剑
按照《21天学通C语言》(第七版),有如下二个例子。
一个例子输出一段话,另一个例子计算年份:
//1、files:hello1.c hello1no.s hello1.exe
#include
int main(void)
{
printf("This is an example of sometjing printed!");
return 0;
}
//2、files:helloyear.c helloyearno.s helloyear.exe
#include
#define TARGET_AGE 88
int year1, year2;
int calcYear(int year1);
int main(void)
{
printf("What year was the subject born?");
printf("Enter as a 4-digit year (YYYY):");
scanf("%d", &year1);
year2 = calcYear(year1);
printf("Someone born in %d will be %d in %d.", year1, TARGET_AGE, year2);
return 0;
}
int calcYear(int year1)
{
return (year1 + TARGET_AGE);
}
我是用GCC 8.1.0在命令行窗口运行的。
GCC编译时候,其实有四个步骤:
1、预处理:gcc -E hello.c -o hello.i
2、编译生成汇编语言:gcc -S hello.i -o hello.s
3、汇编:gcc -c hello.s -o hello.o
4、链接生成可执行文件:gcc hello.o -o hello
这时候,我想看一下第二步生成的汇编语言代码。
因为本身就是研究操作系统的一部分,所以要研究底层的汇编语言。
结果发现是AT&T格式的汇编语言风格,而且大量使用汇编宏。
所以启用了GCC的编译选项-fno-asynchronous-unwind-tables,直接预处理为cfi宏的汇编指令。
这里就是包含*no.s形式的文件。
比较发现,第一个文件和第二个文件,关键看寄存器的使用以及子函数的调用。
还有就是全局变量的使用。
我们学习汇编语言,因为零基础不知道学习什么?所以先有个概念。
比如这时候,几节课下来,就知道了该认真学习寄存器的用法了。
这是一个基本概念,现在反复用到了。
最后的LEAVE指令,等效于“MOV ESP,EBP”和“POP EBP”两条指令。
这个指令调整了数据栈指针ESP,并将EBP的数值恢复到调用这个函数之前的初始状态。
毕竟函数要在退出之前恢复这些寄存器的值。