/* This file contains the code for performing four system calls relating to * status and directories. * * The entry points into this file are * do_chdir: perform the CHDIR system call * do_chroot: perform the CHROOT system call * do_stat: perform the STAT system call * do_fstat: perform the FSTAT system call * do_fstatfs: perform the FSTATFS system call * do_lstat: perform the LSTAT system call * do_rdlink: perform the RDLNK system call */ #include "fs.h" #include #include #include #include #include "buf.h" #include "file.h" #include "fproc.h" #include "inode.h" #include "param.h" #include "super.h" FORWARD _PROTOTYPE( int change, (struct inode **iip, char *name_ptr, int len)); FORWARD _PROTOTYPE( int change_into, (struct inode **iip, struct inode *ip)); FORWARD _PROTOTYPE( int stat_inode, (struct inode *rip, struct filp *fil_ptr, char *user_addr) ); /*===========================================================================* * do_fchdir * *===========================================================================*/ PUBLIC int do_fchdir() { /* Change directory on already-opened fd. */ struct filp *rfilp; /* Is the file descriptor valid? */ if ( (rfilp = get_filp(m_in.fd)) == NIL_FILP) return(err_code); dup_inode(rfilp->filp_ino); return change_into(&fp->fp_workdir, rfilp->filp_ino); } /*===========================================================================* * do_chdir * *===========================================================================*/ PUBLIC int do_chdir() { /* Change directory. This function is also called by MM to simulate a chdir * in order to do EXEC, etc. It also changes the root directory, the uids and * gids, and the umask. */ int r; register struct fproc *rfp; if (who_e == PM_PROC_NR) { int slot; if(isokendpt(m_in.endpt1, &slot) != OK) return EINVAL; rfp = &fproc[slot]; put_inode(fp->fp_rootdir); dup_inode(fp->fp_rootdir = rfp->fp_rootdir); put_inode(fp->fp_workdir); dup_inode(fp->fp_workdir = rfp->fp_workdir); /* MM uses access() to check permissions. To make this work, pretend * that the user's real ids are the same as the user's effective ids. * FS calls other than access() do not use the real ids, so are not * affected. */ fp->fp_realuid = fp->fp_effuid = rfp->fp_effuid; fp->fp_realgid = fp->fp_effgid = rfp->fp_effgid; fp->fp_umask = rfp->fp_umask; return(OK); } /* Perform the chdir(name) system call. */ r = change(&fp->fp_workdir, m_in.name, m_in.name_length); return(r); } /*===========================================================================* * do_chroot * *===========================================================================*/ PUBLIC int do_chroot() { /* Perform the chroot(name) system call. */ register int r; if (!super_user) return(EPERM); /* only su may chroot() */ r = change(&fp->fp_rootdir, m_in.name, m_in.name_length); return(r); } /*===========================================================================* * change * *===========================================================================*/ PRIVATE int change(iip, name_ptr, len) struct inode **iip; /* pointer to the inode pointer for the dir */ char *name_ptr; /* pointer to the directory name to change to */ int len; /* length of the directory name string */ { /* Do the actual work for chdir() and chroot(). */ struct inode *rip; /* Try to open the new directory. */ if (fetch_name(name_ptr, len, M3) != OK) return(err_code); if ( (rip = eat_path(user_path)) == NIL_INODE) return(err_code); return change_into(iip, rip); } /*===========================================================================* * change_into * *===========================================================================*/ PRIVATE int change_into(iip, rip) struct inode **iip; /* pointer to the inode pointer for the dir */ struct inode *rip; /* this is what the inode has to become */ { register int r; /* It must be a directory and also be searchable. */ if ( (rip->i_mode & I_TYPE) != I_DIRECTORY) r = ENOTDIR; else r = forbidden(rip, X_BIT); /* check if dir is searchable */ /* If error, return inode. */ if (r != OK) { put_inode(rip); return(r); } /* Everything is OK. Make the change. */ put_inode(*iip); /* release the old directory */ *iip = rip; /* acquire the new one */ return(OK); } /*===========================================================================* * do_stat * *===========================================================================*/ PUBLIC int do_stat() { /* Perform the stat(name, buf) system call. */ register struct inode *rip; register int r; /* Both stat() and fstat() use the same routine to do the real work. That * routine expects an inode, so acquire it temporarily. */ if (fetch_name(m_in.name1, m_in.name1_length, M1) != OK) return(err_code); if ( (rip = eat_path(user_path)) == NIL_INODE) return(err_code); r = stat_inode(rip, NIL_FILP, m_in.name2); /* actually do the work.*/ put_inode(rip); /* release the inode */ return(r); } /*===========================================================================* * do_fstat * *===========================================================================*/ PUBLIC int do_fstat() { /* Perform the fstat(fd, buf) system call. */ register struct filp *rfilp; /* Is the file descriptor valid? */ if ( (rfilp = get_filp(m_in.fd)) == NIL_FILP) return(err_code); return(stat_inode(rfilp->filp_ino, rfilp, m_in.buffer)); } /*===========================================================================* * stat_inode * *===========================================================================*/ PRIVATE int stat_inode(rip, fil_ptr, user_addr) register struct inode *rip; /* pointer to inode to stat */ struct filp *fil_ptr; /* filp pointer, supplied by 'fstat' */ char *user_addr; /* user space address where stat buf goes */ { /* Common code for stat and fstat system calls. */ struct stat statbuf; mode_t mo; int r, s; /* Update the atime, ctime, and mtime fields in the inode, if need be. */ if (rip->i_update) update_times(rip); /* Fill in the statbuf struct. */ mo = rip->i_mode & I_TYPE; /* true iff special */ s = (mo == I_CHAR_SPECIAL || mo == I_BLOCK_SPECIAL); statbuf.st_dev = rip->i_dev; statbuf.st_ino = rip->i_num; statbuf.st_mode = rip->i_mode; statbuf.st_nlink = rip->i_nlinks; statbuf.st_uid = rip->i_uid; statbuf.st_gid = rip->i_gid; statbuf.st_rdev = (dev_t) (s ? rip->i_zone[0] : NO_DEV); statbuf.st_size = rip->i_size; if (rip->i_pipe == I_PIPE) { statbuf.st_mode &= ~I_REGULAR; /* wipe out I_REGULAR bit for pipes */ if (fil_ptr != NIL_FILP && fil_ptr->filp_mode & R_BIT) statbuf.st_size -= fil_ptr->filp_pos; } statbuf.st_atime = rip->i_atime; statbuf.st_mtime = rip->i_mtime; statbuf.st_ctime = rip->i_ctime; /* Copy the struct to user space. */ r = sys_datacopy(FS_PROC_NR, (vir_bytes) &statbuf, who_e, (vir_bytes) user_addr, (phys_bytes) sizeof(statbuf)); return(r); } /*===========================================================================* * do_fstatfs * *===========================================================================*/ PUBLIC int do_fstatfs() { /* Perform the fstatfs(fd, buf) system call. */ struct statfs st; register struct filp *rfilp; int r; /* Is the file descriptor valid? */ if ( (rfilp = get_filp(m_in.fd)) == NIL_FILP) return(err_code); st.f_bsize = rfilp->filp_ino->i_sp->s_block_size; r = sys_datacopy(FS_PROC_NR, (vir_bytes) &st, who_e, (vir_bytes) m_in.buffer, (phys_bytes) sizeof(st)); return(r); } /*===========================================================================* * do_lstat * *===========================================================================*/ PUBLIC int do_lstat() { /* Perform the lstat(name, buf) system call. */ register int r; /* return value */ register struct inode *rip; /* target inode */ if (fetch_name(m_in.name1, m_in.name1_length, M1) != OK) return(err_code); if ((rip = parse_path(user_path, (char *) 0, EAT_PATH_OPAQUE)) == NIL_INODE) return(err_code); r = stat_inode(rip, NIL_FILP, m_in.name2); put_inode(rip); return(r); } /*===========================================================================* * do_rdlink * *===========================================================================*/ PUBLIC int do_rdlink() { /* Perform the readlink(name, buf) system call. */ register int r; /* return value */ block_t b; /* block containing link text */ struct buf *bp; /* buffer containing link text */ register struct inode *rip; /* target inode */ int copylen; copylen = m_in.m1_i2; if(copylen < 0) return EINVAL; if (fetch_name(m_in.name1, m_in.name1_length, M1) != OK) return(err_code); if ((rip = parse_path(user_path, (char *) 0, EAT_PATH_OPAQUE)) == NIL_INODE) return(err_code); r = EACCES; if (S_ISLNK(rip->i_mode) && (b = read_map(rip, (off_t) 0)) != NO_BLOCK) { if (m_in.name2_length <= 0) r = EINVAL; else if (m_in.name2_length < rip->i_size) r = ERANGE; else { if(rip->i_size < copylen) copylen = rip->i_size; bp = get_block(rip->i_dev, b, NORMAL); r = sys_vircopy(SELF, D, (vir_bytes) bp->b_data, who_e, D, (vir_bytes) m_in.name2, (vir_bytes) copylen); if (r == OK) r = copylen; put_block(bp, DIRECTORY_BLOCK); } } put_inode(rip); return(r); }