linux块设备驱动之实例
1、注册:向内核注册个块设备驱动,其实就是用主设备号告诉内核这个代表块设备驱动
- sbull_major = register_blkdev(sbull_major, "sbull");
- if (0 >= sbull_major){
- printk(KERN_WARNING "sbull: unable to get major number!\n");
- return -EBUSY;
- }
2、定义设备结构体:
- struct sbull_dev{
- int size; // 以扇区为单位,设备的大小
- u8 *data; // 数据数组
- short users; // 用户数目
- short media_change; // 介质改变标识
- spinlock_t lock; // 用于互斥
- struct request_queue *queue; // 设备请求队列
- struct gendisk *gd; // gendisk结构
- struct timer_list time; // 用来模拟介质改变
- };
3、初始化设备结构体:
- memset(dev, 0, sizeof(struct sbull_dev));
- dev->size = nsectors * hardsect_size;
- dev->data = vmalloc(dev->size);
- if (dev->data == NULL){
- printk(KERN_NOTICE "vmalloc failure.\n");
- return;
- }
- spin_lock_init(&dev->lock);//初始化自旋锁,为了下一步的队列分配
4、创建设备的请求队列:
dev->queue = blk_init_queue(sbull_request, &dev->lock);
5、分配、初始化及安装相应的gendisk结构:
- dev->gd = alloc_disk(SBULL_MINORS);
- if (!dev->gd) {
- printk (KERN_NOTICE "alloc_disk failure.\n");
- goto out_vfree;
- }
- dev->gd->major = sbull_major;
- dev->gd->first_minor = which*SBULL_MINORS;
- dev->gd->fops = &sbull_ops;
- dev->gd->queue = dev->queue;
- dev->gd->private_data= dev;
- snprintf(dev->gd->disk_name, 32, "sbull%c", which + 'a');
- set_capacity(dev->gd, nsectors*(hardsect_size/KERNEL_SECTOR_SIZE));//使用KERNEL_SECTOR_SIZE本地常量,进行内核512字节扇区到实际使用的扇区大小转换
- add_disk(dev->gd);
SBULL_MINORS是每个设备所支持的次设备号的数量,地一个设备名为 sbulla,第二个为sbullb.用户空间可以添加分区,第二个设备上的第三个分区可能是 /dev/sbullb3。
6、设置队列支持的扇区大小
通知内核设备所支持的扇区大小,硬件扇区大小作为一个参数放在队列中,而不是在gendisk中。当分配好队列后就要马上调用下面函数:
blk_queue_hardsect_size(dev->queue, hardset_size);
调用了上面的函数后,内核就会对我们的设备使用设定的硬件扇区大小,所有的I/O请求毒定位在硬件扇区的开始位置,并且每个请求的大小都将是扇区大小的整数倍。记住:内核总是认为扇区大小是512字节,因此必须将所有的扇区数进行转换。
7、实现操作函数:
打开设备函数:
- static int sbull_open(struct inode *inode, struct file *filp)
- {
- struct sbull_dev *dev = inode->i_bdev->bd_disk->private_data;
- del_timer_sync(&dev->timer);//移除定时器
- filp->private_data = dev;
- spin_lock(&dev->lock);
- if (!dev->users)
- check_disk_change(inode->i_bdev);//检查驱动器中的介质是否改变
- dev->users++;// 增加用户计数
- spin_unlock(&dev->lock);
- return 0;
- }
关闭设备函数:
- static int sbull_release(struct inode *inode, struct file *filp)
- {
- struct sbull_dev *dev = inode->i_bdev->bd_disk->private_data;
- spin_lock(&dev->lock);
- dev->users--;
- if (!dev->users){
- dev->timer.expires = jiffies + INVALIDATE_DELAY;//设置定时器
- add_timer(&dev->timer);
- }
- spin_unlock(&dev->lock);
- return 0;
- }
其他的函数也是一样实现,和字符设备驱动的类似。这里就不写了,接下来看看核心部分,对于一个块设备驱动来说核心部分就是请求,几乎所有的重心都在请求函数;
8、处理请求操作
- dev->queue = blk_queue_init(&sbull_request, &dev->lock);
- static void sbull_request(request_queue_t *q)
- {
- struct request *req;
- while((req = elv_next_request(q)) != NULL){//获取队列中第一个未完成的请求,没有则返回NULL。处理完后不删除该请求
- struct sbull_dev *dev = req->rq_disk->private_data;
- if (! blk_fs_request(req)){// 判断是否是一个文件系统请求,即是不是块设备请求
- printk(KERN_NOTICE "skip non-fs request.\n");
- end_request(req, 0);
- continue;
- }
- //sbull_transfer()函数是真正的处理块设备请求函数
- sbull_transfer(dev, req->sector, req->current_nr_sectors, req->buffer, rq_data_dir(req));
- end_request(req, 1);
- }
- }
- void end_request(struct request* req, int succeeded);
- // sector开始扇区的索引号,指的是512字节的扇区,如果是2048字节的扇区,则要sector/4再传递
- // nsect 表示要传递多少个扇区; buffer 数据缓存的地址指针;write 表示数据传递的方向,即:read/write;
- static void sbull_transfer(struct sbull_dev *dev, unsigned long sector, unsigned long nsect, char *buffer, int write)
- {
- unsigned long offset = sector*KERNEL_SECTOR_SIZE;
- unsigned long nbytes = nsect*KERNEL_SECTOR_SIZE;
- if ((offset + nbytes) > dev->size){
- printk(KERN_NOTICE "Beyond-end write (%ld %ld)\n", offset, nbytes);
- return;
- }
- if (write)
- memcpy(dev->data + offset, buffer, nbytes);
- else
- memcpy(buffer, dev->data + offset, nbytes);
- }

