虚拟内存模型

Linux 内核本身并不运行在虚拟空间中,其使用的是物理寻址模式。

物理内存被分割为界面,一个内存页面的大小由PAGE_SIZE宏决定。

虚拟地址空间的方式使程序员可以将巨大的结构用于连续的地址,而不必考虑物理内存上的限制。

线性地址到物理地址

线性地址需要由处理器或者一个单独的MMU转换为物理地址,转换方式如下:

解析的方式为:

1.用线性地址中的第一个位段为下标可以在页面目录中找的一个表项,这个表项指向某个中间目录。

2.用线性地址中的第二个位段为下标可以在该中间目录中找到一个表项,该表项指向某个页面表。

3.用线性地址中的第三个位段为下标可以在该页面表中找到一个表项,该表项指向物理内存中的某个物理页面。

4.用线性地址中的第四个位段为偏移量,将次偏移量与该物理页面的基地址相加便得到相应的物理地址。

注 : TLB中存的是页面目录基地址。

而内核空间的线性地址解析成物理地址的过程相对简单,因为系统本身要为自己维护一张页表是一件很恶心的事情,所以内核空间的地址和物理地址是简单的直接映射,0xc0000000就是两者的偏移量。

也就是说:对于系统空间,给定一个虚地址x,其物理地址是x中减去PAGE_OFFSET;相应的,给定一个物理地址x,其虚地址是x+PAGE_OFFSET。

当然,不能全部用来做简单映射,因为如果内存有4G,那么如果直接全部做简单映射,那么内核只能访问0-1G的内存了,所以内核中实际是前896M做直接映射,剩下的用来做vmalloc申请区,永久内存映射区,固定映射区等。如下图:

几个创建进程函数

Fork:子进程只是复制父进程的资源,两进程内存空间独立,运行时并行。为了降低运行时的开销,对资源采用写时复制(也就是fork并不直接拷贝资源,只是当发生写入的时候才完成拷贝)。

Vfork:共享地址空间,创建后进程父进程阻塞直到子进程结束。

Clone:有参数,可以有选择的将数据复制给子进程,剩下的通过指针共享。

几个申请内存的函数的区别

1. kmalloc和vmalloc分配内核空间的内存,malloc分配用户空间的内存。

2. kmalloc物理连续,vmalloc虚拟地址连续。

3. kmalloc能分配的空间大小有限,vmalloc和malloc能分配的较大。

4. 内存在被DMA时需要物理连续。

5. Kmalloc更快

6. Kmalloc使用slab机制,返回地址之后就已经对应实际内存了。

缺页中断

如果MMU处理器不能访问一个页面,它将产生一个缺页中断。Linux缺页异常程序必须能区分由编程引起的以及由引用属于地址空间还未分配物理页框的页引起的异常,如果合法则分配新的页框。

每个进程都有自己的页表。

页面管理

请求换页

为了节省内存,操作系统只会加载那些正在被执行程序使用的虚拟页面。比如,某个数据库程序可能要对某个数据库进行查询操作,此时并不是数据库的所有内容都要加载到内存中去,而只加载那些要用的部分。这种仅将要访问的界面载入的技术叫请求换页。

当进程试图访问不在内存中的虚拟地址时,将触发一个缺页中断。

如果出错的地址是无效的,比如进行了一个随机的写操作,则操作系统终止此进程,此进程的出错不会影响其他进程。

如果出错的地址是有效的,但是它访问的页面不在内存中。则操作系统必须将此界面从磁盘映像中读出来。访盘时间比较长,进程必须等待一段时间直到页面被取出来。取过来的页面放在一个空闲的物理页框之中,同时此进程的页表中将添加对应此虚拟页面框号的入口。然后进程从出错的地方从新开始运行。

交换

如果进程需要把一个虚拟页面调入物理内存而正好系统没有空闲的物理页面,操作系统必须丢弃系统中某些页面来为之腾出空间。

