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*49732Smckusick * @(#)vfs_vnops.c 7.30 (Berkeley) 05/15/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 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; 5148030Smckusick if (error = VOP_CREATE(ndp, vap, p)) 5237731Smckusick return (error); 5337731Smckusick fmode &= ~FTRUNC; 5437731Smckusick vp = ndp->ni_vp; 5537731Smckusick } else { 56*49732Smckusick 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; 6337731Smckusick if (fmode & FEXCL) { 6437731Smckusick error = EEXIST; 6537731Smckusick goto bad; 6637731Smckusick } 6737731Smckusick fmode &= ~FCREAT; 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 } 7937731Smckusick if ((fmode & FCREAT) == 0) { 8037731Smckusick if (fmode & FREAD) { 8148030Smckusick if (error = VOP_ACCESS(vp, VREAD, cred, p)) 8237731Smckusick goto bad; 8337731Smckusick } 8437731Smckusick if (fmode & (FWRITE|FTRUNC)) { 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 } 9437731Smckusick if (fmode & FTRUNC) { 9541362Smckusick VATTR_NULL(vap); 9637731Smckusick vap->va_size = 0; 9748030Smckusick if (error = VOP_SETATTR(vp, vap, cred, p)) 9837731Smckusick goto bad; 9937731Smckusick } 10037731Smckusick VOP_UNLOCK(vp); 10148030Smckusick error = VOP_OPEN(vp, fmode, cred, p); 10238351Smckusick if (error) 10338351Smckusick vrele(vp); 10438351Smckusick return (error); 10537731Smckusick 10637731Smckusick bad: 10737731Smckusick vput(vp); 10840707Skarels return (error); 10937731Smckusick } 11037731Smckusick 11137731Smckusick /* 11238401Smckusick * Check for write permissions on the specified vnode. 11338401Smckusick * The read-only status of the file system is checked. 11438401Smckusick * Also, prototype text segments cannot be written. 11537731Smckusick */ 11638401Smckusick vn_writechk(vp) 11737731Smckusick register struct vnode *vp; 11818Sbill { 11918Sbill 12038401Smckusick /* 12138401Smckusick * Disallow write attempts on read-only file systems; 12238401Smckusick * unless the file is a socket or a block or character 12338401Smckusick * device resident on the file system. 12438401Smckusick */ 12545060Smckusick if (vp->v_mount->mnt_flag & MNT_RDONLY) { 12645060Smckusick switch (vp->v_type) { 12745060Smckusick case VREG: case VDIR: case VLNK: 12845060Smckusick return (EROFS); 12945060Smckusick } 13045060Smckusick } 13138401Smckusick /* 13238401Smckusick * If there's shared text associated with 13338401Smckusick * the vnode, try to free it up once. If 13438401Smckusick * we fail, we can't allow writing. 13538401Smckusick */ 13645744Smckusick if ((vp->v_flag & VTEXT) && !vnode_pager_uncache(vp)) 13738401Smckusick return (ETXTBSY); 13838401Smckusick return (0); 13937731Smckusick } 14037731Smckusick 14137731Smckusick /* 14237731Smckusick * Vnode version of rdwri() for calls on file systems. 14337731Smckusick */ 14448030Smckusick vn_rdwr(rw, vp, base, len, offset, segflg, ioflg, cred, aresid, p) 14537731Smckusick enum uio_rw rw; 14637731Smckusick struct vnode *vp; 14737731Smckusick caddr_t base; 14837731Smckusick int len; 14937731Smckusick off_t offset; 15037731Smckusick enum uio_seg segflg; 15137731Smckusick int ioflg; 15237731Smckusick struct ucred *cred; 15337731Smckusick int *aresid; 15448030Smckusick struct proc *p; 15537731Smckusick { 15637731Smckusick struct uio auio; 15737731Smckusick struct iovec aiov; 15837731Smckusick int error; 15937731Smckusick 16039590Smckusick if ((ioflg & IO_NODELOCKED) == 0) 16139590Smckusick VOP_LOCK(vp); 16237731Smckusick auio.uio_iov = &aiov; 16337731Smckusick auio.uio_iovcnt = 1; 16437731Smckusick aiov.iov_base = base; 16537731Smckusick aiov.iov_len = len; 16637731Smckusick auio.uio_resid = len; 16737731Smckusick auio.uio_offset = offset; 16837731Smckusick auio.uio_segflg = segflg; 16937731Smckusick auio.uio_rw = rw; 17048030Smckusick auio.uio_procp = p; 17137731Smckusick if (rw == UIO_READ) 17239590Smckusick error = VOP_READ(vp, &auio, ioflg, cred); 17337731Smckusick else 17439590Smckusick error = VOP_WRITE(vp, &auio, ioflg, cred); 17537731Smckusick if (aresid) 17637731Smckusick *aresid = auio.uio_resid; 17737731Smckusick else 17837731Smckusick if (auio.uio_resid && error == 0) 17937731Smckusick error = EIO; 18039590Smckusick if ((ioflg & IO_NODELOCKED) == 0) 18139590Smckusick VOP_UNLOCK(vp); 18237731Smckusick return (error); 18337731Smckusick } 18437731Smckusick 18537731Smckusick vn_read(fp, uio, cred) 18637731Smckusick struct file *fp; 18737731Smckusick struct uio *uio; 18837731Smckusick struct ucred *cred; 18937731Smckusick { 19039590Smckusick register struct vnode *vp = (struct vnode *)fp->f_data; 19139590Smckusick int count, error; 19237731Smckusick 19339590Smckusick VOP_LOCK(vp); 19439590Smckusick uio->uio_offset = fp->f_offset; 19539590Smckusick count = uio->uio_resid; 19639590Smckusick error = VOP_READ(vp, uio, (fp->f_flag & FNDELAY) ? IO_NDELAY : 0, 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 21037731Smckusick if (vp->v_type == VREG && (fp->f_flag & FAPPEND)) 21137731Smckusick ioflag |= IO_APPEND; 21237731Smckusick if (fp->f_flag & FNDELAY) 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 */ 35148030Smckusick vn_close(fp, p) 35237731Smckusick register struct file *fp; 35348030Smckusick struct proc *p; 35437731Smckusick { 35537731Smckusick struct vnode *vp = ((struct vnode *)fp->f_data); 35637731Smckusick int error; 35737731Smckusick 35811162Ssam /* 35937731Smckusick * Must delete vnode reference from this file entry 36037731Smckusick * before VOP_CLOSE, so that only other references 36137731Smckusick * will prevent close. 36211162Ssam */ 36337731Smckusick fp->f_data = (caddr_t) 0; 36448030Smckusick error = VOP_CLOSE(vp, fp->f_flag, fp->f_cred, p); 36537731Smckusick vrele(vp); 36637731Smckusick return (error); 36737731Smckusick } 36837731Smckusick 36937731Smckusick /* 37037731Smckusick * vn_fhtovp() - convert a fh to a vnode ptr (optionally locked) 37137731Smckusick * - look up fsid in mount list (if not found ret error) 37237731Smckusick * - get vp by calling VFS_FHTOVP() macro 37337731Smckusick * - if lockflag lock it with VOP_LOCK() 37418Sbill */ 37537731Smckusick vn_fhtovp(fhp, lockflag, vpp) 37637731Smckusick fhandle_t *fhp; 37737731Smckusick int lockflag; 37837731Smckusick struct vnode **vpp; 37918Sbill { 38037731Smckusick register struct mount *mp; 38118Sbill 38237731Smckusick if ((mp = getvfs(&fhp->fh_fsid)) == NULL) 38337731Smckusick return (ESTALE); 38438146Smckusick if (VFS_FHTOVP(mp, &fhp->fh_fid, vpp)) 38538146Smckusick return (ESTALE); 38638146Smckusick if (!lockflag) 38738146Smckusick VOP_UNLOCK(*vpp); 38837731Smckusick return (0); 38937731Smckusick } 390