博客
关于我
在内核中分配物理地址连续的大内存.
阅读量:576 次
发布时间:2019-03-11

本文共 2118 字,大约阅读时间需要 7 分钟。

 

kernel version 2.6.29

内核函数常常需要临时分配一块任意大小的物理地址连续的内存空间. 所以先介绍内核中两个分配物理地址连续的内存空间的API.

kmalloc

由于采用了SLUB作为默认内存分配器, 所以 kmalloc 工作于 SLUB 分配器之上。内核初始化时,创建一组共 13 个通用对象的缓冲区。kmalloc_caches 数组存放了这些缓冲区的 kmem_cache 数据结构(kmalloc_sizes.h)。由于 kmem_cache 数据结构是通过 kmalloc 来分配的,故而只能用静态分配的 kmem_cache 结构数组来描述通用对象的缓冲区。其中 kmalloc_caches[0] 代表的缓冲区专门分配 kmem_cache_node 结构。kmalloc_caches[1] 缓冲区对象大小为64,kmalloc_caches[2] 缓冲区对象大小为192,其余第 i(3-12)号缓冲区对象大小为 2^i。如果请求分配超过物理页面大小(4096)的对象,直接调用页框分配器__get_free_pages.

__get_free_pages

__get_free_pages 采用的内存分配方式为Buddy算法. 所以一般分配的数据大小为故是2 ^ order个页面大小. 内核中定义了一个宏 MAX_ORDER, 表示一次请求能分配的最大物理页数不能 >= MAX_ORDER, 也就是最大可以分配到的内存块. 2.6.29.6 中MAX_ORDER为 11 即最大分配大小为 ( 2 ^ 10 ) * 4096 = 4M.

如果我们需要在内核中分配4M以上的连续物理内存,怎么办,当前内核对应的方法还有两个:
1> 使用static或全局变量数组, 直接定义变量大小为所需数据大小.
例:
    static char buffer[ 512 * 1024 * 1024 ];
 定义512M大小数组. 不过此方法应用到模块中话, 会导致加载模块速度奇慢.

2> 使用alloc_bootmem系列API在start_kernel调用mem_init()之前申请所需的连续大内存. 或添加一内核参数根据需要来调整数据缓冲大小. 不过此段内存也就永久保留, 除非直接引用所分配的内存地址.

例:

    以下定义一内核参数pf_buf_len=nn[KMG],可定制分配的内存大小. 并EXPORT地址与长度信息.

参考:

Linux slab 分配器剖析
Linux SLUB 分配器详解

注:

还有一内核参数memmap=nn[KMG]$ss[KMG] 可以将ss位置起始一段大小为nn的内存保留,并不会将其映射到地址空间内. 一般在检测Bad RAM时有用,可以直接跳过Bad RAM物理地址空间段进行映射. 驱动中也可以直接使用其物理地址,常用的驱动是framebuffer驱动 driver/video/sgivwfb.c

转载地址:http://goxvz.baihongyu.com/

你可能感兴趣的文章
mysql 递归查找父节点_MySQL递归查询树状表的子节点、父节点具体实现
查看>>
mysql 通过查看mysql 配置参数、状态来优化你的mysql
查看>>
mysql 里对root及普通用户赋权及更改密码的一些命令
查看>>
Mysql 重置自增列的开始序号
查看>>
mysql 锁机制 mvcc_Mysql性能优化-事务、锁和MVCC
查看>>
MySQL 错误
查看>>
mysql 随机数 rand使用
查看>>
MySQL 面试题汇总
查看>>
MySQL 面试,必须掌握的 8 大核心点
查看>>
MySQL 高可用性之keepalived+mysql双主
查看>>
mysql 默认事务隔离级别下锁分析
查看>>
Mysql--逻辑架构
查看>>
MySql-2019-4-21-复习
查看>>
mysql-5.7.18安装
查看>>
MySQL-Buffer的应用
查看>>
mysql-cluster 安装篇(1)---简介
查看>>
mysql-connector-java各种版本下载地址
查看>>
mysql-EXPLAIN
查看>>
mysql-group_concat
查看>>
MySQL-redo日志
查看>>