Linux进程虚拟地址空间

在linux中,各个进程的虚拟地址空间是独立的。各个进程的虚拟地址空间起始于0,延伸至TASK_SIZE -1.

在32 bit系统中,内核分配1GB,而各个用户空间进程可用的部分为3GB。

进程虚拟地址空间由若干个区域组成,

1.当前运行代码的二进制代码.text段。

2.程序使用的动态库代码。

3.存储区局变量和静态变量的数据段,bss,data段

4.保存动态分配数据的堆

5.保存局部变量和实现函数 调用的栈

6.环境变量和命令行参数。

7.文件内容映射到虚拟地址空间的内存映射。

具体布局如下图所示,

 

如果全局变量randomize_va_space设置为1,那么启用地址空间随机化机制(上图的ramdom xxx offset)。用户可以通过/proc/sys/kernel/randomize_va_space停用该特性。

每个进程都有mm_struct(linux/mm_types.h)的实例,保存进程虚拟内存管理信息。

struct mm_struct {
struct vm_area_struct *mmap; /* list of VMAs */
struct rb_root mm_rb;
#ifdef CONFIG_MMU
unsigned long (*get_unmapped_area) (struct file *filp,unsigned long addr, unsigned long len,unsigned long pgoff, unsigned long flags);
#endif
unsigned long mmap_base; /* base of mmap area */虚拟地址空间中用于内存映射的起始地址。
unsigned long mmap_legacy_base; /* base of mmap area in bottom-up allocations */
unsigned long task_size; /* size of task vm space */进程地址空间的size.

struct list_head mmlist; /* List of maybe swapped mm's. These are globally strung

unsigned long start_code, end_code, start_data, end_data;
unsigned long start_brk, brk, start_stack;//堆首地址,堆尾地址,栈首地址。
unsigned long arg_start, arg_end, env_start, env_end;

....

};

进程虚拟地址空间由多个VMA组成。有两种组织VMA的方式,链表(mmap)和红黑树(mm_rb)

VMA结构体如下:

struct vm_area_struct {
  /* The first cache line has the info for VMA tree walking. */

  unsigned long vm_start; /* Our start address within vm_mm. */
  unsigned long vm_end; /* The first byte after our end address within vm_mm. */

  /* linked list of VM areas per task, sorted by address */
  struct vm_area_struct *vm_next, *vm_prev;

  struct rb_node vm_rb;

  struct mm_struct *vm_mm; /* The address space we belong to. */

  /* Function pointers to deal with this struct. */
  const struct vm_operations_struct *vm_ops;

  struct file * vm_file; /* File we map to (can be NULL). */
  void * vm_private_data; /* was vm_pte (shared mem) */

};

VMA链表组织形式如下图:

 VMA红黑树组织形式如下:

 

 

posted @ 2016-12-25 22:02  fellow_jing  阅读(1445)  评论(0编辑  收藏  举报