linux子系统包括:linux
物理地址:出如今CPU地址总线上的寻址物理内存的地址信号,是地址变换的最终结果。架构
线性地址(虚拟地址):在32位CPU架构下,能够表示4G的地址空间,16进制0xffffffff。spa
逻辑地址:操作系统
逻辑地址--->物理地址:code
首先cpu利用段氏内存管理单元,将逻辑地址转换成线性地址。blog
再利用页式内存管理单元,把线性地址最终转换成物理地址。进程
段氏管理(16位CPU):16位CPU内部拥有20位的地址线,寻址范围是2的20次方,也就是1M的内存空间。但只有16位CPU用于存放地址的寄存器(只能访问65536个存储单元,64K)。内存
为了可以访问1M的内存空间,CPU就采用了内存分段的管理模式,并在CPU内部加入了段寄存器。16位CPU把1M内存空间分为若干个逻辑段,每一个逻辑段要求以下:内存管理
1.逻辑段的起始地址(段地址)必须是16的倍数,即最后4个二进制位必须是全0.class
2.逻辑段的最大容量为64K
页式管理:线性地址被分为固定长度的组,称为页(page)。
32位PC采用两种不一样的工做方式:实模式和保护模式。
通常工做在保护模式。
内存管理子系统职能:1.管理:虚拟地址与物理地址的映射。
2.管理:物理内存的分配。
linux内存管理:
全部段的基地址均为0.
由于每一个段的基地址为0,所以,逻辑地址与线性地址保持一致。也就是在linux中所提到的逻辑地址和线性地址可认为一致。
虚拟内存:linux操做系统采用虚拟内存管理技术,使得每一个进程都有独立的进程地址空间。用户看到和接触的都是虚拟地址,没法看到实际的物理地址。
利用这种虚拟地址不但能起到保护操做系统的做用,并且更重要的是用户进程可以使用比实际物理内存更大的地址空间。
进程空间:用户空间对应进程,每当进程切换,用户空间就会跟着变化。
每一个进程的用户空间都是彻底独立、互不相干的。( 如把同一个程序同时运行10次,会看到10个进程使用的线性地址如出一辙)
ps aux
cat /proc/<pid>/maps
实际物理内存只有当进程真的去访问新获取的虚拟地址时,才会由“请页机制”产生“缺页”异常,进而进入分配实际页框的程序。以后虚拟地址才实实在在地映射到了物理地址上。
在linux中,由kmalloc来分配动态内存(应用程序中malloc)
#include <linux/slab.h> void *kmalloc(size_t size,int flags) //size:要分配的内存大小 //flags:分配标志。它控制kmalloc的行为。
分配标志:
内存使用:
内核空间:内核空间是由内核负责映射,它并不会跟着进程改变,是固定的。
高端内存:物理内存896M以上的部分称为高端内存。
4G的虚拟地址空间又分为:
1.用户空间(0~3G):用户程序
2.内核空间(3G~4G):
直接映射区(3G~3G+896M):由于该区域的线性地址和物理地址之间存在线性转换关系:线性地址=3G+物理地址。
动态内存映射区(Vmalloc区):线性地址空间连续,对应物理空间不必定连续。vmalloc分配的线性地址所对应的物理页可能处于低端内存,页可能处于高端内存。
KMAP区(永久内存映射区):
固定映射区: