linux分配地址连续的大内存

kernel version 2.6.29
内核函数经常须要临时分配一块任意大小的物理地址连续的内存空间. 因此先介绍内核中两个分配物理地址连续的内存空间的API.
node

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()以前申请所需的连续大内存. 或添加一内核参数根据须要来调整数据缓冲大小. 不过此段内存也就永久保留, 除非直接引用所分配的内存地址.函数

例:
spa

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

unsigned long long pf_buf_len = 0x0;
EXPORT_SYMBOL( pf_buf_len );

void *pf_buf_addr = NULL;
EXPORT_SYMBOL( pf_buf_addr );

static int __init pf_buf_len_setup(char *str)
{
    unsigned long long size;
    unsigned int       nid = 0;
    void              *pbuff = NULL;
 // 分析参数
    size = memparse( str, &str );
    if ( *str == '@' ){
        str ++;
        get_option( &str, &nid );
    }
    //printk( KERN_INFO "pf_buf_len: Allocating %llu bytes/n", size );
 // 分配内存
    pbuff = alloc_bootmem( size );
    if ( likely( NULL != pbuff ) ) {
        printk( KERN_INFO "pf_buf_len: Allocated %llu bytes at 0x%p(0x%p) on node %u/n",
            size, pbuff, (void *)virt_to_phys(pbuff), nid);
        pf_buf_addr = pbuff;
        pf_buf_len  = size;
        goto out;
    }
    printk( KERN_ERR "pf_buf_len: Allocated %llu bytes fail./n", size );
out:
    return 1;

}
__setup( "pf_buf_len=", pf_buf_len_setup);
相关文章
相关标签/搜索