123398Smckusick /* 237731Smckusick * Copyright (c) 1982, 1986, 1989 Regents of the University of California. 337731Smckusick * All rights reserved. 423398Smckusick * 544460Sbostic * %sccs.include.redist.c% 637731Smckusick * 7*48030Smckusick * @(#)vfs_vnops.c 7.29 (Berkeley) 04/16/91 823398Smckusick */ 918Sbill 1017099Sbloom #include "param.h" 1117099Sbloom #include "systm.h" 1237731Smckusick #include "kernel.h" 1317099Sbloom #include "file.h" 1437731Smckusick #include "stat.h" 1537731Smckusick #include "buf.h" 1637731Smckusick #include "proc.h" 1737731Smckusick #include "mount.h" 18*48030Smckusick #include "namei.h" 1937731Smckusick #include "vnode.h" 2037731Smckusick #include "ioctl.h" 2137731Smckusick #include "tty.h" 2218Sbill 2337731Smckusick struct fileops vnops = 2437731Smckusick { vn_read, vn_write, vn_ioctl, vn_select, vn_close }; 2537520Smckusick 2618Sbill /* 2737731Smckusick * Common code for vnode open operations. 2837731Smckusick * Check permissions, and call the VOP_OPEN or VOP_CREATE routine. 2918Sbill */ 3047540Skarels vn_open(ndp, p, fmode, cmode) 3137731Smckusick register struct nameidata *ndp; 3247540Skarels struct proc *p; 3337731Smckusick int fmode, cmode; 3437731Smckusick { 3537731Smckusick register struct vnode *vp; 3647540Skarels register struct ucred *cred = p->p_ucred; 3737731Smckusick struct vattr vat; 3837731Smckusick struct vattr *vap = &vat; 3937731Smckusick int error; 4037731Smckusick 4137731Smckusick if (fmode & FCREAT) { 4237731Smckusick ndp->ni_nameiop = CREATE | LOCKPARENT | LOCKLEAF; 4337731Smckusick if ((fmode & FEXCL) == 0) 4437731Smckusick ndp->ni_nameiop |= FOLLOW; 4547540Skarels if (error = namei(ndp, p)) 4637731Smckusick return (error); 4737731Smckusick if (ndp->ni_vp == NULL) { 4841362Smckusick VATTR_NULL(vap); 4937731Smckusick vap->va_type = VREG; 5037731Smckusick vap->va_mode = cmode; 51*48030Smckusick if (error = VOP_CREATE(ndp, vap, p)) 5237731Smckusick return (error); 5337731Smckusick fmode &= ~FTRUNC; 5437731Smckusick vp = ndp->ni_vp; 5537731Smckusick } else { 5639761Smckusick if (ndp->ni_dvp == ndp->ni_vp) 5739761Smckusick vrele(ndp->ni_dvp); 5843343Smckusick else 5939761Smckusick vput(ndp->ni_dvp); 6039761Smckusick ndp->ni_dvp = NULL; 6137731Smckusick vp = ndp->ni_vp; 6237731Smckusick if (fmode & FEXCL) { 6337731Smckusick error = EEXIST; 6437731Smckusick goto bad; 6537731Smckusick } 6637731Smckusick fmode &= ~FCREAT; 6737731Smckusick } 6837731Smckusick } else { 6937731Smckusick ndp->ni_nameiop = LOOKUP | FOLLOW | LOCKLEAF; 7047540Skarels if (error = namei(ndp, p)) 7137731Smckusick return (error); 7237731Smckusick vp = ndp->ni_vp; 7337731Smckusick } 7437731Smckusick if (vp->v_type == VSOCK) { 7537731Smckusick error = EOPNOTSUPP; 7637731Smckusick goto bad; 7737731Smckusick } 7837731Smckusick if ((fmode & FCREAT) == 0) { 7937731Smckusick if (fmode & FREAD) { 80*48030Smckusick if (error = VOP_ACCESS(vp, VREAD, cred, p)) 8137731Smckusick goto bad; 8237731Smckusick } 8337731Smckusick if (fmode & (FWRITE|FTRUNC)) { 8437731Smckusick if (vp->v_type == VDIR) { 8537731Smckusick error = EISDIR; 8637731Smckusick goto bad; 8737731Smckusick } 8838401Smckusick if ((error = vn_writechk(vp)) || 89*48030Smckusick (error = VOP_ACCESS(vp, VWRITE, cred, p))) 9038401Smckusick goto bad; 9137731Smckusick } 9237731Smckusick } 9337731Smckusick if (fmode & FTRUNC) { 9441362Smckusick VATTR_NULL(vap); 9537731Smckusick vap->va_size = 0; 96*48030Smckusick if (error = VOP_SETATTR(vp, vap, cred, p)) 9737731Smckusick goto bad; 9837731Smckusick } 9937731Smckusick VOP_UNLOCK(vp); 100*48030Smckusick error = VOP_OPEN(vp, fmode, cred, p); 10138351Smckusick if (error) 10238351Smckusick vrele(vp); 10338351Smckusick return (error); 10437731Smckusick 10537731Smckusick bad: 10637731Smckusick vput(vp); 10740707Skarels return (error); 10837731Smckusick } 10937731Smckusick 11037731Smckusick /* 11138401Smckusick * Check for write permissions on the specified vnode. 11238401Smckusick * The read-only status of the file system is checked. 11338401Smckusick * Also, prototype text segments cannot be written. 11437731Smckusick */ 11538401Smckusick vn_writechk(vp) 11637731Smckusick register struct vnode *vp; 11718Sbill { 11818Sbill 11938401Smckusick /* 12038401Smckusick * Disallow write attempts on read-only file systems; 12138401Smckusick * unless the file is a socket or a block or character 12238401Smckusick * device resident on the file system. 12338401Smckusick */ 12445060Smckusick if (vp->v_mount->mnt_flag & MNT_RDONLY) { 12545060Smckusick switch (vp->v_type) { 12645060Smckusick case VREG: case VDIR: case VLNK: 12745060Smckusick return (EROFS); 12845060Smckusick } 12945060Smckusick } 13038401Smckusick /* 13138401Smckusick * If there's shared text associated with 13238401Smckusick * the vnode, try to free it up once. If 13338401Smckusick * we fail, we can't allow writing. 13438401Smckusick */ 13545744Smckusick if ((vp->v_flag & VTEXT) && !vnode_pager_uncache(vp)) 13638401Smckusick return (ETXTBSY); 13738401Smckusick return (0); 13837731Smckusick } 13937731Smckusick 14037731Smckusick /* 14137731Smckusick * Vnode version of rdwri() for calls on file systems. 14237731Smckusick */ 143*48030Smckusick vn_rdwr(rw, vp, base, len, offset, segflg, ioflg, cred, aresid, p) 14437731Smckusick enum uio_rw rw; 14537731Smckusick struct vnode *vp; 14637731Smckusick caddr_t base; 14737731Smckusick int len; 14837731Smckusick off_t offset; 14937731Smckusick enum uio_seg segflg; 15037731Smckusick int ioflg; 15137731Smckusick struct ucred *cred; 15237731Smckusick int *aresid; 153*48030Smckusick struct proc *p; 15437731Smckusick { 15537731Smckusick struct uio auio; 15637731Smckusick struct iovec aiov; 15737731Smckusick int error; 15837731Smckusick 15939590Smckusick if ((ioflg & IO_NODELOCKED) == 0) 16039590Smckusick VOP_LOCK(vp); 16137731Smckusick auio.uio_iov = &aiov; 16237731Smckusick auio.uio_iovcnt = 1; 16337731Smckusick aiov.iov_base = base; 16437731Smckusick aiov.iov_len = len; 16537731Smckusick auio.uio_resid = len; 16637731Smckusick auio.uio_offset = offset; 16737731Smckusick auio.uio_segflg = segflg; 16837731Smckusick auio.uio_rw = rw; 169*48030Smckusick auio.uio_procp = p; 17037731Smckusick if (rw == UIO_READ) 17139590Smckusick error = VOP_READ(vp, &auio, ioflg, cred); 17237731Smckusick else 17339590Smckusick error = VOP_WRITE(vp, &auio, ioflg, cred); 17437731Smckusick if (aresid) 17537731Smckusick *aresid = auio.uio_resid; 17637731Smckusick else 17737731Smckusick if (auio.uio_resid && error == 0) 17837731Smckusick error = EIO; 17939590Smckusick if ((ioflg & IO_NODELOCKED) == 0) 18039590Smckusick VOP_UNLOCK(vp); 18137731Smckusick return (error); 18237731Smckusick } 18337731Smckusick 18437731Smckusick vn_read(fp, uio, cred) 18537731Smckusick struct file *fp; 18637731Smckusick struct uio *uio; 18737731Smckusick struct ucred *cred; 18837731Smckusick { 18939590Smckusick register struct vnode *vp = (struct vnode *)fp->f_data; 19039590Smckusick int count, error; 19137731Smckusick 19239590Smckusick VOP_LOCK(vp); 19339590Smckusick uio->uio_offset = fp->f_offset; 19439590Smckusick count = uio->uio_resid; 19539590Smckusick error = VOP_READ(vp, uio, (fp->f_flag & FNDELAY) ? IO_NDELAY : 0, cred); 19639590Smckusick fp->f_offset += count - uio->uio_resid; 19739590Smckusick VOP_UNLOCK(vp); 19839590Smckusick return (error); 19937731Smckusick } 20037731Smckusick 20137731Smckusick vn_write(fp, uio, cred) 20237731Smckusick struct file *fp; 20337731Smckusick struct uio *uio; 20437731Smckusick struct ucred *cred; 20537731Smckusick { 20637731Smckusick register struct vnode *vp = (struct vnode *)fp->f_data; 20739590Smckusick int count, error, ioflag = 0; 20837731Smckusick 20937731Smckusick if (vp->v_type == VREG && (fp->f_flag & FAPPEND)) 21037731Smckusick ioflag |= IO_APPEND; 21137731Smckusick if (fp->f_flag & FNDELAY) 21237731Smckusick ioflag |= IO_NDELAY; 21339590Smckusick VOP_LOCK(vp); 21439590Smckusick uio->uio_offset = fp->f_offset; 21539590Smckusick count = uio->uio_resid; 21639590Smckusick error = VOP_WRITE(vp, uio, ioflag, cred); 21739590Smckusick if (ioflag & IO_APPEND) 21839590Smckusick fp->f_offset = uio->uio_offset; 21939590Smckusick else 22039590Smckusick fp->f_offset += count - uio->uio_resid; 22139590Smckusick VOP_UNLOCK(vp); 22239590Smckusick return (error); 22337731Smckusick } 22437731Smckusick 22537731Smckusick /* 22637731Smckusick * Get stat info for a vnode. 22737731Smckusick */ 228*48030Smckusick vn_stat(vp, sb, p) 22937731Smckusick struct vnode *vp; 23037731Smckusick register struct stat *sb; 231*48030Smckusick struct proc *p; 23237731Smckusick { 23337731Smckusick struct vattr vattr; 23437731Smckusick register struct vattr *vap; 23537731Smckusick int error; 23637731Smckusick u_short mode; 23737731Smckusick 23837731Smckusick vap = &vattr; 239*48030Smckusick error = VOP_GETATTR(vp, vap, p->p_ucred, p); 24037731Smckusick if (error) 24137731Smckusick return (error); 24237731Smckusick /* 24337731Smckusick * Copy from vattr table 24437731Smckusick */ 24537731Smckusick sb->st_dev = vap->va_fsid; 24637731Smckusick sb->st_ino = vap->va_fileid; 24737731Smckusick mode = vap->va_mode; 24837731Smckusick switch (vp->v_type) { 24937731Smckusick case VREG: 25039382Smckusick mode |= S_IFREG; 25137731Smckusick break; 25237731Smckusick case VDIR: 25339382Smckusick mode |= S_IFDIR; 25437731Smckusick break; 25537731Smckusick case VBLK: 25639382Smckusick mode |= S_IFBLK; 25737731Smckusick break; 25837731Smckusick case VCHR: 25939382Smckusick mode |= S_IFCHR; 26037731Smckusick break; 26137731Smckusick case VLNK: 26239382Smckusick mode |= S_IFLNK; 26337731Smckusick break; 26437731Smckusick case VSOCK: 26539382Smckusick mode |= S_IFSOCK; 26637731Smckusick break; 26740284Smckusick case VFIFO: 26840284Smckusick mode |= S_IFIFO; 26940284Smckusick break; 27037731Smckusick default: 27137731Smckusick return (EBADF); 27237731Smckusick }; 27337731Smckusick sb->st_mode = mode; 27437731Smckusick sb->st_nlink = vap->va_nlink; 27537731Smckusick sb->st_uid = vap->va_uid; 27637731Smckusick sb->st_gid = vap->va_gid; 27737731Smckusick sb->st_rdev = vap->va_rdev; 27837731Smckusick sb->st_size = vap->va_size; 27937731Smckusick sb->st_atime = vap->va_atime.tv_sec; 28037731Smckusick sb->st_spare1 = 0; 28137731Smckusick sb->st_mtime = vap->va_mtime.tv_sec; 28237731Smckusick sb->st_spare2 = 0; 28337731Smckusick sb->st_ctime = vap->va_ctime.tv_sec; 28437731Smckusick sb->st_spare3 = 0; 28537731Smckusick sb->st_blksize = vap->va_blocksize; 28638257Smckusick sb->st_flags = vap->va_flags; 28738257Smckusick sb->st_gen = vap->va_gen; 28838769Smckusick sb->st_blocks = vap->va_bytes / S_BLKSIZE; 28937731Smckusick return (0); 29037731Smckusick } 29137731Smckusick 29237731Smckusick /* 29337731Smckusick * Vnode ioctl call 29437731Smckusick */ 295*48030Smckusick vn_ioctl(fp, com, data, p) 29637731Smckusick struct file *fp; 29737731Smckusick int com; 29837731Smckusick caddr_t data; 299*48030Smckusick struct proc *p; 30037731Smckusick { 30137731Smckusick register struct vnode *vp = ((struct vnode *)fp->f_data); 30237731Smckusick struct vattr vattr; 30337731Smckusick int error; 30437731Smckusick 30537731Smckusick switch (vp->v_type) { 30637731Smckusick 30737731Smckusick case VREG: 30837731Smckusick case VDIR: 30937731Smckusick if (com == FIONREAD) { 310*48030Smckusick if (error = VOP_GETATTR(vp, &vattr, p->p_ucred, p)) 31137731Smckusick return (error); 31237731Smckusick *(off_t *)data = vattr.va_size - fp->f_offset; 31337731Smckusick return (0); 31418Sbill } 31537731Smckusick if (com == FIONBIO || com == FIOASYNC) /* XXX */ 31637731Smckusick return (0); /* XXX */ 31737731Smckusick /* fall into ... */ 31837731Smckusick 31937731Smckusick default: 32037731Smckusick return (ENOTTY); 32137731Smckusick 32240284Smckusick case VFIFO: 32337731Smckusick case VCHR: 32437731Smckusick case VBLK: 325*48030Smckusick error = VOP_IOCTL(vp, com, data, fp->f_flag, p->p_ucred, p); 32639564Smarc if (error == 0 && com == TIOCSCTTY) { 32747540Skarels p->p_session->s_ttyvp = vp; 32839564Smarc VREF(vp); 32939564Smarc } 33039564Smarc return (error); 33118Sbill } 33237731Smckusick } 33337731Smckusick 33437731Smckusick /* 33537731Smckusick * Vnode select call 33637731Smckusick */ 337*48030Smckusick vn_select(fp, which, p) 33837731Smckusick struct file *fp; 33937731Smckusick int which; 340*48030Smckusick struct proc *p; 34137731Smckusick { 34247540Skarels 34340707Skarels return (VOP_SELECT(((struct vnode *)fp->f_data), which, fp->f_flag, 344*48030Smckusick p->p_ucred, p)); 34537731Smckusick } 34637731Smckusick 34737731Smckusick /* 34837731Smckusick * Vnode close call 34937731Smckusick */ 350*48030Smckusick vn_close(fp, p) 35137731Smckusick register struct file *fp; 352*48030Smckusick struct proc *p; 35337731Smckusick { 35437731Smckusick struct vnode *vp = ((struct vnode *)fp->f_data); 35537731Smckusick int error; 35637731Smckusick 35711162Ssam /* 35837731Smckusick * Must delete vnode reference from this file entry 35937731Smckusick * before VOP_CLOSE, so that only other references 36037731Smckusick * will prevent close. 36111162Ssam */ 36237731Smckusick fp->f_data = (caddr_t) 0; 363*48030Smckusick error = VOP_CLOSE(vp, fp->f_flag, fp->f_cred, p); 36437731Smckusick vrele(vp); 36537731Smckusick return (error); 36637731Smckusick } 36737731Smckusick 36837731Smckusick /* 36937731Smckusick * vn_fhtovp() - convert a fh to a vnode ptr (optionally locked) 37037731Smckusick * - look up fsid in mount list (if not found ret error) 37137731Smckusick * - get vp by calling VFS_FHTOVP() macro 37237731Smckusick * - if lockflag lock it with VOP_LOCK() 37318Sbill */ 37437731Smckusick vn_fhtovp(fhp, lockflag, vpp) 37537731Smckusick fhandle_t *fhp; 37637731Smckusick int lockflag; 37737731Smckusick struct vnode **vpp; 37818Sbill { 37937731Smckusick register struct mount *mp; 38018Sbill 38137731Smckusick if ((mp = getvfs(&fhp->fh_fsid)) == NULL) 38237731Smckusick return (ESTALE); 38338146Smckusick if (VFS_FHTOVP(mp, &fhp->fh_fid, vpp)) 38438146Smckusick return (ESTALE); 38538146Smckusick if (!lockflag) 38638146Smckusick VOP_UNLOCK(*vpp); 38737731Smckusick return (0); 38837731Smckusick } 389