ll_rw_block是文件系统对下访问实际的块设备驱动的接口,应用程序对实际文件(非设备文件)的操作,最终都是
通过文件系统来调用ll_rw_block来操作实际的存储设备的。
当然ll_rw_block的实际作用远非一个接口那么简单,他里面对维护了一个读写请求队列,并且对队列里读写请求进
行了优化,比如,对机械硬盘来说,使用了电梯算法,对队列里的读写请求进行优化,排序,合并等操作,提高了
访问硬盘的效率。优化后,最后调用实际的块设备驱动程序来操作真正的存储设备。
下面,以bbb的TI官方linux源码(sdk6)来大概将ll_rw_block的流程写一下,以免忘记。
void ll_rw_block(int rw, int nr, struct buffer_head *bhs[]) // fs/buffer.c
{
//rw为读写标志,nr为bhs数组的长度, bhs为实际要对写数据的信息数组,源,目的,长度
for (i = 0; i < nr; i++) {
struct buffer_head *bh = bhs[i];
if (rw == WRITE) {
submit_bh(WRITE, bh); //提交写请求
} else {
submit_bh(rw, bh); //提交读请求
}
}
}
int submit_bh(int rw, struct buffer_head * bh) // fs/buffer.c
{
struct bio *bio;
//这里省略了一系列的使用bh来构造bio的代码,最终提交bio
bio_get(bio);
submit_bio(rw, bio); //提交bio
}
void submit_bio(int rw, struct bio *bio) //block/blk-core.c
{
generic_make_request(bio); //使用bio来构造请求
}
void generic_make_request(struct bio *bio) //block/blk-core.c
{
struct request_queue *q = bdev_get_queue(bio->bi_bdev); //获得请求队列
q->make_request_fn(q, bio); //调用请求队列的make_request_fn
}
make_request_fn函数的定义如下:
void blk_queue_bio(struct request_queue *q, struct bio *bio) //block/blk-core.c
{
el_ret = elv_merge(q, &req, bio); //调用电梯算法对读写操作进行优化处理
__blk_run_queue(q); //看下面
}
void __blk_run_queue(struct request_queue *q) //block/blk-core.c
{
q->request_fn(q); //这里最终调用请求队列的request_fn函数进行真正的磁盘操作
}