快设备驱动程序之是内存模拟的磁盘支持格式化
2012-07-10 18:06
267 查看
在块设备驱动程序分析之框架构建那一节里面已经提到过 block_device_operations这样一个结构体。顾名思义,它的作用就是来进行块设备的处理,对应的就是各种函数,但是并不对应块设备的读写处理函数,这是因为对块设备的读写在别的地方完成了,具体来说就是在request函数中被完成了,对于这个函数,我们今后会来分析的,现在先放一放。
在本节里面我们要完成的功能是使块设备支持分区,对块设备的这种操作由函数.getgeo来完成。这个函数是为一些没有磁盘特性的块设备构造磁盘的一些属性,代码如下:
#include <linux/errno.h>
#include <linux/interrupt.h>
#include <linux/mm.h>
#include <linux/fs.h>
#include <linux/kernel.h>
#include <linux/timer.h>
#include <linux/genhd.h>
#include <linux/hdreg.h>
#include <linux/ioport.h>
#include <linux/init.h>
#include <linux/wait.h>
#include <linux/blkdev.h>
#include <linux/blkpg.h>
#include <linux/delay.h>
#include <linux/io.h>
#include <asm/system.h>
#include <asm/uaccess.h>
#include <asm/dma.h>
static struct gendisk *ramblock_disk;
static request_queue_t *ramblock_queue;
static unsigned char *ramblock_buf;
static int major;
#define RAMBLOCK_SIZE (1024*1024)
static DEFINE_SPINLOCK(ramblock_lock);
static int ramblock_getgeo(struct block_device *bdev, struct hd_geometry *geo)
{
/* 容量=heads*cylinders*sectors*512 */
geo->heads = 2; //表示有2个面
geo->cylinders = 32;//表示每个面有32个环
geo->sectors = RAMBLOCK_SIZE/2/32/512;//表示每个环有多少扇区
return 0;
}
static struct block_device_operations ramblock_fops = {
.owner
= THIS_MODULE,
.getgeo
= ramblock_getgeo,
};
#define RAMBLOCK_SIZE (1024*1024)
static void do_ramblock_request(request_queue_t * q)
{
//static int cnt = 0;
struct request *req;
//printk("do_ramblock_request %d\n", ++cnt);
while ((req = elv_next_request(q)) != NULL) {
unsigned long offset = req->sector * 512;
unsigned long len = req->current_nr_sectors * 512;
if (rq_data_dir(req) == READ)
{
memcpy(req->buffer, ramblock_buf+offset, len);
}
else
{
memcpy(ramblock_buf+offset, req->buffer, len);
}
end_request(req, 1);
}
}
static int ramblock_init(void)
{
ramblock_disk = alloc_disk(16);
ramblock_queue = blk_init_queue(do_ramblock_request, &ramblock_lock);
ramblock_disk->queue = ramblock_queue;
major = register_blkdev(0, "ramblock");
ramblock_disk->major = major;
ramblock_disk->first_minor = 0;
sprintf(ramblock_disk->disk_name, "ramblock");
ramblock_disk->fops = &ramblock_fops;
set_capacity(ramblock_disk, RAMBLOCK_SIZE / 512);
ramblock_buf = kzalloc(RAMBLOCK_SIZE, GFP_KERNEL);
add_disk(ramblock_disk);
return 0;
}
static void ramblock_exit(void)
{
unregister_blkdev(major, "ramblock");
del_gendisk(ramblock_disk);
put_disk(ramblock_disk);
blk_cleanup_queue(ramblock_queue);
kfree(ramblock_buf);
}
module_init(ramblock_init);
module_exit(ramblock_exit);
MODULE_LICENSE("GPL");
测试:
(1) insmod ramblock.ko
(2) ls /dev/ramblock* 信息如下:brw-rw---- 1 0 0 254, 0 Jan 1 02:26 /dev/ramblock
(3)分区:fdisk /dev/ramblock
输入:m 可以查看帮助信息
输入:n 用于创建新的分区
输入:p 用于建立主分区
然后根据提示可以创建分区。
在本节里面我们要完成的功能是使块设备支持分区,对块设备的这种操作由函数.getgeo来完成。这个函数是为一些没有磁盘特性的块设备构造磁盘的一些属性,代码如下:
#include <linux/errno.h>
#include <linux/interrupt.h>
#include <linux/mm.h>
#include <linux/fs.h>
#include <linux/kernel.h>
#include <linux/timer.h>
#include <linux/genhd.h>
#include <linux/hdreg.h>
#include <linux/ioport.h>
#include <linux/init.h>
#include <linux/wait.h>
#include <linux/blkdev.h>
#include <linux/blkpg.h>
#include <linux/delay.h>
#include <linux/io.h>
#include <asm/system.h>
#include <asm/uaccess.h>
#include <asm/dma.h>
static struct gendisk *ramblock_disk;
static request_queue_t *ramblock_queue;
static unsigned char *ramblock_buf;
static int major;
#define RAMBLOCK_SIZE (1024*1024)
static DEFINE_SPINLOCK(ramblock_lock);
static int ramblock_getgeo(struct block_device *bdev, struct hd_geometry *geo)
{
/* 容量=heads*cylinders*sectors*512 */
geo->heads = 2; //表示有2个面
geo->cylinders = 32;//表示每个面有32个环
geo->sectors = RAMBLOCK_SIZE/2/32/512;//表示每个环有多少扇区
return 0;
}
static struct block_device_operations ramblock_fops = {
.owner
= THIS_MODULE,
.getgeo
= ramblock_getgeo,
};
#define RAMBLOCK_SIZE (1024*1024)
static void do_ramblock_request(request_queue_t * q)
{
//static int cnt = 0;
struct request *req;
//printk("do_ramblock_request %d\n", ++cnt);
while ((req = elv_next_request(q)) != NULL) {
unsigned long offset = req->sector * 512;
unsigned long len = req->current_nr_sectors * 512;
if (rq_data_dir(req) == READ)
{
memcpy(req->buffer, ramblock_buf+offset, len);
}
else
{
memcpy(ramblock_buf+offset, req->buffer, len);
}
end_request(req, 1);
}
}
static int ramblock_init(void)
{
ramblock_disk = alloc_disk(16);
ramblock_queue = blk_init_queue(do_ramblock_request, &ramblock_lock);
ramblock_disk->queue = ramblock_queue;
major = register_blkdev(0, "ramblock");
ramblock_disk->major = major;
ramblock_disk->first_minor = 0;
sprintf(ramblock_disk->disk_name, "ramblock");
ramblock_disk->fops = &ramblock_fops;
set_capacity(ramblock_disk, RAMBLOCK_SIZE / 512);
ramblock_buf = kzalloc(RAMBLOCK_SIZE, GFP_KERNEL);
add_disk(ramblock_disk);
return 0;
}
static void ramblock_exit(void)
{
unregister_blkdev(major, "ramblock");
del_gendisk(ramblock_disk);
put_disk(ramblock_disk);
blk_cleanup_queue(ramblock_queue);
kfree(ramblock_buf);
}
module_init(ramblock_init);
module_exit(ramblock_exit);
MODULE_LICENSE("GPL");
测试:
(1) insmod ramblock.ko
(2) ls /dev/ramblock* 信息如下:brw-rw---- 1 0 0 254, 0 Jan 1 02:26 /dev/ramblock
(3)分区:fdisk /dev/ramblock
输入:m 可以查看帮助信息
输入:n 用于创建新的分区
输入:p 用于建立主分区
然后根据提示可以创建分区。
相关文章推荐
- 快设备驱动程序之是内存模拟的磁盘支持格式化
- 编写块设备驱动之内存模拟磁盘
- 块设备驱动1—用内存模拟磁盘
- Tiny6410开发板下块设备驱动程序的编写驱动之用内存模拟磁盘(二)
- 内存模拟磁盘,让你体验下急速
- 嵌入式Linux驱动学习之路(二十二)用内存模拟磁盘
- 自己封装的一个iphone上的缓存,支持内存缓存以及磁盘缓存,实际项目已经应用。
- 块设备驱动程序分析之内存模拟磁盘
- arm 驱动进阶:块设备驱动程序程序设计之用内存模拟磁盘
- Tiny6410开发板下块设备驱动程序的编写驱动之用内存模拟磁盘(一)
- 块设备驱动程序分析之内存模拟磁盘
- 深入理解云存储和云计算-内存和磁盘
- [Linux] 磁盘的分区、格式化、检验与挂载
- 如何测试java支持的最大内存
- WIN7 不用格式化磁盘怎么把FAT32系统改成NTFS系统
- Python功能模块之psutil------ Linux性能(CPU、磁盘、内存、网卡)监控
- 使用kyototycoon挂载leveldb,映射内存磁盘的使用心得
- 格式化文本支持:JTextPane
- Android之ListView异步加载网络图片(优化缓存机制)和对图片资源进行优化,并且实现内存双缓存 + 磁盘缓存
- Glide 缓存策略 内存缓存和磁盘缓存