kmalloc和vmalloc的区别和联系

来源:互联网 发布:索尼手机更新软件 编辑:程序博客网 时间:2024/05/16 17:36

------------------------------------------------------------------------------------------------------------------

void *kmalloc(size_t size, unsigned int flags);

参数1:要分配空间的大小 参数2:分配

flags的取值:GFP_KERNEL和GFP_ATOMIC

GFP_KERNEL由运行在内核态的进程调用,分配内存,可能引起睡眠
GFP_ATOMIC在中断出来函数、taskklet、内核定时器和持有自旋锁的时候申请内存,必须使用GFP_ATOMIC分配标志


kmalloc申请的内存的物理地址是连续的,对应的内核逻辑地址也是连续的,物理地址和内核逻辑地址是一一对应的关系,两者之间只相差一个固定的偏移量,即:逻辑地址 = 物理地址 + 偏移量

kmalloc这个函数存在于<linux/slab.h>这个头文件内

------------------------------------------------------------------------------------------------------------------

void *vmalloc(unsigned long size);

参数:分配的内核虚拟内存的大小


vmalloc分配虚拟地址空间的连续区域,这段区域在物理上可能是不连续的。

并且vmalloc这个函数不可以用于原子上下文,因为它执行的是kmalloc(size_t size, GFP_KERNEL)操作

------------------------------------------------------------------------------------------------------------------

分配页面函数或宏
unsigned long get_zeroed_page(unsigned int flags);//返回指向新页面的指针,并将页面清零
unsigned long __get_free_page(unsigned int flags);  //和上面的类似,但不清零页面
unsigned long __get_free_pages(unsigned int flags, unsigned int order);//分配若干个连续的页面,返回指向该内存区域的指针,但也不清零这些内存区域

对应的释放函数或宏

void free_page(unsigned long addr);
void free_pages(unsigned long addr, unsigned long order);//释放的页数为2^(order)个

------------------------------------------------------------------------------------------------------------------

void *kmalloc(size_t size, unsigned int flags);       对应       void kfree(void *addr);

void *vmalloc(unsigned long size);                    对应       void vfree(void *addr);

------------------------------------------------------------------------------------------------------------------

虚拟地址和物理地址之间的关系

内核虚拟地址转化为物理地址
#define __pa(x) ((unsigned long) (x) - PAGE_OFFSET)  //物理地址 = 虚拟地址 - 偏移量
extern inline unsigned long virt_to_phys(volatile void *address)
{
return __pa(address);
}


物理地址转化为内核虚拟地址
#define __va(x) ((void *) ((unsigned long)(x) + PAGE_OFFSET))//虚拟地址 = 物理地址 + 偏移量
extern inline void *phys_to_virt(unsigned long address)
{
return __va(address);
}


偏移量的通常选值为3GB