内核态用vmalloc申请大块内存

内核态用vmalloc申请大块内存

   

 

       本文欢迎自由转载,请标明出处,并保证本文的完整性。


经常在内核版看到网友发帖,问如何在内核态申请大块内存,上百兆甚至上G的内存。用kmalloc或get_free_pages都不能满足这样的要求。今天又在内核版看到这样的帖子,其中白金兄回复说可以用vmalloc实现,并且给出了例程。本人也进行了实践,初步看来是分配成功了,但至于是否可以有效的应用于生产环境还有待于验证。本文的总结仅作为学习环境上的探讨和时间。

以下是申请大块内存的代码。代码是以白金兄提供的代码(http://linux.chinaunix.net/bbs/viewthread.php?tid=1139485&page=2#pid7139143)为蓝本,本人仅添加了传递模块参数的代码。传递的模块参数为申请内存的大小,单位为Mbyte。


#include <linux/module.h>

#include <linux/vmalloc.h>

MODULE_AUTHOR("platinum");

MODULE_DESCRIPTION("This is a module sample.");

MODULE_LICENSE("GPL");

/*Godbach added module parameter*/

static int memsize = 100;/*Unit: Mbyte*/

module_param(memsize, int, S_IRUGO);

__u8 *data;

int

init_module (void)

{

data = vmalloc(1024 * 1024 * memsize);

if (!data)

return -ENOMEM;

memset(data, 0xff, 1024 * 1024 * memsize);

printk("module loaded.\n");

return 0;

}

void

cleanup_module(void)

{

vfree(data);

printk("module unloaded.\n");

}

本人的内核版本是2.6.18.3。编译该内核模块,生成alloc_large_mem.ko.

以下是默认申请100M内存时系统前后内存的变化。

[root@localhost alloc_large_mem]# free
             total       used       free     shared    buffers     cached
Mem:        385624     188760     196864          0       1908      54936
-/+ buffers/cache:     131916     253708
Swap:       522104       8068     514036
[root@localhost alloc_large_mem]# insmod alloc_large_mem.ko
[root@localhost alloc_large_mem]# free
             total       used       free     shared    buffers     cached
Mem:        385624     290388      95236          0       1960      54948
-/+ buffers/cache:     233480     152144
Swap:       522104       8068     514036
[root@localhost alloc_large_mem]#

由此可见,insmod模块之后系统使用的内粗增加了100M多一点。

那么,另外一个问题,vmalloc可以分配的内存上限是多少呢?参考白金兄的说法,vmalloc分配的内存上限是和cat /proc/meminfo中VmallocTotal的值有关的。我这里的显示结果为:


[root@localhost alloc_large_mem]# cat /proc/meminfo

……

VmallocTotal:  638968 kB

VmallocUsed:      3512 kB

VmallocChunk:  633644 kB


因此,vmalloc最大可分配的内存为600M,由于我的剩余物理内存只有200M左右。所以这里尝试300M的内存观察一下结果。


[root@localhost alloc_large_mem]# free

total       used       free    shared    buffers     cached

Mem:       385624     101216     284408          0       4348     46800

-/+ buffers/cache:      50068    335556

Swap:      522104      88400     433704

[root@localhost alloc_large_mem]# cat /proc/meminfo | grep Vmalloc

VmallocTotal:  638968 kB

VmallocUsed:      3512 kB

VmallocChunk:  633644 kB

[root@localhost alloc_large_mem]#

[root@localhost alloc_large_mem]# insmod alloc_large_mem.ko memsize=300

[root@localhost alloc_large_mem]# free

total       used       free    shared    buffers     cached

Mem:       385624     380040       5584          0       1796     21896

-/+ buffers/cache:     356348     29276

Swap:      522104      88400     433704

[root@localhost alloc_large_mem]# cat /proc/meminfo | grep Vmalloc

VmallocTotal:  638968 kB

VmallocUsed:   311032kB

VmallocChunk:  326440 kB

[root@localhost alloc_large_mem]#


这里测试的情况是虚拟内存的使用情况却是增加了300M左右,而物理内存基本上已经耗尽了。

白金兄曾经测试在启动时 kernel 里 vmalloc=1280M,然后insmod内核模块成功分配到1G的内存。我也曾测试分配500M或400M的情况(虚拟内存上限600多M),但都失败了:


[root@localhost alloc_large_mem]# insmod alloc_large_mem.ko memsize=500

insmod: error inserting 'alloc_large_mem.ko':-1 Cannot allocate memory

[root@localhost alloc_large_mem]# insmod alloc_large_mem.ko memsize=400

insmod: error inserting 'alloc_large_mem.ko':-1 Cannot allocate memory


究其原因,应该是超过了内核启动时默认的vmalloc的值,可以通过指定内核启动时vmalloc=xxx来解决。

以上是用vmalloc申请大块内存的总结。不足之处,请大家多多指教。

你可能感兴趣的:(内核态用vmalloc申请大块内存)