linux系统调用之sync源码解析(基于linux0.11)

我们知道write函数写入的数据不是实时同步硬盘的,系统提供了一个函数让我们的数据可以实时地同步到硬盘,那就是sync。但这个实时也是相对的,毕竟同步数据也需要时间的,如果正在同步,就断电,那同步就会失败。

int sys_sync(void)
{
	int i;
	struct buffer_head * bh;
	// 把所有inode写入buffer,等待回写,见下面代码
	sync_inodes();		/* write out inodes into buffers */
	bh = start_buffer;
	for (i=0 ; i<NR_BUFFERS ; i++,bh++) {
		wait_on_buffer(bh);
		if (bh->b_dirt)
			// 请求底层写硬盘操作,等待底层驱动回写到硬盘,不一定立刻写入
			ll_rw_block(WRITE,bh);
	}
	return 0;
}

我们先看sync_inode。该函数把inode table里的,即进程打开的文件对应的inode节点,写入到buffer里。

// 遍历所有inode,从硬盘读包括该inode的数据块,然后用内存的inode覆盖硬盘读进来的,存在buffer里,等待回写	
void sync_inodes(void)
{
	int i;
	struct m_inode * inode;
		
	inode = 0+inode_table;
	for(i=0 ; i<NR_INODE ; i++,inode++) {
		wait_on_inode(inode);
		// 管道的内容存放在内存,所以不需要同步
		if (inode->i_dirt && !inode->i_pipe)
			write_inode(inode);
	}
}
// 先把inode从硬盘中读进来,然后覆盖,等待回写
static void write_inode(struct m_inode * inode)
{
	struct super_block * sb;
	struct buffer_head * bh;
	int block;

	lock_inode(inode);
	if (!inode->i_dirt || !inode->i_dev) {
		unlock_inode(inode);
		return;
	}
	if (!(sb=get_super(inode->i_dev)))
		panic("trying to write inode without device");
	// 算出inode的块号,2 + inode位图块数 + 块位图块数 + inode的相对偏移
	block = 2 + sb->s_imap_blocks + sb->s_zmap_blocks +
		(inode->i_num-1)/INODES_PER_BLOCK;
	// 读入包含该inode的整个数据块
	if (!(bh=bread(inode->i_dev,block)))
		panic("unable to read i-node block");
	// 找到数据块中inode所属的位置,写到高速缓存等待回写到硬盘
	((struct d_inode *)bh->b_data)
		[(inode->i_num-1)%INODES_PER_BLOCK] =
			*(struct d_inode *)inode;
	bh->b_dirt=1;
	inode->i_dirt=0;
	brelse(bh);
	unlock_inode(inode);
}

把inode写入到buffer里后,就遍历buffer,需要回写到硬盘的则通过ll_rw_block请求驱动进行回写操作。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值