- 38
- 0
- 约21.52万字
- 发布于河南
- 举报
- 文档已下架,其它文档更精彩
ldd3(中)
第 8 章分配内存
上一页
下一页
至此, 我们已经使用 kmalloc 和 kfree 来分配和释放内存. Linux 内核提供了更丰富的
一套内存分配原语, 但是. 在本章, 我们查看在设备驱动中使用内存的其他方法和如何优化
你的系统的内存资源. 我们不涉及不同的体系实际上如何管理内存. 模块不牵扯在分段, 分
页等问题中, 因为内核提供一个统一的内存管理驱动接口. 另外, 我们不会在本章描述内存
管理的内部细节, 但是推迟在 15 章.
8.1. kmalloc 的真实故事
上一页
下一页
kmalloc 分配引擎是一个有力的工具并且容易学习因为它对 malloc 的相似性. 这个函数
快(除非它阻塞)并且不清零它获得的内存; 分配的区仍然持有它原来的内容.[28] 分配的区
也是在物理内存中连续. 在下面几节, 我们详细讨论 kmalloc, 因此你能比较它和我们后来
要讨论的内存分配技术.
8.1.1. flags 参数
记住 kmalloc 原型是:
#include linux/slab.h
void *kmalloc(size_t size, int flags);
给 kmalloc 的第一个参数是要分配的块的大小. 第 2 个参数, 分配标志, 非常有趣, 因为
它以几个方式控制 kmalloc 的行为.
最一般使用的标志, GFP_KERNEL, 意思是这个分配((内部最终通过调用
__get_free_pages 来进行, 它是 GFP_ 前缀的来源) 代表运行在内核空间的进程而进
行的. 换句话说, 这意味着调用函数是代表一个进程在执行一个系统调用. 使用
GFP_KENRL 意味着 kmalloc 能够使当前进程在少内存的情况下睡眠来等待一页. 一
个使用 GFP_KERNEL 来分配内存的函数必须, 因此, 是可重入的并且不能在原子上下
文中运行. 当当前进程睡眠, 内核采取正确的动作来定位一些空闲内存, 或者通过刷新缓存
到磁盘或者交换出去一个用户进程的内存.
GFP_KERNEL 不一直是使用的正确分配标志; 有时 kmalloc 从一个进程的上下文的
外部调用. 例如, 这类的调用可能发生在中断处理, tasklet, 和内核定时器中. 在这个情况
下, 当前进程不应当被置为睡眠, 并且驱动应当使用一个 GFP_ATOMIC 标志来代替. 内
核正常地试图保持一些空闲页以便来满足原子的分配. 当使用 GFP_ATOMIC 时,
kmalloc 能够使用甚至最后一个空闲页. 如果这最后一个空闲页不存在, 但是, 分配失败.
其他用来代替或者增添 GFP_KERNEL 和 GFP_ATOMIC 的标志, 尽管它们 2 个涵
盖大部分设备驱动的需要. 所有的标志定义在 linux/gfp.h, 并且每个标志用一个双下
划线做前缀, 例如 __GFP_DMA. 另外, 有符号代表常常使用的标志组合; 这些缺乏前缀
并且有时被称为分配优先级. 后者包括:
GFP_ATOMIC
用来从中断处理和进程上下文之外的其他代码中分配内存. 从不睡眠.
GFP_KERNEL
内核内存的正常分配. 可能睡眠.
GFP_USER
用来为用户空间页来分配内存; 它可能睡眠.
GFP_HIGHUSER
如同 GFP_USER, 但是从高端内存分配, 如果有. 高端内存在下一个子节描述.
GFP_NOIO
GFP_NOFS
这个标志功能如同 GFP_KERNEL, 但是它们增加限制到内核能做的来满足请求. 一个
GFP_NOFS 分配不允许进行任何文件系统调用, 而 GFP_NOIO 根本不允许任何 I/O
初始化. 它们主要地用在文件系统和虚拟内存代码, 那里允许一个分配睡眠, 但是递归的文
件系统调用会是一个坏注意.
上面列出的这些分配标志可以是下列标志的相或来作为参数, 这些标志改变这些分配如何进
行:
__GFP_DMA
这个标志要求分配在能够 DMA 的内存区. 确切的含义是平台依赖的并且在下面章节来解
释.
__GFP_HIGHMEM
这个标志指示分配的内存可以位于高端内存.
__GFP_COLD
正常地, 内存分配器尽力返回缓冲热的页 -- 可能在处理器缓冲中找到的页. 相反, 这个标
志请求一个冷页, 它在一段时间没被使用. 它对分配页作 DMA 读是有用的, 此时在处理
器缓冲中出现是无用的. 一个完整的对如何分配 DMA 缓存的讨论看直接内存存取一节
在第 1 章.
__GFP_NOWAR
原创力文档

文档评论(0)