7. Buffer and Multiprocess File 343
7.5 Function of the Count, Lock, Wait, Request
7.5.3 Function of b_lock and *b_wait
The kernel applies a buffer block for the process, especially when the b_count in the buffer block turns out to be 0, and considering the synchronization, there is great possibility that this buffer block is transacting data with the hard disk. So the b_lock field is set in this buffer_head structure. If this field is set to be 1, it means this buffer block is interacting data with the hard disk. The kernel will stop the process from conducting operations to the buffer block until the end of the interaction with the hard disk. The interception will be relieved when the field is set to 0.
If the b_lock field in the buffer block, which is applied by the process, is set to 1, the process also needs to be suspended even though the process has already got the buffer block. The buf- fer block can only be accessed until it is unlocked. When the buffer block is locked, no matter how many processes had applied the buffer block, they cannot immediately operate the buffer block. All these processes should be suspended and switch to other processes to execute. We need to record which processes are suspended while waiting for this buffer block. Due to the use of a waiting queue of the process, a field can solve this record. The field is *b_wait.
These two fields are often used in combination. The specific code is as follows:
b_lock is set to 0, and *b_wait is set to null when we initialize the buffer block.
After the buffer block is applied, this block should be locked before the operation on the bottom layer, which means b_lock should be set to 1. The specific code is as follows:
//code path:fs/buffer.c:
void buffer_init(long buffer_end) {
……
h->b_dev = 0;
h->b_dirt = 0;
h->b_count = 0;
h->b_lock = 0;
h->b_uptodate = 0;
h->b_wait = NULL;
h->b_next = NULL;
h->b_prev = NULL;
h->b_data = (char *) b;
h->b_prev_free = h-1;
h->b_next_free = h+1;
……
}
//code path:kernel/blk_drv/ll_rw_block.c:
static void make_request(int major,int rw, struct buffer_head * bh) {
……
if (rw! = READ && rw! = WRITE)
panic(“Bad block dev command, must be R/W/RA/WA”);
lock_buffer(bh); //lock the buffer block if ((rw = = WRITE && !bh->b_dirt) || (rw = = READ && bh->b_
uptodate)) {
Before the hard disk data blocks start transacting data with the buffer blocks, the function lock_buffer() will judge whether the buffer block is locked. If it is locked, it is possible that this buffer block is already applied by other processes, and it is transacting data with the hard disk. So we could call function sleep_on() to suspend the process and switch to the other processes to execute. Lock the buffer block again to prevent it from being misused by other process when we switch back to the current process later. B_lock and *b_wait’s combination is not only reflected here, but also all the conditions when the buffer block’s state should be determined. The specific code is as follows:
unlock_buffer(bh);
return;
}
……
}
static inline void lock_buffer(struct buffer_head * bh) {
cli();
while (bh->b_lock) //if the buffer block is already locked sleep_on(&bh->b_wait); //suspend the process directly bh->b_lock = 1; //lock the buffer block sti();
}
//code path:fs/buffer.c:
struct buffer_head * bread(int dev,int block) {
struct buffer_head * bh;
if (!(bh = getblk(dev,block)))
panic(“bread: getblk returned NULL\n”);
if (bh->b_uptodate) return bh;
ll_rw_block(READ,bh);
wait_on_buffer(bh); //Detect whether process need to //wait for the buffer block to be //unlock or not
if (bh->b_uptodate) return bh;
brelse(bh);
return NULL;
}
static inline void lock_buffer(struct buffer_head * bh) {
cli();
while (bh->b_lock) //if the buffer block is already locked sleep_on(&bh->b_wait); //suspend the process
bh->b_lock = 1; //lock the buffer block sti();
}
They should be used in combination when we lock the buffer and suspend the process.
And they are also used in combination when we unlock the buffer block and wake the process. The specific code is as follows:
After transacting the data, an interrupt service routine will run. And this will unlock the buffer block. The processes waiting for this buffer block will be awakened later. The specific code is as follows:
//code path:kernel/blk_drv/ll_rw_block.c:
static void make_request(int major,int rw, struct buffer_head * bh) {
……
lock_buffer(bh);
if ((rw = = WRITE && !bh->b_dirt) || (rw = = READ && bh->b_uptodate)) {
unlock_buffer(bh); //unlock the buffer block and wake up the process return;
}
if (rw = = READ) req = request+NR_REQUEST;
……
}
static inline void unlock_buffer(struct buffer_head * bh) //unlock the buffer block and //wake up the process {
if (!bh->b_lock)
printk(“ll_rw_block.c: buffer not locked\n\r”);
bh->b_lock = 0;
wake_up(&bh->b_wait);
}
//code path:kernel/blk_drv/blk.h:
extern inline void end_request(int uptodate) //Process the rehabilitation work after the //completion of the request operation DEVICE_OFF(CURRENT->dev);
if (CURRENT->bh) {
CURRENT->bh->b_uptodate = uptodate;
unlock_buffer(CURRENT->bh); //unlock the buffer block and wake up the //process
} if (!uptodate) {
printk(DEVICE_NAME “ I/O error\n\r”);
printk(“dev%04x, block%d\n\r”,CURRENT->dev, CURRENT->bh->b_blocknr);
}
……
}
static inline void unlock_buffer(struct buffer_head * bh) //unlock the buffer block and //wake up the process {
if (!bh->b_lock)
printk(DEVICE_NAME “: free buffer being unlocked\n”);
bh->b_lock = 0; //release the lock of buffer block
wake_up(&bh->b_wait); //wake up the process waiting for the buffer block }