Linux系统的虚拟内存,内核空间和用户空间,以下都是以80x86 CPU为例进行分析
(一)以32位的80x86 CPU为例,CPU有32根地址总线,Linux虚拟内存的大小是4GB(2^32 Bytes),内核会将这4G字节的空间分为两部分:最高的1G字节(虚地址0xC0000000到0xFFFFFFFF)分配给内核使用,称为“内核空间”,较低的3G字节(虚地址0x00000000到0XBFFFFFFF)供用户使用,称为“用户空间”,由于每个进程都可以通过系统调用进入内核,因此Linux内核是由所有进程共享的,从具体的进程角度来看,每个进程就会拥有大小为4G的虚拟内存空间,在任意一个时刻,CPU上只有一个进程在运行,因此对CPU来说,这一时刻整个系统就只会存在一个4G的虚拟内存空间,其他进程的虚拟内存空间对CPU来说是未知的。随着进程的切换,虚拟地址空间也随之切换。
(二)内核空间是被所有的进程和内核所共享,但是用户空间是仅被各进程所独享,也就是说,某一个进程在其用户空间的某一个地址读写数据与另一个进程在其用户空间的相同地址读写数据并不冲突。
(三)程序编译链接完成后的地址是一个虚拟地址,但是程序最终会运行在物理内存中。Linux用户空间的虚拟地址映射到物理地址是通过分页机制来完成(也就是页表),任一时刻,CPU只会有当前运行进程的页表,来实现虚拟地址到物理地址的映射;用户空间的虚拟地址到物理地址的映射与之不同,在Linux代码中规定了3GB作为_PAGE_OFFSET,内核空间的一个虚拟地址减去_PAGE_OFFSET,就完成了虚拟地址到物理地址的映射。