linux
kamlloc函数原型:
#include<linux/slab.h>
Void *kmalloc(size_t size, int flags);
1.使用GFP_KERNEL容许kmalloc在分配空闲内存时候如果内存不足容许把当前进程睡眠以等待。因此这时分配函数必须是可重入的。如果在进程上下文之外如:中断处理程序、tasklet以及内核定时器中这种情况下current进程不该睡眠,驱动程序该使用GFP_ATOMIC.
2.GFP_ATOMIC用来从中断处理和进程上下文之外的其他代码中分配内存. 从不睡眠.
kmalloc 能够处理的最小分配是 32 或者 64 字节, 依赖系统的体系所使用的页大小. kmalloc 能够分配的内存块的大小有一个上限.kmalloc最多只能开辟大小为32XPAGE_SIZE的内存,一般的PAGE_SIZE=4kB,也就是128kB的大小的内存。如果大于这个,应该用vmalloc,或者用get_free_pages直接申请页kmalloc和get_free_page申请的内存位于物理内存映射区域,而且在物理上也是连续的,它们与真实的物理地址只有一个固定的偏移,因此存在较简单的转换关系,virt_to_phys()可以实现内核虚拟地址转化为物理地址:将虚拟地址减去3G(PAGE_OFFSET=0XC000000)
kmalloc和vmalloc的区别
• vmalloc()与 kmalloc()都可用于分配内存
• kmalloc()分配的内存处于3GB~high_memory之 间,这段内核空间与物理内存的映射一一对应,vmalloc()分配的内存在 VMALLOC_START~4GB之间,这段非连续内 存区映射到物理内存也可能是非连续的
• 在内核空间中调用kmalloc()分配连续物理空间,而调用vmalloc()分配非物理连续空间。vfree
• 把kmalloc()所分配内核空间中的地址称为内核逻辑地址, 把vmalloc()分配的内核空间中的地址称 为内核虚拟地址
• vmalloc()在分配过程中须更新内核页表
winodws
windows在内存使用上进行了严格的控制,如果不遵守level,就可能造成系统不稳定,并且这种bug很难调试。所有开发时必须对分配非常了解。相对linux界限较为明显,其实大同小异。不同的中断级别调用的分页/非分页是不一样的,如这段话:中断处理程序、tasklet以及内核定时器中这种情况下current进程不该睡眠,驱动程序该使用GFP_ATOMIC.
kmalloc 类似于 win非分页,虽然一个参数,但是有本质的差别
vmalloc 类似于 win分页
linux的 vmalloc()和window的分页内存有相似之处(个人认为),vmalloc是从物理内存之后开始进行映射的,随时可能被缺页中断。
内核空间中,从3G到vmalloc_start这段地址是物理内存映射区域(该区域中包含了内核镜像、物理页框表mem_map等等),比如我们使用 的 VMware虚拟系统内存是160M,那么3G~3G+160M这片内存就应该映射物理内存。在物理内存映射区之后,就是vmalloc区域。对于 160M的系统而言,vmalloc_start位置应在3G+160M附近(在物理内存映射区与vmalloc_start期间还存在一个8M的gap 来防止跃界),vmalloc_end的位置接近4G(最后位置系统会保留一片128k大小的区域用于专用页面映射)