/* * linux/fs/file_dev.c * * (C) 1991 Linus Torvalds */ #include #include #include #include #include #include #define MIN(a,b) (((a)<(b))?(a):(b)) #define MAX(a,b) (((a)>(b))?(a):(b)) int minix_file_read(struct inode * inode, struct file * filp, char * buf, int count) { int read,left,chars,nr; struct buffer_head * bh; if (filp->f_pos > inode->i_size) left = 0; else left = inode->i_size - filp->f_pos; if (left > count) left = count; read = 0; while (left > 0) { if (nr = bmap(inode,(filp->f_pos)>>BLOCK_SIZE_BITS)) { if (!(bh=bread(inode->i_dev,nr))) return read?read:-EIO; } else bh = NULL; nr = filp->f_pos & (BLOCK_SIZE-1); chars = MIN( BLOCK_SIZE-nr , left ); filp->f_pos += chars; left -= chars; read += chars; if (bh) { char * p = nr + bh->b_data; while (chars-->0) put_fs_byte(*(p++),buf++); brelse(bh); } else { while (chars-->0) put_fs_byte(0,buf++); } } inode->i_atime = CURRENT_TIME; return read; } int minix_file_write(struct inode * inode, struct file * filp, char * buf, int count) { off_t pos; int written,block,c; struct buffer_head * bh; char * p; /* * 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 = inode->i_size; else pos = filp->f_pos; written = 0; while (writteni_dev,block))) { if (!written) written = -EIO; break; } c = pos % BLOCK_SIZE; p = c + bh->b_data; bh->b_dirt = 1; c = BLOCK_SIZE-c; if (c > count-written) c = count-written; pos += c; if (pos > inode->i_size) { inode->i_size = pos; inode->i_dirt = 1; } written += 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; inode->i_dirt = 1; } return written; }