(本章基于:Linux-4.4.0-37)
在内核中常用kmalloc申请较小的内存块,并用kfree释放所申请的内存。说申请较小的内存块意思是在申请大块内存时kmalloc效率很低,而且kmalloc一般最多只能申请128k-16字节数据。kmalloc与用户空间使用的malloc非常相似,不同之处在于kmalloc不对获取到的内存空间清零,这就意味着我们在使用的时候需要显示的清零。声明如下:
static __always_inline void *kmalloc(size_t size, gfp_t flags);void kfree(const void *);
相关头文件:linux/slab.h
size为需要申请的内存大小,kmalloc返回内存首地址。flags是分配标志,可控制kmalloc的分配方式,常用KFP_KERNEL:
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 读是有用的, 此时在处理器缓冲中出现是无用的.
__GFP_NOWARN
这个很少用到的标志阻止内核来发出警告(使用 printk ), 当一个分配无法满足.
__GFP_HIGH
这个标志标识了一个高优先级请求, 它被允许来消耗甚至被内核保留给紧急状况的最后的内存页.
__GFP_REPEAT
__GFP_NOFAIL
__GFP_NORETRY
这些标志修改分配器如何动作, 当它有困难满足一个分配. __GFP_REPEAT 意思是" 更尽力些尝试" 通过重复尝试 -- 但是分配可能仍然失败. __GFP_NOFAIL 标志告诉分配器不要失败; 它尽最大努力来满足要求. 使用 __GFP_NOFAIL 是强烈不推荐的; 可能从不会有有效的理由在一个设备驱动中使用它. 最后, __GFP_NORETRY 告知分配器立即放弃如果得不到请求的内存.
例:
malloc_m.c
#include <linux/init.h>
#include <linux/module.h>
#include <linux/slab.h>
#define MAX_BUF_SIZE (1024)
static __init int malloc_init(void)
{
char *buf = NULL;
buf = kmalloc(MAX_BUF_SIZE, GFP_KERNEL);
if(!buf) {
printk(KERN_WARNING "memory error!\n");
return -EFAULT;
}
memset(buf, 0, MAX_BUF_SIZE);
strcpy(buf, "abcdefghijklmnopqrstuvwxyz\n");
printk(KERN_ALERT "buf:%s\n", buf);
kfree(buf);
printk(KERN_ALERT "malloc init success!\n");
return 0;
}
static __exit void malloc_exit(void)
{
printk(KERN_WARNING "malloc exit!\n");
}
module_init(malloc_init);
module_exit(malloc_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("Stone");