如果那些从物理内存中丢弃出来的界面来自于磁盘上的可执行文件或者数据文件,并且没有修改过则不需要保存那些界面。当进程再次需要此页面时,直接从可执行文件或者数据文件中读出即可。

但是如果页面被修改过,则系统必须将其保存以备再次访问。这种界面叫做dirty界面。当从内存中移出来的时候,他们必须保存在叫做交换文件的特殊文件中。相对于cpu和内存的速度,访问交换文件的速度很慢,所以操作系统必须衡量好将那些dirty页交换或将其保留在内存中做出一个选择。

如果丢弃页面的算法选择的不够好,则可能不断地出现页面被写入又被从磁盘中读回的情况使效率变低。Linux使用最近最少使用(LRU)页面衰老算法来公平的选择将要从系统中抛弃的页面。这种算法为每个页面设置一个年龄,它随页面访问次数而变化。页面被访问的次数越多则越年轻,相反则衰老。

Linux高速缓冲

缓冲区高速缓冲(buffer cache)

缓冲区高速缓冲包含了由块设备使用的数据缓冲区。这些缓冲区包含了从设备中读取的数据块或写入设备的数据块。如果数据能够在缓冲区高速缓冲中找到,则系统没有必要在物理块设备上进行实际的读操作。

页面高速缓冲(page cache)

页面高速缓冲是页面I/O操作访问数据所使用的磁盘高速缓存。我们在文件系统会看到,read(),write()和mmap()系统调用对常规文件的访问都是通过页面高速缓存来完成的。

缓冲区高速缓存和页面高速缓存的区别:缓冲区高速缓存是块设备的cache,页面高速缓冲是用来做文件系统的cache,也就是它直接记忆我们打开的文件(例如使用两次man命令,可以明显感觉第二次更快)。

交换高速缓冲(swap cache)

就是交换空间在内存中的缓存。当写入交换空间的文件并没有被再次修改,那么下次再丢弃他的时候就不需要再次写入交换空间,这个就是通过swap cache来做的。

注意:

使用top命令会发现有buffer和cache的概念:

Buffer:简单的说是要被写入磁盘的,磁盘和主存的速度不一,所以需要缓冲区做一个中间层,写入数据的话先写入缓冲区,这样的话写入的进程没有必要陷入等待。

Cache(非CPU和主存件的cache):简单的说就是从磁盘读入的,存储在cache里以备后面再次使用。经常被用在磁盘I/O请求上,如一个文件被访问,则其将被放入cache中,以备以后再有进程访问。

Cache和buffer都是需要占用内存的。

原文:

A buffer is something that has yet to be "written" to disk. A cache is something that has been "read" from the disk and stored for later use.

