进程的编译过程及内存布局
1. 进程的编译过程
一个c程序,编译时一般分为三步,预处理、编译、链接。
1.1 预处理
预处理相当于如下命令:
g++ -E hello.cpp -o hello.i
-E就是执行预编译,具体内容是展开#define、#include、#ifndef等等预编译指令。
1.2 编译
编译就是处理如下命令:
g++ -S hello.i -o hello.s
-S就是执行编译,将源代码转换为机器能识别的汇编代码,具体内容就是语法分析,词法分析,代码优化以及最后转换成汇编代码。
1.3 链接(先生成目标文件,然后链接)
链接又分为几种情况,一种是生成静态库,一种是生成动态库,还有就是直接生成可执行文件。
静态链接命令如下:
g++ -c add.cpp -o add.o
g++ -c sub.cpp -o sub.o
ar cr libmymath.o sub.o add.o
动态链接命令如下:
g++ -fPIC -shared -olibmymath.so add.cpp sub.cpp
-fPIC 表示编译为位置独立的代码,不用这个选项的话,编译后的代码是位置相关的,所以动态载入时是通过代码复制的方式来满足不同进程的需要,而不能达到真正代码段共享的目的。
可执行文件如下:
g++ -o main.cpp -L. -lmymath
无非就是先生成.o文件,然后根据不同的编译选项生成需要的文件。
所以,链接其实就是把各个模块之间,按照一定的关系组装起来,还包括地址和空间分配、符号决议和重定位等等。
2. c程序的内存布局
一个程序在内存中一般都包含三个部分,即:BSS段、数据段、代码段。
bss段包含未初始化的全局变量和未初始化的static变量,如果变量都已初始化,则bss段被清零;
数据段包含堆、栈和真正的静态区;
代码段则用于存放代码,是只读区域;
按照地址从低到高,顺序排列依次是:代码段、堆、待使用的空白区域、栈、静态区、bss段。