在堆上分配内存
进程可以通过增加堆的大小来分配内存,所谓堆是一段长度可变的连续虚拟内存,始于进程的未初始化数据段末尾,随着内存的分配和释放而增减。通常将堆的当前内存边界称为"program break"。
下面也会学习一族函数brk,sbrk和malloc。
调整program break:brk和sbrk
改变堆的大小(即分配或释放内存),其实就像命令内核改变进程的program break位置一样简单。最初,program break正好位于未初始化数据段末尾之后(和&end位置相同)。
在program break的位置抬升后,程序可以访问新分配区域内的任何内存地址,而此时物理内存页尚未分配。内核会在进程首次试图访问这些虚拟内存地址时自动分配新的物理内存页。
传统的UNIX系统虽然提供了两个操纵program break的系统调用:brk和sbrk,在Linux中依然可以用。虽然代码很少直接使用这些系统调用,但了解它们有助于弄清内存分配的工作过程。
int brk(void *end_data_segment);
void *sbrk(intptr_t increment);
系统调用brk会将program break设置为参数end_data_segment所指定的位置。由于虚拟内存页以页为单位进行分配,end_data_segment实际会四舍五入到下一个内存页的边界处。
当试图将program break设置为一个低于其初始值(即低于&end)的位置时,有可能会导致无法预知的行为,例如,当程序试图访问的数据位于初始化或未初始化数据段中当前尚不存在的部分时,就会引发段内存访问错误(segmentation default)(SIGSEGV信号)。
program break可以设定的精确上限取决于一系列因素。