vmalloc分配虚拟地址空间的连续区域,物理上可能不连续。vmalloc是内存分配机制的基础。
kmalloc和_get_free_pages返回的内存地址也是虚拟地址,其实际值要有内存管理单元处理才能转为物理内存,使用的虚拟地址范围与物理内存一一对应,不需要修改页表。
vmalloc在如何使用硬件上没有区别,区别在于内核如何执行分配任务。vmalloc和ioremap使用的地址范围完全是虚拟的,要通过对页表的设置来建立虚拟内存区域。
用vmalloc分配得到的地址是不能再微处理器之外使用的,只在处理器的内存管理单元上才有意义。不能在原子上下文中使用。
头文件:<linux/vmalloc.h>
api:
void vmalloc(unsigned long size);
void vfree(void *addr);
void *ioremap(unsigned long offset, unsigned long size);
void iounmap(void *addr);
和vmalloc不同的是,ioremap并不实际分配内存,返回一个特殊的虚拟地址,可以访问特定的物理内存区域。更多地用于映射物理的PCI缓冲区地址到虚拟的内核空间。返回的地址不能直接访问。
#include <linux/init.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/vmalloc.h>
#define TEST_ORDER 0
static void *ptr = NULL;
static int __init hello_init(void)
{
printk("hello_init. \n");
ptr = (void *)vmalloc(PAGE_SIZE << TEST_ORDER);
if (!ptr)
{
printk("vmalloc failed \n");
return -1;
}
memset(ptr, 0x0, PAGE_SIZE << TEST_ORDER);
return 0;
}
static void __exit hello_exit(void)
{
printk("hello_exit. \n");
if (ptr)
{
vfree(ptr);
ptr = NULL;
}
}
MODULE_LICENSE("GPL");
module_init(hello_init);
module_exit(hello_exit);