在块设备驱动的模块家在函数中通常需要完成如下工作:
1. 分配、初始化请求队列,绑定请求队列和请求函数。
2. 分配、初始化gendisk,给gendisk的major、fops、queue等成员赋值,最后添加gendisk。
3. 注册块设备驱动。
代码清单13.9和13.10分别给出了使用blk_alloc_queue()分配请求队列并使用blk_queue_make_request()绑定“请求队列”和“制造请求”的函数,以及使用blk_init_queue()初始化请求队列并绑定请求队列与请求处理函数两种不同情况下的块设备驱动模块加载函数模板。
代码清单13.9 块设备驱动的模块加载函数模板(使用blk_alloc_queue)
static int _ _init xxx_init(void)
{
/*分配gendisk*/
xxx_disks = alloc_disk(1);
if (!xxx_disks)
goto out;
/* 块设备驱动注册 */
if ( register_blkdev(XXX_MAJOR, "xxx") ) {
err = -EIO;
goto out;
}
/*"请求队列" 分配 */
xxx_queue = blk_alloc_queue(GFP_KERNEL);
if (!xxx_queue)
goto out_queue;
blk_queue_make_request(xxx_queue, &xxx_make_request);//绑定“制造请求“函数
blk_queue_hardsect_size(xxx_queue, xxx_blocksize);//硬件扇区尺寸设置
/*gendisk 初始化*/
xxx_disks->major = XXX_MAJOR;
xxx_disks->first_minor = 0;
xxx_disks->fops = &xxx_op;
xxx_disks->queue = xxx_queue;
sprintf( xxx_disks->disk_name, "xxx%d", i );
add_disk(xxx_disks); //添加gendisk
return 0;
out_queue: unregister_blkdev(XXX_MAJOR, "xxx" );
out : put_disk(xxx_disks);
blk_cleanup_queue(xxx_queue);
return -ENOMEM;
}
代码清单13.10 块设备驱动的模块加载函数模板(使用blk_init_queue)
static int _ _init xxx_init(void)
{
/*块设备驱动注册*/
if (register_blkdev(XXX_MAJOR, "xxx" )) {
err = -EIO;
goto out;
}
/* 请求队列初始化*/
xxx_queue = blk_init_queue(xxx_request, xx_lock);
if (!xxx_queue)
goto out_queue;
blk_queue_hardsect_size(xxx_queue, xxx_blocksize);//硬件扇区尺寸设置
/*gendisk 初始化 */
xxx_disks->major = XXX_MAJOR;
xxx_disks->first_minor = 0;
xxx_disks->fops = &xxx_op;
xxx_disks->queue = xxx_queue;
sprintf(xxx_disks->disk_name, "xxx%d", i);
set_capacity(xxx_disks, xxx_size * 2);
add_disk(xxx_disks); //添加gendisk
return 0;
out_queue: unregister_blkdev(XXX_MAJOR, "xxx");
out : put_disk(xxx_disks);
blk_cleanup_queue(xxx_queue);
return - ENOMEM;
}
代码清单13.11 块设备驱动的模块卸载函数模板
static void _ _exit xxx_exit(void)
{
if (bdev) {
invalidate_bdev(xxx_bdev, 1);
blkdev_put(xxx_bdev);
}
del_gendisk(xxx_disks); //删除gendisk
put_disk(xxx_disks);
blk_cleanup_queue(xxx_queue[i] );//清除请求队列
unregister_blkdev(XXX_MAJOR, "xxx");
}