Linux内核源码分析--文件系统(九、File_dev.c)

        file_dev.c程序是用于访问文件数据的读写程序;


file_read()

      int file_read(struct m_inode * inode, struct file * filp, char * buf, int count)函数,从i节点为inode的文件中 filp中pos位置处开始读取count个字节数据存放到用户空buf内
        功能:从i节点为inode的文件中 filp中pos处开始读取count个字节数据存放到用户空间buf处
        参数:inode   文件i节点;filp  文件结构;buf  用户空间;count 字符数;
        实现:
        1、根据inode和filp中的pos可以得到文件所在的逻辑块,以及映射到缓存区中
        2、根据file->pos可以得到该逻辑块中的偏移量,然后和count比较
        3、读取指定大小字符,然后移动pos循环到下一个逻辑块读取数据

//普通文件的读取数据函数和块设备文件读取函数类似
//inode 文件i节点可以得到dev;filp  文件结构可以得到pos;buf 用户空间;count 读取的字节数;
int file_read(struct m_inode * inode, struct file * filp, char * buf, int count)
{
	int left,chars,nr;
	struct buffer_head * bh;

	if ((left=count)<=0)//读取的字数小于0,则返回
		return 0;
	while (left) {//读取的字数
		if (nr = bmap(inode,(filp->f_pos)/BLOCK_SIZE)) {//可以得到逻辑块号(其实是查找到i_zone[i])
			if (!(bh=bread(inode->i_dev,nr)))//读取到映射缓存区
				break;
		} else
			bh = NULL;
		nr = filp->f_pos % BLOCK_SIZE;//某个逻辑块中的偏移量
		chars = MIN( BLOCK_SIZE-nr , left );//如果剩下的空间下,则先填满一个数据块
		filp->f_pos += chars;//往后读取数据
		left -= chars;//获取剩下需要读取的数据
		if (bh) {
			char * p = nr + bh->b_data;//得到开始读取数据的位置
			while (chars-->0)//拷贝到用户空间
				put_fs_byte(*(p++),buf++);
			brelse(bh);
		} else {//如缓存块数据没有,则读取0填充
			while (chars-->0)
				put_fs_byte(0,buf++);
		}
	}
	inode->i_atime = CURRENT_TIME;
	return (count-left)?(count-left):-ERROR;
}

file_write()

      int file_read(struct m_inode * inode, struct file * filp, char * buf, int count)函数,从i节点为inode的文件中 filp中pos位置处开始读取count个字节数据存放到用户空buf内
        功能:从i节点为inode的文件中 filp中pos处开始读取count个字节数据存放到用户空间buf处
        参数:inode   文件i节点;filp  文件结构;buf  用户空间;count 字符数;
        实现:
        1、首先判断是否写入方式是追加还是从指定光标位置开始写入(主要是确定pos值)
        2、根据file->pos可以得到该逻辑块中的偏移量,然后和count比较
        3、读取指定大小字符,然后移动pos循环到下一个逻辑块读取数据

int file_write(struct m_inode * inode, struct file * filp, char * buf, int count)
{
	off_t pos;
	int block,c;
	struct buffer_head * bh;
	char * p;
	int i=0;

/*
 * ok, append may not work when many processes are writing at the same time
 * but so what. That way leads to madness anyway.
 */
	if (filp->f_flags & O_APPEND)//如果以追加的方式写入,则pos要到数据最后
		pos = inode->i_size;
	else
		pos = filp->f_pos;//否则为文件结构中指定的长度
	while (i<count) {
		if (!(block = create_block(inode,pos/BLOCK_SIZE)))//获取开始写入的数据块,如果没有,则创建新的
			break;
		if (!(bh=bread(inode->i_dev,block)))//获取到映射缓存块
			break;
		c = pos % BLOCK_SIZE;//在逻辑块中的偏移量
		p = c + bh->b_data;//在缓存中开始写入数据的地方
		bh->b_dirt = 1;
		c = BLOCK_SIZE-c;//该逻辑块中还能够写入多少字符串
		if (c > count-i) c = count-i;//如果剩下数据能全部写完,,,
		pos += c;//文件指针往后推
		if (pos > inode->i_size) {//如果写入的数据比原来存在的多,则修改属性
			inode->i_size = pos;
			inode->i_dirt = 1;
		}
		i += c;//增加写入的字数
		while (c-->0)//开始写入
			*(p++) = get_fs_byte(buf++);
		brelse(bh);
	}
	inode->i_mtime = CURRENT_TIME;
	//更新下文件指针和访问时间
	if (!(filp->f_flags & O_APPEND)) {
		filp->f_pos = pos;
		inode->i_ctime = CURRENT_TIME;
	}
	return (i?i:-1);
}

        转载请注明作者和原文出处,原文地址:http://blog.csdn.net/yuzhihui_no1/article/details/44650785

        若有不正确之处,望大家指正,共同学习!谢谢!!!


郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。