正如我在关于Linux进程环境所讲每个C程序执行main函数之前都有一段启动代码,完成启动main所需的设置。
glibc中的启动代码的实现可以参见:程序入口函数和glibc及C++全局构造和析构
而我在本文中将实现一个简单的启动程序(32位系统),启动程序分为start.s和init.c两个文件。代码见下方。
start.s:
.text
.globl init
start:
call init
够简单吧。
我们可以使用as start.s -o start.o 生成目标文件。
init.c:
extern int main(int,char **,char **);
extern int exit(int);
int init(int argc,char *string)
{
char **argv;
char **envp;
argv=&string;
envp=argv+argc+1; //argc保存了argv中指针的个数,但是需要记住的是argv是以0结尾的,所以这里需要加上1
main(argc,argv,envp);
exit(1);
}
int main(int argc,char **argv,char **envp)
{
return 0;
}
我们使用gcc -S init.c生成汇编文件init.s
as init.s -o init.o 生成目标文件
使用ld -o a.out start.o init.o (ld -s -M start.o init.o -o a.out )生成可执行文件a.out (start.o一定要在init.o文件前,因为ld链接是是按输入的文件名的顺序链接目标文件的,要保证start.o的内容被连接在最前面)
文章最后附一个完全用C语言写的启动代码
extern int main(int,char **,char **);
int init(char *string)
{
int argc;
char **argv;
char **envp;
int *argcp;
argv=&string;
argcp=(int *)(argv-1);
argc=*argcp;
envp=argv+argc+1;
main(argc,argv,envp);
exit(1);
}
对于生成的可执行文件可以使用objdump -S a.out来查看反汇编代码