转载地址:linux块设备驱动之实例
http://blog.csdn.net/yuzhihui_no1/article/details/46808947
linux块设备驱动之实例的更多相关文章
- Linux块设备驱动详解
<机械硬盘> a:磁盘结构 -----传统的机械硬盘一般为3.5英寸硬盘,并由多个圆形蝶片组成,每个蝶片拥有独立的机械臂和磁头,每个堞片的圆形平面被划分了不同的同心圆,每一个同心圆称为一个 ...
- Linux块设备驱动(二) _MTD驱动及其用户空间编程
MTD(Memory Technology Device)即常说的Flash等使用存储芯片的存储设备,MTD子系统对应的是块设备驱动框架中的设备驱动层,可以说,MTD就是针对Flash设备设计的标准化 ...
- linux块设备驱动
块设备驱动程序<1>.块设备和字符设备的区别1.读取数据的单元不同,块设备读写数据的基本单元是块,字符设备的基本单元是字节.2.块设备可以随机访问,字符设备只能顺序访问. 块设备的访问:当 ...
- linux块设备驱动---相关结构体(转)
上回最后面介绍了相关数据结构,下面再详细介绍 块设备对象结构 block_device 内核用结构block_device实例代表一个块设备对象,如:整个硬盘或特定分区.如果该结构代表一个分区,则其成 ...
- linux块设备驱动---概念与框架(转)
基本概念 块设备(blockdevice) --- 是一种具有一定结构的随机存取设备,对这种设备的读写是按块进行的,他使用缓冲区来存放暂时的数据,待条件成熟后,从缓存一次性写入设备或者从设备一次性 ...
- Linux 块设备驱动 (一)
1.块设备的I/O操作特点 字符设备与块设备的区别: 块设备只能以块为单位接受输入和返回输出,而字符设备则以字符为单位. 块设备对于I/O请求有对应的缓冲区,因此它们可以选择以什么顺序进行响应,字符设 ...
- Linux块设备驱动(一) _驱动模型
块设备是Linux三大设备之一,其驱动模型主要针对磁盘,Flash等存储类设备,本文以3.14为蓝本,探讨内核中的块设备驱动模型 框架 下图是Linux中的块设备模型示意图,应用层程序有两种方式访问一 ...
- Linux块设备驱动_WDS
推荐书:<Linux内核源代码情景分析> 1.字符设备驱动和使用中等待某一事件的方法①查询方式②休眠唤醒,但是这种没有超时时间③poll机制,在休眠唤醒基础上加一个超时时间④异步通知,异步 ...
- linux 块设备驱动 (三)块设备驱动开发
一: 块设备驱动注册与注销 块设备驱动中的第1个工作通常是注册它们自己到内核,完成这个任务的函数是 register_blkdev(),其原型为:int register_blkdev(unsigne ...
随机推荐
- 给HashMap排序,使之成为有序Map
如何给HashMap中的值排序?这个问题很多人都遇到过,很常见的一个方案是使用LinkedHashMap,因为LinkedHashMap可以记住元素放入的顺序,可以认为是真正的“有序”(想让HashM ...
- __attribute__
转来的: http://www.cnblogs.com/astwish/p/3460618.html __attribute__ 你知多少? GNU C 的一大特色就是__attribute__ 机制 ...
- VC6.0中友元函数无法访问类私有成员的解决办法
举个例子: 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 #inclu ...
- sql 中set和select区别
基于SQL中SET与SELECT赋值的区别详解 2012年09月06日 ⁄ 综合 ⁄ 共 912字 ⁄ 字号 小 中 大 ⁄ 评论关闭 最近的项目写的SQL比较多,经常会用到对变量赋值,而我使用SET ...
- git安装及命令使用和github网站
最近参与别人的github项目时,学习了git的使用,首先需要在https://github.com/网站上注册账号和邮箱,然后fork一个开源项目,然后下载目前Windows下最新版本的git,下载 ...
- Eclipse的安装与调试
1.首先进入eclipse官网下载eclipse 2.根据自己的系统选择相应版本进行下载.由于我的是64bit,所以我选择64bit版下载. 3.点开下载的程序,进行安装. 4.随后进入了安装页面.和 ...
- 在Page_Loaded下删除PivotItem出错的解决方案
之前我一个例子中出现无法再页面Loaded事件中删除PivotItem的情况,页面会报错 Value does not fall within the expected range. 附图 原因是因为 ...
- (一)NOR FALSH 讲解
nor flash 起始地址: 1F FFFF = 1 1111 1111 1111 1111 1111 = 2^20 *2 = 2^21=2M Nor Flash 采用了 A0- ...
- 搭建Android底层开发环境
为了开发linux驱动方便些,我们一般将linux作为Android的开发环境,那么就需要搭建Android的开发环境,下面是一些搭建Android底层时的心得: (1)安装JDK:除了普遍使用的下载 ...
- 滚动轮播效果,.net 没得混看来只能去写js 了
<!DOCTYPE html> <html> <head> <title> 滚动图片 </title> <style type=&qu ...