1.1从指令角度掌握调用堆栈详细过程
先看demo程序
#include<iostream>
using namespace std;
int sum(int a, int b)
{
int temp = 0;
temp = a + b;
return temp;
}
int main()
{
int a = 10;
int b = 20;
int ret = sum(a, b);
cout << "ret:" << ret << endl;
return 0;
}
问题1:main 函数调用sum,sum执行完以后,怎么知道回到哪个函数?
从main函数进入sum时,会把当前的栈的起始地址(main函数的ebp)压入sum函数栈的栈底,当函数结束后弹出,此时就知道main函数在哪里了。
问题2:sum函数执行完,回到main以后,怎么知道从哪一行指令继续运行?
sum函数最后会存之前进入sum函数的指令的下一条指令地址放到PC寄存器。此时就知道是main函数的哪一条了。
辅助记忆
1.2内存分配
从上往下
------用户区------
- 系统预留空间(不可访问)
- 代码段(.text .rodata)程序运行时指令存放的地方
如char *p="hello world"
,其中hello world
就存在rodata区(只读代码段) - 数据段(.data)— 存放已经初始化了,且初始化不为0的数据
- 数据段 (.bss)— 存放未初始化,或初始化为0
- 堆(.heap)地址从上往下增长
- 加载共享库,动态链接库(.dll, .so)
- 栈(stack)— 每个线程私有(地址从下往上增长)
- 命令行参数和环境变量(包含程序搜索头文件和库文件的命令)
------ 内核区------ - ZONE_DMA
- ZONE_NORMAL pcb,内核空间的线程,栈空间,进程控制块
- ZONE_HIGHMEM 地址映射
1.3 从编译器角度理解c++代码的编译与链接
分为编译和链接两个过程
编译过程
将*.cpp文件(源文件)=》*.o文件(二进制可重定位目标文件)
链接过程:编译完成的所有.o文件+静态库文件=》.exe(可执行文件).out
步骤一:
所有.o文件段的合并
符号表合并后,进行符号解析
步骤二:
符号的重定位(重定向)