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*50110Smckusick * @(#)vfs_vnops.c 7.32 (Berkeley) 06/21/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" 1848030Smckusick #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 4149943Smckusick if (fmode & O_CREAT) { 4237731Smckusick ndp->ni_nameiop = CREATE | LOCKPARENT | LOCKLEAF; 4349943Smckusick if ((fmode & O_EXCL) == 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; 5148030Smckusick if (error = VOP_CREATE(ndp, vap, p)) 5237731Smckusick return (error); 5349943Smckusick fmode &= ~O_TRUNC; 5437731Smckusick vp = ndp->ni_vp; 5537731Smckusick } else { 5649732Smckusick VOP_ABORTOP(ndp); 5739761Smckusick if (ndp->ni_dvp == ndp->ni_vp) 5839761Smckusick vrele(ndp->ni_dvp); 5943343Smckusick else 6039761Smckusick vput(ndp->ni_dvp); 6139761Smckusick ndp->ni_dvp = NULL; 6237731Smckusick vp = ndp->ni_vp; 6349943Smckusick if (fmode & O_EXCL) { 6437731Smckusick error = EEXIST; 6537731Smckusick goto bad; 6637731Smckusick } 6749943Smckusick fmode &= ~O_CREAT; 6837731Smckusick } 6937731Smckusick } else { 7037731Smckusick ndp->ni_nameiop = LOOKUP | FOLLOW | LOCKLEAF; 7147540Skarels if (error = namei(ndp, p)) 7237731Smckusick return (error); 7337731Smckusick vp = ndp->ni_vp; 7437731Smckusick } 7537731Smckusick if (vp->v_type == VSOCK) { 7637731Smckusick error = EOPNOTSUPP; 7737731Smckusick goto bad; 7837731Smckusick } 7949943Smckusick if ((fmode & O_CREAT) == 0) { 8037731Smckusick if (fmode & FREAD) { 8148030Smckusick if (error = VOP_ACCESS(vp, VREAD, cred, p)) 8237731Smckusick goto bad; 8337731Smckusick } 8449943Smckusick if (fmode & (FWRITE | O_TRUNC)) { 8537731Smckusick if (vp->v_type == VDIR) { 8637731Smckusick error = EISDIR; 8737731Smckusick goto bad; 8837731Smckusick } 8938401Smckusick if ((error = vn_writechk(vp)) || 9048030Smckusick (error = VOP_ACCESS(vp, VWRITE, cred, p))) 9138401Smckusick goto bad; 9237731Smckusick } 9337731Smckusick } 9449943Smckusick if (fmode & O_TRUNC) { 9541362Smckusick VATTR_NULL(vap); 9637731Smckusick vap->va_size = 0; 9748030Smckusick if (error = VOP_SETATTR(vp, vap, cred, p)) 9837731Smckusick goto bad; 9937731Smckusick } 100*50110Smckusick if (error = VOP_OPEN(vp, fmode, cred, p)) 101*50110Smckusick goto bad; 102*50110Smckusick if (fmode & FWRITE) 103*50110Smckusick vp->v_writecount++; 104*50110Smckusick return (0); 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 */ 14348030Smckusick 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; 15348030Smckusick 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; 16948030Smckusick 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; 19549943Smckusick error = VOP_READ(vp, uio, (fp->f_flag & FNONBLOCK) ? IO_NDELAY : 0, 19649943Smckusick cred); 19739590Smckusick fp->f_offset += count - uio->uio_resid; 19839590Smckusick VOP_UNLOCK(vp); 19939590Smckusick return (error); 20037731Smckusick } 20137731Smckusick 20237731Smckusick vn_write(fp, uio, cred) 20337731Smckusick struct file *fp; 20437731Smckusick struct uio *uio; 20537731Smckusick struct ucred *cred; 20637731Smckusick { 20737731Smckusick register struct vnode *vp = (struct vnode *)fp->f_data; 20839590Smckusick int count, error, ioflag = 0; 20937731Smckusick 21049943Smckusick if (vp->v_type == VREG && (fp->f_flag & O_APPEND)) 21137731Smckusick ioflag |= IO_APPEND; 21249943Smckusick if (fp->f_flag & FNONBLOCK) 21337731Smckusick ioflag |= IO_NDELAY; 21439590Smckusick VOP_LOCK(vp); 21539590Smckusick uio->uio_offset = fp->f_offset; 21639590Smckusick count = uio->uio_resid; 21739590Smckusick error = VOP_WRITE(vp, uio, ioflag, cred); 21839590Smckusick if (ioflag & IO_APPEND) 21939590Smckusick fp->f_offset = uio->uio_offset; 22039590Smckusick else 22139590Smckusick fp->f_offset += count - uio->uio_resid; 22239590Smckusick VOP_UNLOCK(vp); 22339590Smckusick return (error); 22437731Smckusick } 22537731Smckusick 22637731Smckusick /* 22737731Smckusick * Get stat info for a vnode. 22837731Smckusick */ 22948030Smckusick vn_stat(vp, sb, p) 23037731Smckusick struct vnode *vp; 23137731Smckusick register struct stat *sb; 23248030Smckusick struct proc *p; 23337731Smckusick { 23437731Smckusick struct vattr vattr; 23537731Smckusick register struct vattr *vap; 23637731Smckusick int error; 23737731Smckusick u_short mode; 23837731Smckusick 23937731Smckusick vap = &vattr; 24048030Smckusick error = VOP_GETATTR(vp, vap, p->p_ucred, p); 24137731Smckusick if (error) 24237731Smckusick return (error); 24337731Smckusick /* 24437731Smckusick * Copy from vattr table 24537731Smckusick */ 24637731Smckusick sb->st_dev = vap->va_fsid; 24737731Smckusick sb->st_ino = vap->va_fileid; 24837731Smckusick mode = vap->va_mode; 24937731Smckusick switch (vp->v_type) { 25037731Smckusick case VREG: 25139382Smckusick mode |= S_IFREG; 25237731Smckusick break; 25337731Smckusick case VDIR: 25439382Smckusick mode |= S_IFDIR; 25537731Smckusick break; 25637731Smckusick case VBLK: 25739382Smckusick mode |= S_IFBLK; 25837731Smckusick break; 25937731Smckusick case VCHR: 26039382Smckusick mode |= S_IFCHR; 26137731Smckusick break; 26237731Smckusick case VLNK: 26339382Smckusick mode |= S_IFLNK; 26437731Smckusick break; 26537731Smckusick case VSOCK: 26639382Smckusick mode |= S_IFSOCK; 26737731Smckusick break; 26840284Smckusick case VFIFO: 26940284Smckusick mode |= S_IFIFO; 27040284Smckusick break; 27137731Smckusick default: 27237731Smckusick return (EBADF); 27337731Smckusick }; 27437731Smckusick sb->st_mode = mode; 27537731Smckusick sb->st_nlink = vap->va_nlink; 27637731Smckusick sb->st_uid = vap->va_uid; 27737731Smckusick sb->st_gid = vap->va_gid; 27837731Smckusick sb->st_rdev = vap->va_rdev; 27937731Smckusick sb->st_size = vap->va_size; 28037731Smckusick sb->st_atime = vap->va_atime.tv_sec; 28137731Smckusick sb->st_spare1 = 0; 28237731Smckusick sb->st_mtime = vap->va_mtime.tv_sec; 28337731Smckusick sb->st_spare2 = 0; 28437731Smckusick sb->st_ctime = vap->va_ctime.tv_sec; 28537731Smckusick sb->st_spare3 = 0; 28637731Smckusick sb->st_blksize = vap->va_blocksize; 28738257Smckusick sb->st_flags = vap->va_flags; 28838257Smckusick sb->st_gen = vap->va_gen; 28938769Smckusick sb->st_blocks = vap->va_bytes / S_BLKSIZE; 29037731Smckusick return (0); 29137731Smckusick } 29237731Smckusick 29337731Smckusick /* 29437731Smckusick * Vnode ioctl call 29537731Smckusick */ 29648030Smckusick vn_ioctl(fp, com, data, p) 29737731Smckusick struct file *fp; 29837731Smckusick int com; 29937731Smckusick caddr_t data; 30048030Smckusick struct proc *p; 30137731Smckusick { 30237731Smckusick register struct vnode *vp = ((struct vnode *)fp->f_data); 30337731Smckusick struct vattr vattr; 30437731Smckusick int error; 30537731Smckusick 30637731Smckusick switch (vp->v_type) { 30737731Smckusick 30837731Smckusick case VREG: 30937731Smckusick case VDIR: 31037731Smckusick if (com == FIONREAD) { 31148030Smckusick if (error = VOP_GETATTR(vp, &vattr, p->p_ucred, p)) 31237731Smckusick return (error); 31337731Smckusick *(off_t *)data = vattr.va_size - fp->f_offset; 31437731Smckusick return (0); 31518Sbill } 31637731Smckusick if (com == FIONBIO || com == FIOASYNC) /* XXX */ 31737731Smckusick return (0); /* XXX */ 31837731Smckusick /* fall into ... */ 31937731Smckusick 32037731Smckusick default: 32137731Smckusick return (ENOTTY); 32237731Smckusick 32340284Smckusick case VFIFO: 32437731Smckusick case VCHR: 32537731Smckusick case VBLK: 32648030Smckusick error = VOP_IOCTL(vp, com, data, fp->f_flag, p->p_ucred, p); 32739564Smarc if (error == 0 && com == TIOCSCTTY) { 32847540Skarels p->p_session->s_ttyvp = vp; 32939564Smarc VREF(vp); 33039564Smarc } 33139564Smarc return (error); 33218Sbill } 33337731Smckusick } 33437731Smckusick 33537731Smckusick /* 33637731Smckusick * Vnode select call 33737731Smckusick */ 33848030Smckusick vn_select(fp, which, p) 33937731Smckusick struct file *fp; 34037731Smckusick int which; 34148030Smckusick struct proc *p; 34237731Smckusick { 34347540Skarels 34440707Skarels return (VOP_SELECT(((struct vnode *)fp->f_data), which, fp->f_flag, 34548030Smckusick p->p_ucred, p)); 34637731Smckusick } 34737731Smckusick 34837731Smckusick /* 34937731Smckusick * Vnode close call 35037731Smckusick */ 351*50110Smckusick vn_close(vp, flags, cred, p) 352*50110Smckusick register struct vnode *vp; 353*50110Smckusick int flags; 354*50110Smckusick struct ucred *cred; 35548030Smckusick struct proc *p; 35637731Smckusick { 35737731Smckusick int error; 35837731Smckusick 359*50110Smckusick if (flags & FWRITE) 360*50110Smckusick vp->v_writecount--; 361*50110Smckusick error = VOP_CLOSE(vp, flags, cred, p); 36237731Smckusick vrele(vp); 36337731Smckusick return (error); 36437731Smckusick } 36537731Smckusick 36637731Smckusick /* 36737731Smckusick * vn_fhtovp() - convert a fh to a vnode ptr (optionally locked) 36837731Smckusick * - look up fsid in mount list (if not found ret error) 36937731Smckusick * - get vp by calling VFS_FHTOVP() macro 37037731Smckusick * - if lockflag lock it with VOP_LOCK() 37118Sbill */ 37237731Smckusick vn_fhtovp(fhp, lockflag, vpp) 37337731Smckusick fhandle_t *fhp; 37437731Smckusick int lockflag; 37537731Smckusick struct vnode **vpp; 37618Sbill { 37737731Smckusick register struct mount *mp; 37818Sbill 37937731Smckusick if ((mp = getvfs(&fhp->fh_fsid)) == NULL) 38037731Smckusick return (ESTALE); 38138146Smckusick if (VFS_FHTOVP(mp, &fhp->fh_fid, vpp)) 38238146Smckusick return (ESTALE); 38338146Smckusick if (!lockflag) 38438146Smckusick VOP_UNLOCK(*vpp); 38537731Smckusick return (0); 38637731Smckusick } 387