linux设备驱动程序读书笔记,读书笔记:写一个块设备驱动(4)

首先32位linux内核(2.6.32)中线性地址(虚拟地址)分为两块:3G~4G为内核地址空间其余为进程地址

内核总共占据物理内存1G,分为3部分:DMA(0~15M)NORMAL(16~895M)HIGHMEM(896~)

内核线性地址空间与物理内存地址一一对应,不过分为两部分:线性地址映射区(3G~3G+896M)非线性地址映射区(3G+896M~4G)

线性地址映射区是与物理内存一一映射的,是在内核初始化时完成的,以后页表一直不会变化。

而非线性地址映射区是一直空在那的,只有当需要时,才会把这里的线性地址映射到高端内存,

从而可以顺利使用高端内存,使用完之后必须取消映射,并且归还这些线性地址。

非线性地址映射区是很宝贵的资源,对系统的稳定性有很大的意义。

回到我们的驱动程序来,因为我们的块驱动程序申请了一段16M的连续内存,所以导致在线性映射区找不到这样的连续内存,从而为我们分配了非线性映射区的内存(高端内存),并且在模块加载到内核时,一直霸占着16M的非线性地址映射区的线性地址,这不利于系统的稳定性和良好的系统性能。所以必须做出改变。

我们通过在线性地址映射区(DMA+NORMAL)中申请不连续的16M内存,然后通过基树结构进行管理。

基树:将对象的鉴别号数值ID与对象指针(线性地址)建立联系

基树的初始化和基本操作:void INIT_RADIX_TREE((struct radix_tree_root *root, gfp_t mask);int radix_tree_insert(struct radix_tree_root *root, unsigned long index, void *item);void *radix_tree_delete(struct radix_tree_root *root, unsigned long index);void *radix_tree_lookup(struct radix_tree_root *root, unsigned long index);

内存动态分配,我们采用(一次分配一页)unsigned long __get_free_page(gfp_t gfp_mask); /*gfp_mask = GFP_KERNEL*/

最后simp_blkdev_make_request函数有点麻烦,因为可能会出现跨越两块,需要读写的情况。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值