网站联盟名词解释,微官网和移动网站区别吗,做电影网站被抓,小程序后端数据库搭建From: http://linux.chinaunix.net/techdoc/system/2008/06/16/1011365.shtml
对于提供了MMU#xff08;存储管理器#xff0c;辅助操作系统进行内存管理#xff0c;提供虚实地址转换等硬件支持#xff09;的处理器而言#xff0c;Linux提供了复杂的存储管理系统#x…From: http://linux.chinaunix.net/techdoc/system/2008/06/16/1011365.shtml
对于提供了MMU存储管理器辅助操作系统进行内存管理提供虚实地址转换等硬件支持的处理器而言Linux提供了复杂的存储管理系统使得进程所能访问的内存达到4GB。 进程的4GB内存空间被人为的分为两个部分--用户空间与内核空间。用户空间地址分布从0到3GB(PAGE_OFFSET在0x86中它等于0xC0000000)3GB到4GB为内核空间。 内核空间中从3G到vmalloc_start这段地址是物理内存映射区域该区域中包含了内核镜像、物理页框表mem_map等等比如我们使用的 VMware虚拟系统内存是160M那么3G3G160M这片内存就应该映射物理内存。在物理内存映射区之后就是vmalloc区域。对于 160M的系统而言vmalloc_start位置应在3G160M附近在物理内存映射区与vmalloc_start期间还存在一个8M的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_phys(volatile void * address) { return __pa(address); } 上面转换过程是将虚拟地址减去3GPAGE_OFFSET0XC000000。 与之对应的函数为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_startvmalloc_end之间与物理地址没有简单的转换关系虽然在逻辑上它们也是连续的但是在物理上它们不要求连续。 我们用下面的程序来演示kmalloc、get_free_page和vmalloc的区别 #include #include #include MODULE_LICENSE(GPL); unsigned char *pagemem; unsigned char *kmallocmem; unsigned char *vmallocmem; int __init mem_module_init(void) { //最好每次内存申请都检查申请是否成功 //下面这段仅仅作为演示的代码没有检查 pagemem (unsigned char*)get_free_page(0); printk(pagemem addr%x, pagemem); kmallocmem (unsigned char*)kmalloc(100, 0); printk(kmallocmem addr%x, kmallocmem); vmallocmem (unsigned char*)vmalloc(1000000); printk(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(mem_module_exit); 我们的系统上有160MB的内存空间运行一次上述程序发现pagemem的地址在0xc7997000约3G121M、kmallocmem地址在0xc9bc1380约3G155M、vmallocmem的地址在0xcabeb000约3G171M处符合前文所述的内存布局。 文件: v_k_malloc.tar.bz2 大小: 6KB 下载: 下载 vmalloc和kmalloc区别 kmalloc对应于kfree可以分配连续的物理内存 vmalloc对应于vfree分配连续的虚拟内存但是物理上不一定连续。 vmalloc分配内存的时候逻辑地址是连续的但物理地址一般是不连续的适用于那种一下需要分配大量内存的情况如insert模块的时候。这种分配方式性能不入kmalloc。 kmalloc分配内存是基于slab因此slab的一些特性包括着色对齐等都具备性能较好。物理地址和逻辑地址都是连续的 最主要的区别是 分配大小的问题。 比如你需要28个字节那一定用KMALLOC如果用VMALLOC分配不多次机器就罢工了。 n PAGE_OFFSET为3GBhigh_memory为保存物理地址最高值的变量VMALLOC_START为非连续区的起始地址 在物理地址的末尾与第一个内存区之间插入了一个8MB的区间这是一个安全区目的是为了“捕获”对非连续区的非法访问。出于同样的理由在其他非连续的内存区之间也插入了4K大小的安全区。每个非连续内存区的大小都是4096的倍数。 n vmalloc()与 kmalloc()都可用于分配内存 ü kmalloc()分配的内存处于3GBhigh_memory之间这段内核空间与物理内存的映射一一对应 ü vmalloc()分配的内存在VMALLOC_START4GB之间这段非连续内存区映射到物理内存也可能是非连续的 n vmalloc() 分配的物理地址无需连续而kmalloc() 确保页在物理上是连续的 n 尽管仅仅在某些情况下才需要物理上连续的内存块但是很多内核代码都调用kmalloc()而不是用vmalloc()获得内存。 n 这主要是出于性能的考虑。vmalloc()函数为了把物理上不连续的页面转换为虚拟地址空间上连续的页必须专门建立页表项。还有通过vmalloc()获得的页必须一个一个的进行映射因为它们物理上不是连续的这就会导致比直接内存映射大得多的缓冲区刷新。 n 因为这些原因vmalloc()仅在绝对必要时才会使用——典型的就是为了获得大块内存时例如当模块被动态插入到内核中时就把模块装载到由vmalloc()分配的内存上。