356 lines
9.9 KiB
C
356 lines
9.9 KiB
C
/* This file contains the heart of the mechanism used to read (and write)
|
|
* files. Read and write requests are split up into chunks that do not cross
|
|
* block boundaries. Each chunk is then processed in turn. Reads on special
|
|
* files are also detected and handled.
|
|
*
|
|
* The entry points into this file are
|
|
* do_read: perform the READ system call by calling read_write
|
|
* do_getdents: read entries from a directory (GETDENTS)
|
|
* read_write: actually do the work of READ and WRITE
|
|
*
|
|
* Changes for VFS:
|
|
* Jul 2006 (Balazs Gerofi)
|
|
*/
|
|
|
|
#include "fs.h"
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <minix/com.h>
|
|
#include <minix/u64.h>
|
|
#include "file.h"
|
|
#include "fproc.h"
|
|
#include "param.h"
|
|
#include <dirent.h>
|
|
|
|
#include <minix/vfsif.h>
|
|
#include "vnode.h"
|
|
#include "vmnt.h"
|
|
|
|
|
|
/*===========================================================================*
|
|
* do_read *
|
|
*===========================================================================*/
|
|
PUBLIC int do_read()
|
|
{
|
|
return(read_write(READING));
|
|
}
|
|
|
|
|
|
/*===========================================================================*
|
|
* read_write *
|
|
*===========================================================================*/
|
|
PUBLIC int read_write(rw_flag)
|
|
int rw_flag; /* READING or WRITING */
|
|
{
|
|
/* Perform read(fd, buffer, nbytes) or write(fd, buffer, nbytes) call. */
|
|
register struct filp *f;
|
|
register struct vnode *vp;
|
|
off_t bytes_left, f_size;
|
|
u64_t position;
|
|
unsigned int off, cum_io;
|
|
int op, oflags, r, chunk, usr, seg, block_spec, char_spec;
|
|
int regular, partial_pipe = 0, partial_cnt = 0;
|
|
mode_t mode_word;
|
|
struct filp *wf;
|
|
phys_bytes p;
|
|
struct dmap *dp;
|
|
|
|
/* Request and response structures */
|
|
struct readwrite_req req;
|
|
struct readwrite_res res;
|
|
|
|
/* For block spec files */
|
|
struct breadwrite_req breq;
|
|
|
|
/* PM loads segments by putting funny things in other bits of the
|
|
* message, indicated by a high bit in fd. */
|
|
if (who_e == PM_PROC_NR && (m_in.fd & _PM_SEG_FLAG)) {
|
|
seg = (int) m_in.m1_p2;
|
|
usr = (int) m_in.m1_p3;
|
|
m_in.fd &= ~(_PM_SEG_FLAG); /* get rid of flag bit */
|
|
}
|
|
else {
|
|
usr = who_e; /* normal case */
|
|
seg = D;
|
|
}
|
|
|
|
/* If the file descriptor is valid, get the vnode, size and mode. */
|
|
if (m_in.nbytes < 0) return(EINVAL);
|
|
if ((f = get_filp(m_in.fd)) == NIL_FILP) return(err_code);
|
|
if (((f->filp_mode) & (rw_flag == READING ? R_BIT : W_BIT)) == 0) {
|
|
return(f->filp_mode == FILP_CLOSED ? EIO : EBADF);
|
|
}
|
|
|
|
if (m_in.nbytes == 0)
|
|
return(0); /* so char special files need not check for 0*/
|
|
|
|
/* check if user process has the memory it needs.
|
|
* if not, copying will fail later.
|
|
* do this after 0-check above because umap doesn't want to map 0 bytes. */
|
|
if ((r = sys_umap(usr, seg, (vir_bytes) m_in.buffer, m_in.nbytes, &p)) != OK)
|
|
{
|
|
printf("VFS: read_write: umap failed for process %d\n", usr);
|
|
return r;
|
|
}
|
|
|
|
position = f->filp_pos;
|
|
oflags = f->filp_flags;
|
|
|
|
vp = f->filp_vno;
|
|
f_size = vp->v_size;
|
|
|
|
r = OK;
|
|
if (vp->v_pipe == I_PIPE) {
|
|
/* fp->fp_cum_io_partial is only nonzero when doing partial writes */
|
|
cum_io = fp->fp_cum_io_partial;
|
|
}
|
|
else {
|
|
cum_io = 0;
|
|
}
|
|
|
|
op = (rw_flag == READING ? DEV_READ : DEV_WRITE);
|
|
mode_word = vp->v_mode & I_TYPE;
|
|
regular = mode_word == I_REGULAR || mode_word == I_NAMED_PIPE;
|
|
|
|
if ((char_spec = (mode_word == I_CHAR_SPECIAL ? 1 : 0))) {
|
|
if (vp->v_sdev == NO_DEV)
|
|
panic(__FILE__,"read_write tries to read from "
|
|
"character device NO_DEV", NO_NUM);
|
|
}
|
|
|
|
if ((block_spec = (mode_word == I_BLOCK_SPECIAL ? 1 : 0))) {
|
|
f_size = ULONG_MAX;
|
|
if (vp->v_sdev == NO_DEV)
|
|
panic(__FILE__,"read_write tries to read from "
|
|
" block device NO_DEV", NO_NUM);
|
|
}
|
|
|
|
/* Character special files. */
|
|
if (char_spec) {
|
|
dev_t dev;
|
|
/*dev = (dev_t) f->filp_ino->i_zone[0];*/
|
|
dev = (dev_t) vp->v_sdev;
|
|
r = dev_io(op, dev, usr, m_in.buffer, position, m_in.nbytes, oflags);
|
|
if (r >= 0) {
|
|
cum_io = r;
|
|
position = add64ul(position, r);
|
|
r = OK;
|
|
}
|
|
}
|
|
/* Block special files. */
|
|
else if (block_spec) {
|
|
/* Fill in the fields of the request */
|
|
breq.rw_flag = rw_flag;
|
|
breq.fs_e = vp->v_bfs_e;
|
|
breq.blocksize = vp->v_blocksize;
|
|
breq.dev = vp->v_sdev;
|
|
breq.user_e = usr;
|
|
breq.pos = position;
|
|
breq.num_of_bytes = m_in.nbytes;
|
|
breq.user_addr = m_in.buffer;
|
|
|
|
/* Issue request */
|
|
r = req_breadwrite(&breq, &res);
|
|
|
|
position = res.new_pos;
|
|
cum_io += res.cum_io;
|
|
}
|
|
/* Regular files */
|
|
else {
|
|
if (rw_flag == WRITING && block_spec == 0) {
|
|
/* Check for O_APPEND flag. */
|
|
if (oflags & O_APPEND) position = cvul64(f_size);
|
|
|
|
/* Check in advance to see if file will grow too big. */
|
|
if (cmp64ul(position, vp->v_vmnt->m_max_file_size - m_in.nbytes) > 0)
|
|
return(EFBIG);
|
|
|
|
}
|
|
|
|
/* Pipes are a little different. Check. */
|
|
if (vp->v_pipe == I_PIPE) {
|
|
r = pipe_check(vp, rw_flag, oflags,
|
|
m_in.nbytes, position, &partial_cnt, 0);
|
|
if (r <= 0) return(r);
|
|
}
|
|
|
|
if (partial_cnt > 0) {
|
|
/* So taht we don't need to deal with partial count
|
|
* in the FS process */
|
|
m_in.nbytes = MIN(m_in.nbytes, partial_cnt);
|
|
partial_pipe = 1;
|
|
}
|
|
|
|
/* Fill in request structure */
|
|
req.fs_e = vp->v_fs_e;
|
|
req.rw_flag = rw_flag;
|
|
req.inode_nr = vp->v_inode_nr;
|
|
req.user_e = usr;
|
|
req.seg = seg;
|
|
req.pos = position;
|
|
req.num_of_bytes = m_in.nbytes;
|
|
req.user_addr = m_in.buffer;
|
|
req.inode_index = vp->v_index;
|
|
|
|
/* Issue request */
|
|
r = req_readwrite(&req, &res);
|
|
|
|
if (r >= 0)
|
|
{
|
|
if (ex64hi(res.new_pos))
|
|
panic(__FILE__, "read_write: bad new pos", NO_NUM);
|
|
|
|
position = res.new_pos;
|
|
cum_io += res.cum_io;
|
|
}
|
|
}
|
|
|
|
/* On write, update file size and access time. */
|
|
if (rw_flag == WRITING) {
|
|
if (regular || mode_word == I_DIRECTORY) {
|
|
if (cmp64ul(position, f_size) > 0)
|
|
{
|
|
if (ex64hi(position) != 0)
|
|
{
|
|
panic(__FILE__,
|
|
"read_write: file size too big for v_size",
|
|
NO_NUM);
|
|
}
|
|
vp->v_size = ex64lo(position);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
if (vp->v_pipe == I_PIPE) {
|
|
if (cmp64ul(position, vp->v_size) >= 0) {
|
|
/* Reset pipe pointers */
|
|
vp->v_size = 0;
|
|
position = cvu64(0);
|
|
wf = find_filp(vp, W_BIT);
|
|
if (wf != NIL_FILP) wf->filp_pos = cvu64(0);
|
|
}
|
|
}
|
|
}
|
|
|
|
f->filp_pos = position;
|
|
|
|
if (r == OK) {
|
|
if (partial_pipe) {
|
|
partial_pipe = 0;
|
|
/* partial write on pipe with */
|
|
/* O_NONBLOCK, return write count */
|
|
if (!(oflags & O_NONBLOCK)) {
|
|
fp->fp_cum_io_partial = cum_io;
|
|
suspend(XPIPE); /* partial write on pipe with */
|
|
return(SUSPEND); /* nbyte > PIPE_SIZE - non-atomic */
|
|
}
|
|
}
|
|
fp->fp_cum_io_partial = 0;
|
|
return cum_io;
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
|
|
/* Original "uncached" code for block spec files */
|
|
#if 0
|
|
else if (block_spec) {
|
|
char buf[_MIN_BLOCK_SIZE];
|
|
block_t b;
|
|
int bleft = m_in.nbytes;
|
|
dev_t dev = vp->v_sdev;
|
|
|
|
b = position / _MIN_BLOCK_SIZE;
|
|
off = position % _MIN_BLOCK_SIZE;
|
|
|
|
while (bleft) {
|
|
/* First read the whole block */
|
|
r = dev_bio(DEV_READ, dev, FS_PROC_NR, buf, b * _MIN_BLOCK_SIZE,
|
|
_MIN_BLOCK_SIZE);
|
|
|
|
if (r != _MIN_BLOCK_SIZE)
|
|
break;
|
|
|
|
/* How many bytes to copy? */
|
|
chunk = MIN(bleft, _MIN_BLOCK_SIZE - off);
|
|
|
|
if (rw_flag == READING) {
|
|
/* Copy a chunk from the buffer to user space. */
|
|
r = sys_vircopy(FS_PROC_NR, D, (phys_bytes) (&buf[off]),
|
|
usr, seg, (phys_bytes) m_in.buffer,
|
|
(phys_bytes) chunk);
|
|
}
|
|
else {
|
|
/* Copy a chunk from user space to the buffer. */
|
|
r = sys_vircopy(usr, seg, (phys_bytes) m_in.buffer,
|
|
FS_PROC_NR, D, (phys_bytes) (&buf[off]),
|
|
(phys_bytes) chunk);
|
|
}
|
|
|
|
/* Write back if WRITE */
|
|
if (rw_flag == WRITING) {
|
|
r = dev_bio(DEV_WRITE, dev, FS_PROC_NR, buf,
|
|
b * _MIN_BLOCK_SIZE, _MIN_BLOCK_SIZE);
|
|
|
|
if (r != _MIN_BLOCK_SIZE)
|
|
break;
|
|
}
|
|
|
|
bleft -= chunk;
|
|
m_in.buffer += chunk;
|
|
|
|
/* 0 offset in the next block */
|
|
b++;
|
|
off = 0;
|
|
}
|
|
|
|
cum_io = m_in.nbytes - bleft;
|
|
position += cum_io;
|
|
r = OK;
|
|
}
|
|
#endif
|
|
|
|
|
|
/*===========================================================================*
|
|
* do_getdents *
|
|
*===========================================================================*/
|
|
PUBLIC int do_getdents()
|
|
{
|
|
/* Perform the getdents(fd, buf, size) system call. */
|
|
int r;
|
|
off_t pos_change;
|
|
cp_grant_id_t gid;
|
|
register struct filp *rfilp;
|
|
|
|
/* Is the file descriptor valid? */
|
|
if ( (rfilp = get_filp(m_in.fd)) == NIL_FILP) {
|
|
return(err_code);
|
|
}
|
|
|
|
if (!(rfilp->filp_mode & R_BIT))
|
|
return EBADF;
|
|
|
|
if ((rfilp->filp_vno->v_mode & I_TYPE) != I_DIRECTORY)
|
|
return EBADF;
|
|
|
|
gid=cpf_grant_magic(rfilp->filp_vno->v_fs_e, who_e, (vir_bytes) m_in.buffer,
|
|
m_in.nbytes, CPF_WRITE);
|
|
if (gid < 0) panic(__FILE__, "cpf_grant_magic failed", gid);
|
|
|
|
/* Issue request */
|
|
if (ex64hi(rfilp->filp_pos) != 0)
|
|
panic(__FILE__, "do_getdents: should handle large offsets", NO_NUM);
|
|
|
|
r= req_getdents(rfilp->filp_vno->v_fs_e, rfilp->filp_vno->v_inode_nr,
|
|
ex64lo(rfilp->filp_pos), gid, m_in.nbytes, &pos_change);
|
|
|
|
cpf_revoke(gid);
|
|
|
|
if (r > 0)
|
|
rfilp->filp_pos= add64ul(rfilp->filp_pos, pos_change);
|
|
return r;
|
|
}
|
|
|
|
|
|
|