Kmalloc Vmalloc 与malloc 的区别

总的来说

  1. kmalloc和vmalloc是分配内核的内存,malloc分配的是用户空间的内存。
  2. kmalloc 保证分配的内存在物理上是连续的,vmalloc保证的是在虚拟地址空间上的连续,malloc申请的内存不一定连续(用户空间存储以空间链表的方式组织(地址递增),每一个链表块包含一个长度、一个指向下一个链表块的指针以及一个指向自身的存储空间指针。)
  3. kmalloc能分配的大小有限,vmalloc与malloc能分配的空间大小相对较大。
  4. 内存只有在要被DMA访问的时候才需要物理上连续。
  5. vmalloc要不kmalloc要慢。

详细的解释

对于提供了MMU(存储管理器,辅助操作系统进行内存管理,提供虚实地址转换等硬件支持)的处理器而言,linux提供了复杂的内存管理系统,是的进程所能访问的内存达到4G.

进程的4G内存空间被人为分为两个部分-- 用户空间和内核空间。 用户空间的地址分布从0到3GB(PAGE_OFFSET,在0x86中它等于0xC0000000),3GB到4GB为内核空间。

内核空间中,从3G到vmalloc_start这段地址是物理内存映射区域(该区域中包含了内核镜像、物理页框表mem_map等等),比如我们使用的VMware虚拟系统内存是160M,那么3G~3G+160M 这片内存就应该映射物理内存,在物理内存映射区之后,就是vmalloc区域。对于160M的系统而言,vmalloc_start位置应该在3G+160M附近(在物理内存映射区域vmalloc_start器件还存在一个9M的gap来防止越界),vmalloc_end的位置接近4G(最后位置系统会保留一片128k大小的区域用于专用页面映射)

kmalloc和get_free_page申请的内存位于物理内存映射区域,而且在物理上也是连续的,它们与真实的物理地址只有一个固定的偏移,因此存在较简单的转换关系,virt_to_phys() 可以实现内核虚拟地址转换为物理地址:

#define _pa(x) ((unsigned long)(x)-PAGE_OFFSET)

extern inline unsigned long virt_to_phsy(volatile void *address)
{
    return _pa(address);
}

上面的转换过程是将虚拟地址减去3G(PAGE_OFFFSET=0xc0000000)。

与之对应的函数为phys_to_virt(), 将内核的物理地址转换为虚拟地址:

#define _va(x) ((void *)(unsigned long)(x)+PAGE_OFFSET)

extern inline void *phys_to_virt(unsigned long address)
{
    return _va(address);
}

virt_to_phys()和phys_to_virt()都定义在include/asm-i386/io.h中。

而vmalloc申请的内存则位于vmalloc_start~vmalloc_end之间,与物理地址没有简单的转换关系,虽然逻辑上他们也是连续的,但是在物理上不要求连续。

#include <linux/module.h>
#include <linux/slab.h>
#include <linux/vmalloc.h>

MODEULE_LICENSE("GPL");

unsigned char *pagemem;
unsigned cahr *kmallocmen;
unsigned char *vmallocmem;

int __init mem_module_init(void)
{
    //每次内存的申请都要检查是否申请成功,这里只做测试
    pagemen = (unsigned char*)get_free_page(0);
    printk("<1>pagemen addr = %x",pagemem);
    
    kmallocmen = (unsigned char*)kmalloc(100,0);
    pringk("<1>kmallccmem addr = %x",vmallocmen);

    vmallocmem = (unsigned char*)vmalloc(1000000);
    printk("<1>vmallocmem addr=%x",vmallocmem);
    return 0;
}

void __exit mem_module_exit(void)
{
    free_page(pagemem);
    kfree(kmallocmem);
    vfree(vmallocmem);
}

module_init(mem_module_init);
module_exit(men_module_exit);

我们系统上有160MB的内存空间,运行一次上述程序,发现pagemem的四肢在0xc7997000(约3G+121M)、kmallocmem地址在0zc9b1380(约3G+155M)、cmallocmem的地址在0xcabeb(约3G+171M)处,符合前面所述的内存布局。

 3.5.7 vmalloc 实现原理------《深入Linux内核架构》笔记_wong_daniel的博客-CSDN博客 vmallc实现原理。

  • 2
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值