530
技術社區[雲棲]
linux驅動開發--內核空間中內存的申請與釋放
1、Linux內存管理
linux內存最小管理單位為頁(page),通常一頁為4kb。
linux係統中,在初始化時,內核為每個物理內存頁建立一個page的管理結構,操作物理內存時實際上就是操作page頁。
物理地址:出現在cpu地址總線上的尋址物理內存的地址信號,是地址變換的最終結果。
線性地址(虛擬地址):在32位cpu架構上,可以表示4G的地址空間,也就是0x00000000-0xFFFFFFFF。
邏輯地址:實際上是一個相對地址,是程序代碼經過編譯之後在匯編程序中出現的地址。
linux內核地址轉換:出現在機器語言指令(程序編譯後得到的二進製機器碼序列)中的內存地址都是邏輯地址,邏輯地址必須被轉換為線性地址,MMU將線性地址轉換成物理地址,最終實現對應物理內存的訪問。在linux係統中,邏輯地址和線性地址(虛擬地址)是一致的。
2、實例代碼
/** *Copyright (c) 2013.TianYuan *All rights reserved. * *文件名稱: kernelspace.c *文件標識: 內核空間中內存的申請與釋放 * *當前版本:1.0 *作者:wuyq * *取代版本:xxx *原作者:xxx *完成日期:2013-11-18 */ #include <linux/init.h> #include <linux/module.h> #include <linux/fs.h> #include <linux/slab.h> #include <linux/vmalloc.h> MODULE_LICENSE("GPL"); #define PAGE_NUM 4/*編碼幻數*/ unsigned char *kernelkmalloc = NULL; unsigned char *kernelpagemem = NULL; unsigned char *kernelvmalloc = NULL; static int __init kernelspace_init(void) { int ret = -ENOMEM; /* *GFP_KERNEL:分配內存,分配過程中可能導致睡眠。 *GFP_ATOMIC:分配過程中不會導致睡眠。 *GFP_DMA:申請到的內存通常情況下,位於0~16M之間 *__GFP_HIGNMEM:申請高端內存(物理地址896M以上的) */ kernelkmalloc = (unsigned char *)kmalloc(100, GFP_KERNEL); if(IS_ERR(kernelkmalloc)){ printk("kmalloc failed!\n"); ret = PTR_ERR(kernelkmalloc); goto failure_kmalloc; } printk("kmalloc space :0x%lx!\n",(unsigned long)kernelkmalloc); kernelpagemem = (unsigned char *)__get_free_pages(GFP_KERNEL, PAGE_NUM);/*請求或釋放的頁數的2的冪*/ if(IS_ERR(kernelpagemem)){ printk("get_free_pages failed!\n"); ret = PTR_ERR(kernelpagemem); goto failure_get_free_pages; } printk("get_free_pages address:0x%lx\n",(unsigned long)kernelpagemem); kernelvmalloc = (unsigned char *)vmalloc(1024*1024); if(IS_ERR(kernelvmalloc)){ printk("vmalloc failed!\n"); ret = PTR_ERR(kernelvmalloc); goto failure_vmalloc; } printk("vmalloc address: 0x%lx\n",(unsigned long)kernelvmalloc); return 0; failure_vmalloc: free_pages((unsigned long)kernelpagemem, PAGE_NUM); failure_get_free_pages: kfree(kernelkmalloc); failure_kmalloc: return ret; } static void __exit kernelspace_exit(void) { vfree(kernelvmalloc); free_pages((unsigned long)kernelpagemem, PAGE_NUM); kfree(kernelkmalloc); } module_init(kernelspace_init); module_exit(kernelspace_exit);
KERNELDIR ?=/root/Desktop/work/ldd3/linux-2.6.31_TX2440A PWD := $(shell pwd) obj-m += kernelspace.o default: $(MAKE) -C $(KERNELDIR) M=$(PWD) modules clean: @rm -f *.o *.ord* *.sy* *.mod.* *.ko
最後更新:2017-04-03 12:53:58