⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ll_rw_blk.c

📁 带有中文注释的linux内核源码
💻 C
字号:
/** linux/kernel/blk_dev/ll_rw.c** (C) 1991 Linus Torvalds*//** This handles all read/write requests to block devices*//** 该程序处理块设备的所有读/写操作。*/#include <errno.h>		// 错误号头文件。包含系统中各种出错号。(Linus 从minix 中引进的)#include <linux/sched.h>	// 调度程序头文件,定义了任务结构task_struct、初始任务0 的数据,// 还有一些有关描述符参数设置和获取的嵌入式汇编函数宏语句。#include <linux/kernel.h>	// 内核头文件。含有一些内核常用函数的原形定义。#include <asm/system.h>		// 系统头文件。定义了设置或修改描述符/中断门等的嵌入式汇编宏。#include "blk.h"		// 块设备头文件。定义请求数据结构、块设备数据结构和宏函数等信息。/** The request-struct contains all necessary data* to load a nr of sectors into memory*//** 请求结构中含有加载nr 扇区数据到内存的所有必须的信息。*/struct request request[NR_REQUEST];/** used to wait on when there are no free requests*//* 是用于请求数组没有空闲项时的临时等待处 */struct task_struct *wait_for_request = NULL;/* blk_dev_struct is:* do_request-address* next-request*//* blk_dev_struct 块设备结构是:(kernel/blk_drv/blk.h,23)* do_request-address //对应主设备号的请求处理程序指针。* current-request // 该设备的下一个请求。*/// 该数组使用主设备号作为索引(下标)。struct blk_dev_struct blk_dev[NR_BLK_DEV] = {  {NULL, NULL},			/* no_dev */// 0 - 无设备。  {NULL, NULL},			/* dev mem */// 1 - 内存。  {NULL, NULL},			/* dev fd */// 2 - 软驱设备。  {NULL, NULL},			/* dev hd */// 3 - 硬盘设备。  {NULL, NULL},			/* dev ttyx */// 4 - ttyx 设备。  {NULL, NULL},			/* dev tty */// 5 - tty 设备。  {NULL, NULL}			/* dev lp */// 6 - lp 打印机设备。};// 锁定指定的缓冲区bh。如果指定的缓冲区已经被其它任务锁定,则使自己睡眠(不可中断地等待),// 直到被执行解锁缓冲区的任务明确地唤醒。static inline voidlock_buffer (struct buffer_head *bh){  cli ();			// 清中断许可。  while (bh->b_lock)		// 如果缓冲区已被锁定,则睡眠,直到缓冲区解锁。    sleep_on (&bh->b_wait);  bh->b_lock = 1;		// 立刻锁定该缓冲区。  sti ();			// 开中断。}// 释放(解锁)锁定的缓冲区。static inline voidunlock_buffer (struct buffer_head *bh){  if (!bh->b_lock)		// 如果该缓冲区并没有被锁定,则打印出错信息。    printk ("ll_rw_block.c: buffer not locked\n\r");  bh->b_lock = 0;		// 清锁定标志。  wake_up (&bh->b_wait);	// 唤醒等待该缓冲区的任务。}/** add-request adds a request to the linked list.* It disables interrupts so that it can muck with the* request-lists in peace.*//** add-request()向连表中加入一项请求。它关闭中断,* 这样就能安全地处理请求连表了 */*///// 向链表中加入请求项。参数dev 指定块设备,req 是请求的结构信息。  static voidadd_request (struct blk_dev_struct *dev, struct request *req){  struct request *tmp;  req->next = NULL;  cli ();			// 关中断。  if (req->bh)    req->bh->b_dirt = 0;	// 清缓冲区“脏”标志。// 如果dev 的当前请求(current_request)子段为空,则表示目前该设备没有请求项,本次是第1 个// 请求项,因此可将块设备当前请求指针直接指向请求项,并立刻执行相应设备的请求函数。  if (!(tmp = dev->current_request))    {      dev->current_request = req;      sti ();			// 开中断。      (dev->request_fn) ();	// 执行设备请求函数,对于硬盘(3)是do_hd_request()。      return;    }// 如果目前该设备已经有请求项在等待,则首先利用电梯算法搜索最佳位置,然后将当前请求插入// 请求链表中。  for (; tmp->next; tmp = tmp->next)    if ((IN_ORDER (tmp, req) ||	 !IN_ORDER (tmp, tmp->next)) && IN_ORDER (req, tmp->next))      break;  req->next = tmp->next;  tmp->next = req;  sti ();}//// 创建请求项并插入请求队列。参数是:主设备号major,命令rw,存放数据的缓冲区头指针bh。static voidmake_request (int major, int rw, struct buffer_head *bh){  struct request *req;  int rw_ahead;/* WRITEA/READA is special case - it is not really needed, so if the *//* buffer is locked, we just forget about it, else it's a normal read *//* WRITEA/READA 是特殊的情况 - 它们并不是必要的,所以如果缓冲区已经上锁,*//* 我们就不管它而退出,否则的话就执行一般的读/写操作。 */// 这里'READ'和'WRITE'后面的'A'字符代表英文单词Ahead,表示提前预读/写数据块的意思。// 当指定的缓冲区正在使用,已被上锁时,就放弃预读/写请求。  if (rw_ahead = (rw == READA || rw == WRITEA))    {      if (bh->b_lock)	return;      if (rw == READA)	rw = READ;      else	rw = WRITE;    }// 如果命令不是READ 或WRITE 则表示内核程序有错,显示出错信息并死机。  if (rw != READ && rw != WRITE)    panic ("Bad block dev command, must be R/W/RA/WA");// 锁定缓冲区,如果缓冲区已经上锁,则当前任务(进程)就会睡眠,直到被明确地唤醒。  lock_buffer (bh);// 如果命令是写并且缓冲区数据不脏,或者命令是读并且缓冲区数据是更新过的,则不用添加// 这个请求。将缓冲区解锁并退出。  if ((rw == WRITE && !bh->b_dirt) || (rw == READ && bh->b_uptodate))    {      unlock_buffer (bh);      return;    }repeat:/* we don't allow the write-requests to fill up the queue completely:* we want some room for reads: they take precedence. The last third* of the requests are only for reads.*//* 我们不能让队列中全都是写请求项:我们需要为读请求保留一些空间:读操作* 是优先的。请求队列的后三分之一空间是为读准备的。*/// 请求项是从请求数组末尾开始搜索空项填入的。根据上述要求,对于读命令请求,可以直接// 从队列末尾开始操作,而写请求则只能从队列的2/3 处向头上搜索空项填入。  if (rw == READ)    req = request + NR_REQUEST;	// 对于读请求,将队列指针指向队列尾部。  else    req = request + ((NR_REQUEST * 2) / 3);	// 对于写请求,队列指针指向队列2/3 处。/* find an empty request *//* 搜索一个空请求项 */// 从后向前搜索,当请求结构request 的dev 字段值=-1 时,表示该项未被占用。  while (--req >= request)    if (req->dev < 0)      break;/* if none found, sleep on new requests: check for rw_ahead *//* 如果没有找到空闲项,则让该次新请求睡眠:需检查是否提前读/写 */// 如果没有一项是空闲的(此时request 数组指针已经搜索越过头部),则查看此次请求是否是// 提前读/写(READA 或WRITEA),如果是则放弃此次请求。否则让本次请求睡眠(等待请求队列// 腾出空项),过一会再来搜索请求队列。  if (req < request)    {				// 如果请求队列中没有空项,则      if (rw_ahead)	{			// 如果是提前读/写请求,则解锁缓冲区,退出。	  unlock_buffer (bh);	  return;	}      sleep_on (&wait_for_request);	// 否则让本次请求睡眠,过会再查看请求队列。      goto repeat;    }/* fill up the request-info, and add it to the queue *//* 向空闲请求项中填写请求信息,并将其加入队列中 */// 请求结构参见(kernel/blk_drv/blk.h,23)。  req->dev = bh->b_dev;		// 设备号。  req->cmd = rw;		// 命令(READ/WRITE)。  req->errors = 0;		// 操作时产生的错误次数。  req->sector = bh->b_blocknr << 1;	// 起始扇区。(1 块=2 扇区)  req->nr_sectors = 2;		// 读写扇区数。  req->buffer = bh->b_data;	// 数据缓冲区。  req->waiting = NULL;		// 任务等待操作执行完成的地方。  req->bh = bh;			// 缓冲区头指针。  req->next = NULL;		// 指向下一请求项。  add_request (major + blk_dev, req);	// 将请求项加入队列中(blk_dev[major],req)。}//// 低层读写数据块函数。// 该函数主要是在fs/buffer.c 中被调用。实际的读写操作是由设备的request_fn()函数完成。// 对于硬盘操作,该函数是do_hd_request()。(kernel/blk_drv/hd.c,294)voidll_rw_block (int rw, struct buffer_head *bh){  unsigned int major;		// 主设备号(对于硬盘是3)。// 如果设备的主设备号不存在或者该设备的读写操作函数不存在,则显示出错信息,并返回。  if ((major = MAJOR (bh->b_dev)) >= NR_BLK_DEV ||      !(blk_dev[major].request_fn))    {      printk ("Trying to read nonexistent block-device\n\r");      return;    }  make_request (major, rw, bh);	// 创建请求项并插入请求队列。}//// 块设备初始化函数,由初始化程序main.c 调用(init/main.c,128)。// 初始化请求数组,将所有请求项置为空闲项(dev = -1)。有32 项(NR_REQUEST = 32)。voidblk_dev_init (void){  int i;  for (i = 0; i < NR_REQUEST; i++)    {      request[i].dev = -1;      request[i].next = NULL;    }}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -