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

来源:互联网 发布:刷永久qq会员软件 编辑:程序博客网 时间:2024/06/03 18:39

        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;} elsebh = 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;elsepos = 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

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


0 0