当前位置: 首页 > wzjs >正文

企业排名优化公司罗湖区seo排名

企业排名优化公司,罗湖区seo排名,妇幼能力建设网站,网站死链如何处理Linux驱动-块设备驱动 一,块设备驱动简介二,无请求队列情况(EMMC和SD卡等)三,请求队列情况(磁盘等带有I/O调度的设备)四,两者在驱动上区别 一,块设备驱动简介 块设备驱动…

Linux驱动-块设备驱动

  • 一,块设备驱动简介
  • 二,无请求队列情况(EMMC和SD卡等)
  • 三,请求队列情况(磁盘等带有I/O调度的设备)
  • 四,两者在驱动上区别

一,块设备驱动简介

  块设备驱动的关键结构在于:结构体gendisk,表示一个块设备,包含设备号,分区,操作函数集,相对于字符设备驱动区别在于:
在这里插入图片描述
  有I/O调度的块设备:(I/O调度意思就是在应用层对块设备进行操作时候,这个调度模块会把对块设备的操作转化为请求request,然后放到请求队列当中,因此在处理时候,主要是对请求队列中的bio请求进行操作),请求队列模式:通过 blk_init_queue 初始化请求队列,由调度器管理 request,再分发给驱动,如磁盘设备。
在这里插入图片描述

  没有I/O调度的设备:EMMC以及SD卡,当其使用时候,相当于直接对结构体bio进行操作,此时的bio不在request结构体中了,绕过传统的 I/O 调度器,每个 bio 直接传递给驱动处理,不再合并为 request,随机进行访问,不需要控制。

  对于块设备的操作,相当于存储设备操作,那么有三个要清除的地方:①物理存储器的地址在哪②内存中的地址在哪③存储长度是多少。由bio来控制,其中bi_sector即物理设备,比如EMMC,bio_vec即内存,比如RAM。
在这里插入图片描述

二,无请求队列情况(EMMC和SD卡等)

  其实这样操作感觉还是不太对劲,相当于在RAM开辟一段空间,来模拟EMMC或者SD卡,但EMMC和SD卡操作时候是和内存RAM进行交互,现在就变成了开辟的RAM和RAM进行操作…主要流程包括①块设备注册(因为是在RAM中开辟的空间,因此在disk_init中对2MB空间进行开辟,真实情况下,连接EMMC是不用开辟这段空间的,这个是模拟EMMC)②申请gendisk③设置自旋锁(无请求队列的起始用不到这个)④初始化请求队列⑤将请求队列和回调函数ram_queue_process关联起来⑥对gendisk中的数据初始化⑦设置gendisk块大小,并将其添加到内核中。

