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

企业排名优化公司搜索引擎营销优化诊断训练

企业排名优化公司,搜索引擎营销优化诊断训练,台州网站推广,建设工程施工合同2013Linux驱动-块设备驱动 一,块设备驱动简介二,无请求队列情况(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/202408.html

相关文章:

  • 手机域名注册查询seo短视频网页入口引流网站
  • wordpress获取图片的绝对地址seo视频教程汇总
  • 做电影网站还能赚钱深圳市文化广电旅游体育局
  • 英文网站建设成都十大广告公司排名
  • 某购物网站开发项目设计网站大全
  • 杭州网站建设优化案例建网站找哪个平台好呢
  • 网站建设销售怎么做宁波seo外包
  • 做网站后有人抢注关键词深圳谷歌seo公司
  • 做视频给网站到流量域名是什么 有什么用
  • 东凤镇 网站建设谷歌推广seo
  • 网站上怎么做全景看图app营销策划方案
  • 网站如何做分站宁波网站推广优化哪家正规
  • 网站哪些页面会做静态化搜索引擎优化工作
  • 三拼域名做网站青岛seo网站推广
  • 高端网站建设968排名网站
  • 手机网站怎么开发工具搜索引擎seo
  • 绍兴市越城区建设局网站如何制作一个自己的网页网站
  • 学做美食的视频网站有哪些小说网站排名前十
  • 如何做彩票网站信息全网关键词指数查询
  • 网站关键词策略个人免费域名注册网站
  • 上海做网站哪里有今日小说搜索风云榜
  • 外贸网站平台seo推广嘉兴网站建设
  • 青岛网站制作游戏代理平台
  • wordpress网站监测营销心得体会感悟300字
  • 凡科网站建设7个基本流程爱站权重
  • 网站建设框架构建网络推广公司经营范围
  • 网站开发员招聘全球搜索网站排名
  • 如何对网站的文件和资源进行优化?seo学徒
  • 真人百家樂网站建设查询关键词
  • 营口 建设街道 政府网站网络安全培训最强的机构