linux内存管理初学的更多相关文章

  1. 浅谈Linux内存管理机制

    经常遇到一些刚接触Linux的新手会问内存占用怎么那么多?在Linux中经常发现空闲内存很少,似乎所有的内存都被系统占用了,表面感觉是内存不够用了,其实不然.这是Linux内存管理的一个优秀特性,在这 ...

  2. linux内存管理

    一.Linux 进程在内存中的数据结构 一个可执行程序在存储(没有调入内存)时分为代码段,数据段,未初始化数据段三部分:    1) 代码段:存放CPU执行的机器指令.通常代码区是共享的,即其它执行程 ...

  3. Linux内存管理原理

    本文以32位机器为准,串讲一些内存管理的知识点. 1. 虚拟地址.物理地址.逻辑地址.线性地址 虚拟地址又叫线性地址.linux没有采用分段机制,所以逻辑地址和虚拟地址(线性地址)(在用户态,内核态逻 ...

  4. 了解linux内存管理机制(转)

    今天了解了下linux内存管理机制,在这里记录下,原文在这里http://ixdba.blog.51cto.com/2895551/541355 根据自己的理解画了张图: 下面是转载的内容: 一 物理 ...

  5. Linux内存管理原理【转】

    转自:http://www.cnblogs.com/zhaoyl/p/3695517.html 本文以32位机器为准,串讲一些内存管理的知识点. 1. 虚拟地址.物理地址.逻辑地址.线性地址 虚拟地址 ...

  6. Windows内存管理和linux内存管理

    windows内存管理 windows 内存管理方式主要分为:页式管理,段式管理,段页式管理. 页式管理的基本原理是将各进程的虚拟空间划分为若干个长度相等的页:页式管理把内存空间按照页的大小划分成片或 ...

  7. linux 内存管理——内核的shmall 和shmmax 参数

    内核的 shmall 和 shmmax 参数 SHMMAX= 配置了最大的内存segment的大小 ------>这个设置的比SGA_MAX_SIZE大比较好. SHMMIN= 最小的内存seg ...

  8. linux内存管理子系统

    一.Linux内存管理模型 1.虚拟地址与物理地址的映射 2.物理地址的分配二.虚拟地址与物理地址的映射 1.虚拟地址空间分布 32位处理器有32根地址总线,可访问4G的物理空间.其中有0-3G为用户 ...

  9. Linux内核分析(三)----初识linux内存管理子系统

    原文:Linux内核分析(三)----初识linux内存管理子系统 Linux内核分析(三) 昨天我们对内核模块进行了简单的分析,今天为了让我们今后的分析没有太多障碍,我们今天先简单的分析一下linu ...

随机推荐

  1. Spring声明式事务如何选择代理方式?

    Spring声明式事务如何选择代理方式   解决方法: 1.基于注解方法: <tx:annotation-driven transaction-manager="txManager&q ...

  2. cut截取数据

    参考文档 https://blog.csdn.net/caoshunxin01/article/details/79355566 [root@kube-node3 ~]# cat tab_space. ...

  3. iOS-AVFoundation生成缩略图

    使用MPMoviePlayerController来生成缩略图足够简单,但是如果仅仅是是为了生成缩略图而不进行视频播放的话,此刻使用 MPMoviePlayerController就有点大材小用了.其 ...

  4. 性能优化-service进程防杀

    service作为后台服务,其重要性不言而喻,但很多时候service会被杀死,从而失去了我们原本想要其发挥的作用,在这种情况下我们该如何确保我们的service不被杀死就是本节需要讨论的内容了 se ...

  5. Android Studio优化编译速度

    随着Android Studio的不断完善,其安卓开发者阵营也基本从Eclipse转移到了Android Studio,毕竟Android Studio是谷歌亲力亲为开发的官方开发软件.不过其最重要的 ...

  6. go timer定时器 WaitGroup用法

    参考文章: timer定时器: https://www.cnblogs.com/oxspirt/p/7107312.html WaitGroup用法: https://studygolang.com/ ...

  7. Linux利器之perf(火焰图)

    1 Ubuntu下安装perf 在Ubuntu18上默认没有安装perf,可以通过如下命令尝试:sudo apt-get install linux-tools 结果出来以下内容: Reading p ...

  8. java枚举的线程安全及序列化

    原文链接:https://www.cnblogs.com/z00377750/p/9177097.html https://www.cnblogs.com/chiclee/p/9097772.html ...

  9. 记:联调安卓设备的神药-无需usb数据线即可直连

    前言 最近需要调试公司的安卓服务,正常情况下,我们调试都是减安卓设备通过usb连接在我们座位旁,再不济就是我们扛笔记本到硬件旁边,这样调试屡试不爽,但是有一天你突然发现你带的数据线因为各种原因总是终端 ...

  10. MAVEN(二)

    1.本地仓库?Maven到底有哪些仓库?它们什么关系? Maven仓库: 本地仓库路径配置: 包查找路径:本地——>私服——>中央仓库,然后将查找到的jar同步到私服——>本地仓库 ...