七、揭开链接器的面纱(上)
1. 问题
源文件被编译后生成目标文件,这些目标文件如何生存最终的可执行程序?
2. 链接器的意义
链接器的主要作用是把各个模块之间相互引用的部分处理好,使得各个模块之间能够正确的衔接。
3. 目标文件的秘密
- 各个段没有具体的起始地址,只有段大小信息
- 各个标识符没有实际地址,只有段中的相对地址
- 段和标识符的实际地址需要链接器具体确定
4. 链接器的工作内容
- 将目标文件和库文件整合为最终的可执行程序
- 合并各个目标文件中的段( .text, .data, .bss )
- 确定各个段和段中标识符的最终地址(重定位)
- 链接前后对比
5. 问题
main()函数是第一个被调用执行的函数吗?
6. 默认情况下( gcc )
- 程序加载后,_start()是第一个被调用执行的函数
- _start()函数准备好参数后立即调用__libc_start_main()函数
- __libc_start_main()初始化运行环境后调用main()函数执行
- _start()函数的入口地址就是代码段( .text )的起始地址!
7. __libc_start_main()函数的作用
- 调用__libc_csu_init()函数(完成必要的初始化操作)
- 启动程序的第一个线程(主线程),main()为线程入口
- 注册__libc_csu_fini()函数(程序运行终止时被调用)
8. 程序的启动过程
9. 自定义程序入口函数
- gcc提供-e选项用于在链接时指定入口函数
- 自定义入口函数时必须使用-nostartfiles选项进行链接