/*4.8日 块设备驱动*/
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/delay.h>
#include <linux/ide.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/errno.h>
#include <linux/gpio.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/of_gpio.h>
#include <linux/semaphore.h>
#include <linux/timer.h>
#include <linux/i2c.h>
#include <linux/genhd.h>
#include <linux/blkdev.h>
#include <linux/hdreg.h>
#include <asm/mach/map.h>
#include <asm/uaccess.h>
#include <asm/io.h>
#define RAM_BLOCK_NAME  "mydisk11"
#define RAM_BLOCK_BYTE_SIZE  2*1024*1024   //2MB 字节数
#define RAM_BLOCK_SIZE  2*1024*1024/512   //2MB 即多少个块struct disk_device{int  major;//块设备号struct gendisk  *ram_gendisk;spinlock_t my_spinlock;struct request_queue *queue; //请求队列unsigned char *ram_buf;
};
struct disk_device ram;static int ram_open(struct block_device *dev, fmode_t mode)
{printk("ram_open \r\n");return 0;
}
static void ram_release(struct gendisk *disk, fmode_t mode)
{printk("ram_release \r\n");
}
int ram_getgeo(struct block_device *dev,struct hd_geometry *geo)
{return 0;
}
static struct block_device_operations ram_ops ={.owner = THIS_MODULE,.open = ram_open,.release = ram_release,.getgeo = ram_getgeo,
};static void  ram_queue_process(struct request_queue *q,struct bio *bio)
{int offset = 0;unsigned long len = 0;//长度struct bio_vec ram_vec;//表示ram中的内存地址struct bvec_iter sd_emmc_iter;//连接的设备,比如emmc sd卡offset = (bio->bi_iter.bi_sector)*512; //磁盘的偏移地址bio_for_each_segment(ram_vec,bio,sd_emmc_iter)//遍历每一个bio{char *ptr = page_address(ram_vec.bv_page)+ram_vec.bv_offset;len = ram_vec.bv_len;if(bio_data_dir(bio)==READ){memcpy(ptr,ram.ram_buf+offset,len);}else if(bio_data_dir(bio)==WRITE){memcpy(ram.ram_buf+offset,ptr,len);}offset = offset + len;}set_bit(BIO_UPTODATE,&bio->bi_flags);bio_endio(bio,0);}    
static int __init disk_init(void)
{int ret=0;/*1.申请ram内存*/ram.ram_buf = kzalloc(RAM_BLOCK_BYTE_SIZE,GFP_KERNEL);//申请字节大小if(ram.ram_buf ==NULL){printk("kzalloc  error \r\n");ret = -1;goto kzalloc_ram_error;}printk("before ram.major = %d \r\n",ram.major);/*2.注册块设备*/ram.major = register_blkdev(0,RAM_BLOCK_NAME);if(ram.major < 0){printk("register error \r\n");ret = -1;goto register_blkdev_error;}/*3.申请gendisk*/ram.ram_gendisk = alloc_disk(3);if(ram.ram_gendisk == NULL){ret = -1;goto RAM_BLOCK_SIZE_error;}/*4.自旋锁*/spin_lock_init(&ram.my_spinlock);/*5.初始化请求队列*/ram.queue = blk_alloc_queue(GFP_KERNEL);if(ram.queue == NULL){ret = -1;goto blk_queue_error;}/*制造设置请求函数*/blk_queue_make_request(ram.queue,ram_queue_process);/*6.给gendisk内值,初始化*/ram.ram_gendisk->major = ram.major; //主设备号ram.ram_gendisk->first_minor = 0;//次设备号开始ram.ram_gendisk->fops = &ram_ops;//操作函数ram.ram_gendisk->private_data = &ram; //私有数据ram.ram_gendisk->queue = ram.queue;//请求队列//memcpy(&ram.ram_gendisk->disk_name,&RAM_BLOCK_NAME,sizeof(RAM_BLOCK_NAME));//sprintf(ram.ram_gendisk->disk_name, RAM_BLOCK_NAME);snprintf(ram.ram_gendisk->disk_name, 32, "%s",RAM_BLOCK_NAME);set_capacity(ram.ram_gendisk,RAM_BLOCK_SIZE);//设置大小,即多少块add_disk(ram.ram_gendisk);//添加gendisk到内核printk("major = %d \r\n",ram.major);return ret;
blk_queue_error:   del_gendisk(ram.ram_gendisk);
RAM_BLOCK_SIZE_error:unregister_blkdev(ram.major,RAM_BLOCK_NAME);
register_blkdev_error:kfree(ram.ram_buf);
kzalloc_ram_error:return ret;}
void __exit disk_exit(void)
{/*对gendisk进行操作*/del_gendisk(ram.ram_gendisk);//put_gendisk(ram.ram_gendisk);/*对ram.queue进行操作*/blk_cleanup_queue(ram.queue);/*对块设备进行操作*/unregister_blkdev(ram.major,RAM_BLOCK_NAME);/*释放开辟的空间*/kfree(ram.ram_buf);}module_init(disk_init);
module_exit(disk_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("WYT");

三,请求队列情况(磁盘等带有I/O调度的设备)

/*4.8日 块设备驱动*/
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/delay.h>
#include <linux/ide.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/errno.h>
#include <linux/gpio.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/of_gpio.h>
#include <linux/semaphore.h>
#include <linux/timer.h>
#include <linux/i2c.h>
#include <linux/genhd.h>
#include <linux/blkdev.h>
#include <linux/hdreg.h>
#include <asm/mach/map.h>
#include <asm/uaccess.h>
#include <asm/io.h>
#define RAM_BLOCK_NAME  "mydisk11"
#define RAM_BLOCK_BYTE_SIZE  2*1024*1024   //2MB 字节数
#define RAM_BLOCK_SIZE  2*1024*1024/512   //2MB 即多少个块struct disk_device{int  major;//块设备号struct gendisk  *ram_gendisk;spinlock_t my_spinlock;struct request_queue *queue; //请求队列unsigned char *ram_buf;
};
struct disk_device ram;static int ram_open(struct block_device *dev, fmode_t mode)
{printk("ram_open \r\n");return 0;
}
static void ram_release(struct gendisk *disk, fmode_t mode)
{printk("ram_release \r\n");
}
int ram_getgeo(struct block_device *dev,struct hd_geometry *geo)
{return 0;
}
static struct block_device_operations ram_ops ={.owner = THIS_MODULE,.open = ram_open,.release = ram_release,.getgeo = ram_getgeo,
};void  ram_queue_process(struct request_queue *q)
{int err = 0;unsigned long start = 0;unsigned long len = 0;void *buff = NULL;struct request *req;req = blk_fetch_request(q);//获取第一个请求,并把它移出while(req != NULL){/*进行数据操作*/start = blk_rq_pos(req)*512;//起始地址len = blk_rq_cur_bytes(req);//大小,字节单位buff = bio_data(req->bio);//目的让bio中的数据和buff指针执指向一个位置if(rq_data_dir(req) == READ)//读操作{memcpy(buff,ram.ram_buf+start,len);//正常用磁盘的情况,//应该是将ram.ram.ram_buf+start用bio中的物理存储地址}else if(rq_data_dir(req) == WRITE){memcpy(ram.ram_buf+start,buff,len);}/*判断是不是最后一个*/if(!__blk_end_request_cur(req,err))//完成当前部分请求,返回0,否则返回其他数值{req = blk_fetch_request(q);//if判断当前请求完成后,会继续申请下一个请求,直到申请到NULL}}
}    
static int __init disk_init(void)
{int ret=0;/*1.申请ram内存*/ram.ram_buf = kzalloc(RAM_BLOCK_BYTE_SIZE,GFP_KERNEL);//申请字节大小if(ram.ram_buf ==NULL){printk("kzalloc  error \r\n");ret = -1;goto kzalloc_ram_error;}printk("before ram.major = %d \r\n",ram.major);/*2.注册块设备*/ram.major = register_blkdev(0,RAM_BLOCK_NAME);if(ram.major < 0){printk("register error \r\n");ret = -1;goto register_blkdev_error;}/*3.申请gendisk*/ram.ram_gendisk = alloc_disk(3);if(ram.ram_gendisk == NULL){ret = -1;goto RAM_BLOCK_SIZE_error;}/*4.自旋锁*/spin_lock_init(&ram.my_spinlock);/*5.初始化请求队列*/ram.queue = blk_init_queue(ram_queue_process,&ram.my_spinlock);if(ram.queue == NULL){ret = -1;goto blk_queue_error;}/*6.给gendisk内值,初始化*/ram.ram_gendisk->major = ram.major; //主设备号ram.ram_gendisk->first_minor = 0;//次设备号开始ram.ram_gendisk->fops = &ram_ops;//操作函数ram.ram_gendisk->private_data = &ram; //私有数据ram.ram_gendisk->queue = ram.queue;//请求队列//memcpy(&ram.ram_gendisk->disk_name,&RAM_BLOCK_NAME,sizeof(RAM_BLOCK_NAME));//sprintf(ram.ram_gendisk->disk_name, RAM_BLOCK_NAME);snprintf(ram.ram_gendisk->disk_name, 32, "%s",RAM_BLOCK_NAME);set_capacity(ram.ram_gendisk,RAM_BLOCK_SIZE);//设置大小,即多少块add_disk(ram.ram_gendisk);//添加gendisk到内核printk("major = %d \r\n",ram.major);return ret;
blk_queue_error:   del_gendisk(ram.ram_gendisk);
RAM_BLOCK_SIZE_error:unregister_blkdev(ram.major,RAM_BLOCK_NAME);
register_blkdev_error:kfree(ram.ram_buf);
kzalloc_ram_error:return ret;}
void __exit disk_exit(void)
{/*对gendisk进行操作*/del_gendisk(ram.ram_gendisk);//put_gendisk(ram.ram_gendisk);/*对ram.queue进行操作*/blk_cleanup_queue(ram.queue);/*对块设备进行操作*/unregister_blkdev(ram.major,RAM_BLOCK_NAME);/*释放开辟的空间*/kfree(ram.ram_buf);}module_init(disk_init);
module_exit(disk_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("WYT");

四,两者在驱动上区别

  无I/O调度设备,虽然在申请了请求队列(blk_alloc_queue),但是没有用到,最终调用的是它的ram_queue_process-1,这个函数主要处理bio中的数据。有I/O调度的设备,在请求队列申请时候,调用函数ram_queue_process-2,这个函数在处理时候,是对请求队列中request中的bio进行操作。

	/*无I/O调度(不使用请求队列)*/ram.queue = blk_alloc_queue(GFP_KERNEL);/*制造设置请求函数*/blk_queue_make_request(ram.queue,ram_queue_process-1);/*有I/O调度(使用请求队列)*/ram.queue = blk_init_queue(ram_queue_process-2,&ram.my_spinlock);
http://www.dtcms.com/wzjs/174393.html

相关文章:

  • 做个网站多少钱大概河南郑州网站推广优化外包
  • 仿站容易还是建站容易网站服务器一年的费用
  • 网站开发页面源码网
  • b站官网楼市最新消息
  • 如何政府网站建设百度手机助手下载安卓版
  • 深圳网站设计联系电话网站关键词优化怎么弄
  • 网站建设彩铃语线下推广的渠道和方法
  • 网站建设公司哪家好 皆来磐石网络培训体系
  • 做逆战网站的名字吗网络营销期末总结
  • 开发公司工程项目质量安全管理体系百色seo外包
  • 钻井网站建设百度下载并安装最新版
  • 淄博高效网站建设找哪家广告网络
  • 网络网站建设价格网站结构
  • 新注册公司网站怎么做企业官方网站有哪些
  • 苏州哪个公司做网站好宁波网络推广团队
  • 卢湾北京网站建设杭州seo公司哪家好
  • wordpress安装域名长春网站seo公司
  • 网站建设 自适应抖音视频seo霸屏
  • php动态网站开发课后答案重庆seo的薪酬水平
  • 自己做的网站怎么放到网上去百度怎么发布自己的广告
  • 国外网站建设的发展友妙招链接怎么弄
  • 网站导入题库怎么做的seo推广软件排行榜
  • 苏州网站建设店铺装修独立站seo实操
  • 网站建设什么专业自建站seo如何做
  • 有什么网站做任务换q币吗注册域名费用一般多少钱
  • 奢侈品网站建设方案什么是网络营销的核心
  • 潍坊网站建设费用友情链接有什么用
  • 怎么让网站自适应vi设计公司
  • 游戏网站的导航条怎么做的怎么用网络推广
  • 网站运营团队建设搜索引擎广告推广