[BACK]Return to nfs_vnops.c CVS log [TXT][DIR] Up to [local] / sys / nfs

Annotation of sys/nfs/nfs_vnops.c, Revision 1.1.1.1

1.1       nbrk        1: /*     $OpenBSD: nfs_vnops.c,v 1.75 2007/06/01 23:47:57 deraadt Exp $  */
                      2: /*     $NetBSD: nfs_vnops.c,v 1.62.4.1 1996/07/08 20:26:52 jtc Exp $   */
                      3:
                      4: /*
                      5:  * Copyright (c) 1989, 1993
                      6:  *     The Regents of the University of California.  All rights reserved.
                      7:  *
                      8:  * This code is derived from software contributed to Berkeley by
                      9:  * Rick Macklem at The University of Guelph.
                     10:  *
                     11:  * Redistribution and use in source and binary forms, with or without
                     12:  * modification, are permitted provided that the following conditions
                     13:  * are met:
                     14:  * 1. Redistributions of source code must retain the above copyright
                     15:  *    notice, this list of conditions and the following disclaimer.
                     16:  * 2. Redistributions in binary form must reproduce the above copyright
                     17:  *    notice, this list of conditions and the following disclaimer in the
                     18:  *    documentation and/or other materials provided with the distribution.
                     19:  * 3. Neither the name of the University nor the names of its contributors
                     20:  *    may be used to endorse or promote products derived from this software
                     21:  *    without specific prior written permission.
                     22:  *
                     23:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
                     24:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
                     25:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
                     26:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
                     27:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
                     28:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
                     29:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
                     30:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
                     31:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
                     32:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
                     33:  * SUCH DAMAGE.
                     34:  *
                     35:  *     @(#)nfs_vnops.c 8.16 (Berkeley) 5/27/95
                     36:  */
                     37:
                     38:
                     39: /*
                     40:  * vnode op calls for Sun NFS version 2 and 3
                     41:  */
                     42:
                     43: #include <sys/param.h>
                     44: #include <sys/proc.h>
                     45: #include <sys/kernel.h>
                     46: #include <sys/systm.h>
                     47: #include <sys/resourcevar.h>
                     48: #include <sys/poll.h>
                     49: #include <sys/proc.h>
                     50: #include <sys/mount.h>
                     51: #include <sys/buf.h>
                     52: #include <sys/malloc.h>
                     53: #include <sys/pool.h>
                     54: #include <sys/mbuf.h>
                     55: #include <sys/conf.h>
                     56: #include <sys/namei.h>
                     57: #include <sys/vnode.h>
                     58: #include <sys/dirent.h>
                     59: #include <sys/fcntl.h>
                     60: #include <sys/lockf.h>
                     61: #include <sys/hash.h>
                     62:
                     63: #include <uvm/uvm_extern.h>
                     64:
                     65: #include <miscfs/specfs/specdev.h>
                     66: #include <miscfs/fifofs/fifo.h>
                     67:
                     68: #include <nfs/rpcv2.h>
                     69: #include <nfs/nfsproto.h>
                     70: #include <nfs/nfs.h>
                     71: #include <nfs/nfsnode.h>
                     72: #include <nfs/nfsmount.h>
                     73: #include <nfs/xdr_subs.h>
                     74: #include <nfs/nfsm_subs.h>
                     75: #include <nfs/nfs_var.h>
                     76:
                     77: #include <net/if.h>
                     78: #include <netinet/in.h>
                     79: #include <netinet/in_var.h>
                     80:
                     81: /* Defs */
                     82: #define        TRUE    1
                     83: #define        FALSE   0
                     84:
                     85: /*
                     86:  * Global vfs data structures for nfs
                     87:  */
                     88: int (**nfsv2_vnodeop_p)(void *);
                     89: struct vnodeopv_entry_desc nfsv2_vnodeop_entries[] = {
                     90:        { &vop_default_desc, vn_default_error },
                     91:        { &vop_lookup_desc, nfs_lookup },       /* lookup */
                     92:        { &vop_create_desc, nfs_create },       /* create */
                     93:        { &vop_mknod_desc, nfs_mknod },         /* mknod */
                     94:        { &vop_open_desc, nfs_open },           /* open */
                     95:        { &vop_close_desc, nfs_close },         /* close */
                     96:        { &vop_access_desc, nfs_access },       /* access */
                     97:        { &vop_getattr_desc, nfs_getattr },     /* getattr */
                     98:        { &vop_setattr_desc, nfs_setattr },     /* setattr */
                     99:        { &vop_read_desc, nfs_read },           /* read */
                    100:        { &vop_write_desc, nfs_write },         /* write */
                    101:        { &vop_ioctl_desc, nfs_ioctl },         /* ioctl */
                    102:        { &vop_poll_desc, nfs_poll },           /* poll */
                    103:        { &vop_kqfilter_desc, nfs_kqfilter },   /* kqfilter */
                    104:        { &vop_revoke_desc, nfs_revoke },       /* revoke */
                    105:        { &vop_fsync_desc, nfs_fsync },         /* fsync */
                    106:        { &vop_remove_desc, nfs_remove },       /* remove */
                    107:        { &vop_link_desc, nfs_link },           /* link */
                    108:        { &vop_rename_desc, nfs_rename },       /* rename */
                    109:        { &vop_mkdir_desc, nfs_mkdir },         /* mkdir */
                    110:        { &vop_rmdir_desc, nfs_rmdir },         /* rmdir */
                    111:        { &vop_symlink_desc, nfs_symlink },     /* symlink */
                    112:        { &vop_readdir_desc, nfs_readdir },     /* readdir */
                    113:        { &vop_readlink_desc, nfs_readlink },   /* readlink */
                    114:        { &vop_abortop_desc, vop_generic_abortop },     /* abortop */
                    115:        { &vop_inactive_desc, nfs_inactive },   /* inactive */
                    116:        { &vop_reclaim_desc, nfs_reclaim },     /* reclaim */
                    117:        { &vop_lock_desc, nfs_lock },           /* lock */
                    118:        { &vop_unlock_desc, nfs_unlock },       /* unlock */
                    119:        { &vop_bmap_desc, nfs_bmap },           /* bmap */
                    120:        { &vop_strategy_desc, nfs_strategy },   /* strategy */
                    121:        { &vop_print_desc, nfs_print },         /* print */
                    122:        { &vop_islocked_desc, nfs_islocked },   /* islocked */
                    123:        { &vop_pathconf_desc, nfs_pathconf },   /* pathconf */
                    124:        { &vop_advlock_desc, nfs_advlock },     /* advlock */
                    125:        { &vop_bwrite_desc, nfs_bwrite },
                    126:        { NULL, NULL }
                    127: };
                    128: struct vnodeopv_desc nfsv2_vnodeop_opv_desc =
                    129:        { &nfsv2_vnodeop_p, nfsv2_vnodeop_entries };
                    130:
                    131: /*
                    132:  * Special device vnode ops
                    133:  */
                    134: int (**spec_nfsv2nodeop_p)(void *);
                    135: struct vnodeopv_entry_desc spec_nfsv2nodeop_entries[] = {
                    136:        { &vop_default_desc, spec_vnoperate },
                    137:        { &vop_close_desc, nfsspec_close },     /* close */
                    138:        { &vop_access_desc, nfsspec_access },   /* access */
                    139:        { &vop_getattr_desc, nfs_getattr },     /* getattr */
                    140:        { &vop_setattr_desc, nfs_setattr },     /* setattr */
                    141:        { &vop_read_desc, nfsspec_read },       /* read */
                    142:        { &vop_write_desc, nfsspec_write },     /* write */
                    143:        { &vop_fsync_desc, nfs_fsync },         /* fsync */
                    144:        { &vop_inactive_desc, nfs_inactive },   /* inactive */
                    145:        { &vop_reclaim_desc, nfs_reclaim },     /* reclaim */
                    146:        { &vop_lock_desc, nfs_lock },           /* lock */
                    147:        { &vop_unlock_desc, nfs_unlock },       /* unlock */
                    148:        { &vop_print_desc, nfs_print },         /* print */
                    149:        { &vop_islocked_desc, nfs_islocked },   /* islocked */
                    150:        { NULL, NULL }
                    151: };
                    152: struct vnodeopv_desc spec_nfsv2nodeop_opv_desc =
                    153:        { &spec_nfsv2nodeop_p, spec_nfsv2nodeop_entries };
                    154:
                    155: #ifdef FIFO
                    156: int (**fifo_nfsv2nodeop_p)(void *);
                    157: struct vnodeopv_entry_desc fifo_nfsv2nodeop_entries[] = {
                    158:        { &vop_default_desc, fifo_vnoperate },
                    159:        { &vop_close_desc, nfsfifo_close },     /* close */
                    160:        { &vop_access_desc, nfsspec_access },   /* access */
                    161:        { &vop_getattr_desc, nfs_getattr },     /* getattr */
                    162:        { &vop_setattr_desc, nfs_setattr },     /* setattr */
                    163:        { &vop_read_desc, nfsfifo_read },       /* read */
                    164:        { &vop_write_desc, nfsfifo_write },     /* write */
                    165:        { &vop_fsync_desc, nfs_fsync },         /* fsync */
                    166:        { &vop_inactive_desc, nfs_inactive },   /* inactive */
                    167:        { &vop_reclaim_desc, nfsfifo_reclaim }, /* reclaim */
                    168:        { &vop_lock_desc, nfs_lock },           /* lock */
                    169:        { &vop_unlock_desc, nfs_unlock },       /* unlock */
                    170:        { &vop_print_desc, nfs_print },         /* print */
                    171:        { &vop_islocked_desc, nfs_islocked },   /* islocked */
                    172:        { &vop_bwrite_desc, vop_generic_bwrite },
                    173:        { NULL, NULL }
                    174: };
                    175: struct vnodeopv_desc fifo_nfsv2nodeop_opv_desc =
                    176:        { &fifo_nfsv2nodeop_p, fifo_nfsv2nodeop_entries };
                    177: #endif /* FIFO */
                    178:
                    179: /*
                    180:  * Global variables
                    181:  */
                    182: extern u_int32_t nfs_true, nfs_false;
                    183: extern u_int32_t nfs_xdrneg1;
                    184: extern struct nfsstats nfsstats;
                    185: extern nfstype nfsv3_type[9];
                    186: struct proc *nfs_iodwant[NFS_MAXASYNCDAEMON];
                    187: int nfs_numasync = 0;
                    188:
                    189: /*
                    190:  * nfs null call from vfs.
                    191:  */
                    192: int
                    193: nfs_null(vp, cred, procp)
                    194:        struct vnode *vp;
                    195:        struct ucred *cred;
                    196:        struct proc *procp;
                    197: {
                    198:        caddr_t bpos, dpos;
                    199:        int error = 0;
                    200:        struct mbuf *mreq, *mrep, *md, *mb;
                    201:
                    202:        nfsm_reqhead(vp, NFSPROC_NULL, 0);
                    203:        nfsm_request(vp, NFSPROC_NULL, procp, cred);
                    204:        nfsm_reqdone;
                    205:        return (error);
                    206: }
                    207:
                    208: /*
                    209:  * nfs access vnode op.
                    210:  * For nfs version 2, just return ok. File accesses may fail later.
                    211:  * For nfs version 3, use the access rpc to check accessibility. If file modes
                    212:  * are changed on the server, accesses might still fail later.
                    213:  */
                    214: int
                    215: nfs_access(v)
                    216:        void *v;
                    217: {
                    218:        struct vop_access_args *ap = v;
                    219:        struct vnode *vp = ap->a_vp;
                    220:        u_int32_t *tl;
                    221:        caddr_t cp;
                    222:        int32_t t1, t2;
                    223:        caddr_t bpos, dpos, cp2;
                    224:        int error = 0, attrflag;
                    225:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                    226:        u_int32_t mode, rmode;
                    227:        int v3 = NFS_ISV3(vp);
                    228:
                    229:        /*
                    230:         * Disallow write attempts on filesystems mounted read-only;
                    231:         * unless the file is a socket, fifo, or a block or character
                    232:         * device resident on the filesystem.
                    233:         */
                    234:        if ((ap->a_mode & VWRITE) && (vp->v_mount->mnt_flag & MNT_RDONLY)) {
                    235:                switch (vp->v_type) {
                    236:                case VREG:
                    237:                case VDIR:
                    238:                case VLNK:
                    239:                        return (EROFS);
                    240:                default:
                    241:                        break;
                    242:                }
                    243:        }
                    244:        /*
                    245:         * For nfs v3, do an access rpc, otherwise you are stuck emulating
                    246:         * ufs_access() locally using the vattr. This may not be correct,
                    247:         * since the server may apply other access criteria such as
                    248:         * client uid-->server uid mapping that we do not know about, but
                    249:         * this is better than just returning anything that is lying about
                    250:         * in the cache.
                    251:         */
                    252:        if (v3) {
                    253:                nfsstats.rpccnt[NFSPROC_ACCESS]++;
                    254:                nfsm_reqhead(vp, NFSPROC_ACCESS, NFSX_FH(v3) + NFSX_UNSIGNED);
                    255:                nfsm_fhtom(vp, v3);
                    256:                nfsm_build(tl, u_int32_t *, NFSX_UNSIGNED);
                    257:                if (ap->a_mode & VREAD)
                    258:                        mode = NFSV3ACCESS_READ;
                    259:                else
                    260:                        mode = 0;
                    261:                if (vp->v_type == VDIR) {
                    262:                        if (ap->a_mode & VWRITE)
                    263:                                mode |= (NFSV3ACCESS_MODIFY | NFSV3ACCESS_EXTEND |
                    264:                                         NFSV3ACCESS_DELETE);
                    265:                        if (ap->a_mode & VEXEC)
                    266:                                mode |= NFSV3ACCESS_LOOKUP;
                    267:                } else {
                    268:                        if (ap->a_mode & VWRITE)
                    269:                                mode |= (NFSV3ACCESS_MODIFY | NFSV3ACCESS_EXTEND);
                    270:                        if (ap->a_mode & VEXEC)
                    271:                                mode |= NFSV3ACCESS_EXECUTE;
                    272:                }
                    273:                *tl = txdr_unsigned(mode);
                    274:                nfsm_request(vp, NFSPROC_ACCESS, ap->a_p, ap->a_cred);
                    275:                nfsm_postop_attr(vp, attrflag);
                    276:                if (!error) {
                    277:                        nfsm_dissect(tl, u_int32_t *, NFSX_UNSIGNED);
                    278:                        rmode = fxdr_unsigned(u_int32_t, *tl);
                    279:                        /*
                    280:                         * The NFS V3 spec does not clarify whether or not
                    281:                         * the returned access bits can be a superset of
                    282:                         * the ones requested, so...
                    283:                         */
                    284:                        if ((rmode & mode) != mode)
                    285:                                error = EACCES;
                    286:                }
                    287:                nfsm_reqdone;
                    288:                return (error);
                    289:        } else
                    290:                return (nfsspec_access(ap));
                    291: }
                    292:
                    293: /*
                    294:  * nfs open vnode op
                    295:  * Check to see if the type is ok
                    296:  * and that deletion is not in progress.
                    297:  * For paged in text files, you will need to flush the page cache
                    298:  * if consistency is lost.
                    299:  */
                    300: /* ARGSUSED */
                    301: int
                    302: nfs_open(v)
                    303:        void *v;
                    304: {
                    305:        struct vop_open_args *ap = v;
                    306:        struct vnode *vp = ap->a_vp;
                    307:        struct nfsnode *np = VTONFS(vp);
                    308:        struct vattr vattr;
                    309:        int error;
                    310:
                    311:        if (vp->v_type != VREG && vp->v_type != VDIR && vp->v_type != VLNK) {
                    312: #ifdef DIAGNOSTIC
                    313:                printf("open eacces vtyp=%d\n",vp->v_type);
                    314: #endif
                    315:                return (EACCES);
                    316:        }
                    317:
                    318:        /*
                    319:         * Initialize read and write creds here, for swapfiles
                    320:         * and other paths that don't set the creds themselves.
                    321:         */
                    322:
                    323:        if (ap->a_mode & FREAD) {
                    324:                if (np->n_rcred) {
                    325:                        crfree(np->n_rcred);
                    326:                }
                    327:                np->n_rcred = ap->a_cred;
                    328:                crhold(np->n_rcred);
                    329:        }
                    330:        if (ap->a_mode & FWRITE) {
                    331:                if (np->n_wcred) {
                    332:                        crfree(np->n_wcred);
                    333:                }
                    334:                np->n_wcred = ap->a_cred;
                    335:                crhold(np->n_wcred);
                    336:        }
                    337:
                    338:        if (np->n_flag & NMODIFIED) {
                    339:                if ((error = nfs_vinvalbuf(vp, V_SAVE, ap->a_cred,
                    340:                         ap->a_p, 1)) == EINTR)
                    341:                        return (error);
                    342:                uvm_vnp_uncache(vp);
                    343:                np->n_attrstamp = 0;
                    344:                if (vp->v_type == VDIR)
                    345:                        np->n_direofoffset = 0;
                    346:                error = VOP_GETATTR(vp, &vattr, ap->a_cred, ap->a_p);
                    347:                if (error)
                    348:                        return (error);
                    349:                np->n_mtime = vattr.va_mtime.tv_sec;
                    350:        } else {
                    351:                error = VOP_GETATTR(vp, &vattr, ap->a_cred, ap->a_p);
                    352:                if (error)
                    353:                        return (error);
                    354:                if (np->n_mtime != vattr.va_mtime.tv_sec) {
                    355:                        if (vp->v_type == VDIR)
                    356:                                np->n_direofoffset = 0;
                    357:                        if ((error = nfs_vinvalbuf(vp, V_SAVE,
                    358:                                 ap->a_cred, ap->a_p, 1)) == EINTR)
                    359:                                return (error);
                    360:                        uvm_vnp_uncache(vp);
                    361:                        np->n_mtime = vattr.va_mtime.tv_sec;
                    362:                }
                    363:        }
                    364:        np->n_attrstamp = 0; /* For Open/Close consistency */
                    365:        return (0);
                    366: }
                    367:
                    368: /*
                    369:  * nfs close vnode op
                    370:  * What an NFS client should do upon close after writing is a debatable issue.
                    371:  * Most NFS clients push delayed writes to the server upon close, basically for
                    372:  * two reasons:
                    373:  * 1 - So that any write errors may be reported back to the client process
                    374:  *     doing the close system call. By far the two most likely errors are
                    375:  *     NFSERR_NOSPC and NFSERR_DQUOT to indicate space allocation failure.
                    376:  * 2 - To put a worst case upper bound on cache inconsistency between
                    377:  *     multiple clients for the file.
                    378:  * There is also a consistency problem for Version 2 of the protocol w.r.t.
                    379:  * not being able to tell if other clients are writing a file concurrently,
                    380:  * since there is no way of knowing if the changed modify time in the reply
                    381:  * is only due to the write for this client.
                    382:  * (NFS Version 3 provides weak cache consistency data in the reply that
                    383:  *  should be sufficient to detect and handle this case.)
                    384:  *
                    385:  * The current code does the following:
                    386:  * for NFS Version 2 - play it safe and flush/invalidate all dirty buffers
                    387:  * for NFS Version 3 - flush dirty buffers to the server but don't invalidate
                    388:  *                     or commit them (this satisfies 1 and 2 except for the
                    389:  *                     case where the server crashes after this close but
                    390:  *                     before the commit RPC, which is felt to be "good
                    391:  *                     enough". Changing the last argument to nfs_flush() to
                    392:  *                     a 1 would force a commit operation, if it is felt a
                    393:  *                     commit is necessary now.
                    394:  */
                    395: /* ARGSUSED */
                    396: int
                    397: nfs_close(v)
                    398:        void *v;
                    399: {
                    400:        struct vop_close_args *ap = v;
                    401:        struct vnode *vp = ap->a_vp;
                    402:        struct nfsnode *np = VTONFS(vp);
                    403:        int error = 0;
                    404:
                    405:        if (vp->v_type == VREG) {
                    406:            if (np->n_flag & NMODIFIED) {
                    407:                if (NFS_ISV3(vp)) {
                    408:                    error = nfs_flush(vp, ap->a_cred, MNT_WAIT, ap->a_p, 0);
                    409:                    np->n_flag &= ~NMODIFIED;
                    410:                } else
                    411:                    error = nfs_vinvalbuf(vp, V_SAVE, ap->a_cred, ap->a_p, 1);
                    412:                np->n_attrstamp = 0;
                    413:            }
                    414:            if (np->n_flag & NWRITEERR) {
                    415:                np->n_flag &= ~NWRITEERR;
                    416:                error = np->n_error;
                    417:            }
                    418:        }
                    419:        return (error);
                    420: }
                    421:
                    422: /*
                    423:  * nfs getattr call from vfs.
                    424:  */
                    425: int
                    426: nfs_getattr(v)
                    427:        void *v;
                    428: {
                    429:        struct vop_getattr_args *ap = v;
                    430:        struct vnode *vp = ap->a_vp;
                    431:        struct nfsnode *np = VTONFS(vp);
                    432:        caddr_t cp;
                    433:        u_int32_t *tl;
                    434:        int32_t t1, t2;
                    435:        caddr_t bpos, dpos;
                    436:        int error = 0;
                    437:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                    438:        int v3 = NFS_ISV3(vp);
                    439:
                    440:        /*
                    441:         * Update local times for special files.
                    442:         */
                    443:        if (np->n_flag & (NACC | NUPD))
                    444:                np->n_flag |= NCHG;
                    445:        /*
                    446:         * First look in the cache.
                    447:         */
                    448:        if (nfs_getattrcache(vp, ap->a_vap) == 0)
                    449:                return (0);
                    450:        nfsstats.rpccnt[NFSPROC_GETATTR]++;
                    451:        nfsm_reqhead(vp, NFSPROC_GETATTR, NFSX_FH(v3));
                    452:        nfsm_fhtom(vp, v3);
                    453:        nfsm_request(vp, NFSPROC_GETATTR, ap->a_p, ap->a_cred);
                    454:        if (!error)
                    455:                nfsm_loadattr(vp, ap->a_vap);
                    456:        nfsm_reqdone;
                    457:        return (error);
                    458: }
                    459:
                    460: /*
                    461:  * nfs setattr call.
                    462:  */
                    463: int
                    464: nfs_setattr(v)
                    465:        void *v;
                    466: {
                    467:        struct vop_setattr_args *ap = v;
                    468:        struct vnode *vp = ap->a_vp;
                    469:        struct nfsnode *np = VTONFS(vp);
                    470:        struct vattr *vap = ap->a_vap;
                    471:        int error = 0;
                    472:        u_quad_t tsize = 0;
                    473:
                    474:        /*
                    475:         * Setting of flags is not supported.
                    476:         */
                    477:        if (vap->va_flags != VNOVAL)
                    478:                return (EOPNOTSUPP);
                    479:
                    480:        /*
                    481:         * Disallow write attempts if the filesystem is mounted read-only.
                    482:         */
                    483:        if ((vap->va_uid != (uid_t)VNOVAL ||
                    484:            vap->va_gid != (gid_t)VNOVAL || vap->va_atime.tv_sec != VNOVAL ||
                    485:            vap->va_mtime.tv_sec != VNOVAL || vap->va_mode != (mode_t)VNOVAL) &&
                    486:            (vp->v_mount->mnt_flag & MNT_RDONLY))
                    487:                return (EROFS);
                    488:        if (vap->va_size != VNOVAL) {
                    489:                switch (vp->v_type) {
                    490:                case VDIR:
                    491:                        return (EISDIR);
                    492:                case VCHR:
                    493:                case VBLK:
                    494:                case VSOCK:
                    495:                case VFIFO:
                    496:                        if (vap->va_mtime.tv_sec == VNOVAL &&
                    497:                            vap->va_atime.tv_sec == VNOVAL &&
                    498:                            vap->va_mode == (mode_t)VNOVAL &&
                    499:                            vap->va_uid == (uid_t)VNOVAL &&
                    500:                            vap->va_gid == (gid_t)VNOVAL)
                    501:                                return (0);
                    502:                        vap->va_size = VNOVAL;
                    503:                        break;
                    504:                default:
                    505:                        /*
                    506:                         * Disallow write attempts if the filesystem is
                    507:                         * mounted read-only.
                    508:                         */
                    509:                        if (vp->v_mount->mnt_flag & MNT_RDONLY)
                    510:                                return (EROFS);
                    511:                        if (vap->va_size == 0)
                    512:                                error = nfs_vinvalbuf(vp, 0,
                    513:                                     ap->a_cred, ap->a_p, 1);
                    514:                        else
                    515:                                error = nfs_vinvalbuf(vp, V_SAVE,
                    516:                                     ap->a_cred, ap->a_p, 1);
                    517:                        if (error)
                    518:                                return (error);
                    519:                        tsize = np->n_size;
                    520:                        np->n_size = np->n_vattr.va_size = vap->va_size;
                    521:                        uvm_vnp_setsize(vp, np->n_size);
                    522:                };
                    523:        } else if ((vap->va_mtime.tv_sec != VNOVAL ||
                    524:                vap->va_atime.tv_sec != VNOVAL) &&
                    525:                vp->v_type == VREG &&
                    526:                (error = nfs_vinvalbuf(vp, V_SAVE, ap->a_cred,
                    527:                 ap->a_p, 1)) == EINTR)
                    528:                return (error);
                    529:        error = nfs_setattrrpc(vp, vap, ap->a_cred, ap->a_p);
                    530:        if (error && vap->va_size != VNOVAL) {
                    531:                np->n_size = np->n_vattr.va_size = tsize;
                    532:                uvm_vnp_setsize(vp, np->n_size);
                    533:        }
                    534:
                    535:        VN_KNOTE(vp, NOTE_ATTRIB); /* XXX setattrrpc? */
                    536:
                    537:        return (error);
                    538: }
                    539:
                    540: /*
                    541:  * Do an nfs setattr rpc.
                    542:  */
                    543: int
                    544: nfs_setattrrpc(vp, vap, cred, procp)
                    545:        struct vnode *vp;
                    546:        struct vattr *vap;
                    547:        struct ucred *cred;
                    548:        struct proc *procp;
                    549: {
                    550:        struct nfsv2_sattr *sp;
                    551:        caddr_t cp;
                    552:        int32_t t1, t2;
                    553:        caddr_t bpos, dpos, cp2;
                    554:        u_int32_t *tl;
                    555:        int error = 0, wccflag = NFSV3_WCCRATTR;
                    556:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                    557:        int v3 = NFS_ISV3(vp);
                    558:
                    559:        nfsstats.rpccnt[NFSPROC_SETATTR]++;
                    560:        nfsm_reqhead(vp, NFSPROC_SETATTR, NFSX_FH(v3) + NFSX_SATTR(v3));
                    561:        nfsm_fhtom(vp, v3);
                    562:        if (v3) {
                    563:                nfsm_v3attrbuild(vap, TRUE);
                    564:                nfsm_build(tl, u_int32_t *, NFSX_UNSIGNED);
                    565:                *tl = nfs_false;
                    566:        } else {
                    567:                nfsm_build(sp, struct nfsv2_sattr *, NFSX_V2SATTR);
                    568:                if (vap->va_mode == (mode_t)VNOVAL)
                    569:                        sp->sa_mode = nfs_xdrneg1;
                    570:                else
                    571:                        sp->sa_mode = vtonfsv2_mode(vp->v_type, vap->va_mode);
                    572:                if (vap->va_uid == (uid_t)VNOVAL)
                    573:                        sp->sa_uid = nfs_xdrneg1;
                    574:                else
                    575:                        sp->sa_uid = txdr_unsigned(vap->va_uid);
                    576:                if (vap->va_gid == (gid_t)VNOVAL)
                    577:                        sp->sa_gid = nfs_xdrneg1;
                    578:                else
                    579:                        sp->sa_gid = txdr_unsigned(vap->va_gid);
                    580:                sp->sa_size = txdr_unsigned(vap->va_size);
                    581:                txdr_nfsv2time(&vap->va_atime, &sp->sa_atime);
                    582:                txdr_nfsv2time(&vap->va_mtime, &sp->sa_mtime);
                    583:        }
                    584:        nfsm_request(vp, NFSPROC_SETATTR, procp, cred);
                    585:        if (v3) {
                    586:                nfsm_wcc_data(vp, wccflag);
                    587:        } else
                    588:                nfsm_loadattr(vp, (struct vattr *)0);
                    589:        nfsm_reqdone;
                    590:        return (error);
                    591: }
                    592:
                    593: /*
                    594:  * nfs lookup call, one step at a time...
                    595:  * First look in cache
                    596:  * If not found, unlock the directory nfsnode and do the rpc
                    597:  */
                    598: int
                    599: nfs_lookup(v)
                    600:        void *v;
                    601: {
                    602:        struct vop_lookup_args *ap = v;
                    603:        struct componentname *cnp = ap->a_cnp;
                    604:        struct vnode *dvp = ap->a_dvp;
                    605:        struct vnode **vpp = ap->a_vpp;
                    606:        struct proc *p = cnp->cn_proc;
                    607:        int flags;
                    608:        struct vnode *newvp;
                    609:        u_int32_t *tl;
                    610:        caddr_t cp;
                    611:        int32_t t1, t2;
                    612:        struct nfsmount *nmp;
                    613:        caddr_t bpos, dpos, cp2;
                    614:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                    615:        long len;
                    616:        nfsfh_t *fhp;
                    617:        struct nfsnode *np;
                    618:        int lockparent, wantparent, error = 0, attrflag, fhsize;
                    619:        int v3 = NFS_ISV3(dvp);
                    620:
                    621:        cnp->cn_flags &= ~PDIRUNLOCK;
                    622:        flags = cnp->cn_flags;
                    623:
                    624:        *vpp = NULLVP;
                    625:        if ((flags & ISLASTCN) && (dvp->v_mount->mnt_flag & MNT_RDONLY) &&
                    626:            (cnp->cn_nameiop == DELETE || cnp->cn_nameiop == RENAME))
                    627:                return (EROFS);
                    628:        if (dvp->v_type != VDIR)
                    629:                return (ENOTDIR);
                    630:        lockparent = flags & LOCKPARENT;
                    631:        wantparent = flags & (LOCKPARENT|WANTPARENT);
                    632:        nmp = VFSTONFS(dvp->v_mount);
                    633:        np = VTONFS(dvp);
                    634:
                    635:        /*
                    636:         * Before tediously performing a linear scan of the directory,
                    637:         * check the name cache to see if the directory/name pair
                    638:         * we are looking for is known already.
                    639:         * If the directory/name pair is found in the name cache,
                    640:         * we have to ensure the directory has not changed from
                    641:         * the time the cache entry has been created. If it has,
                    642:         * the cache entry has to be ignored.
                    643:         */
                    644:        if ((error = cache_lookup(dvp, vpp, cnp)) >= 0) {
                    645:                struct vattr vattr;
                    646:                int err2;
                    647:
                    648:                if (error && error != ENOENT) {
                    649:                        *vpp = NULLVP;
                    650:                        return (error);
                    651:                }
                    652:
                    653:                if (cnp->cn_flags & PDIRUNLOCK) {
                    654:                        err2 = vn_lock(dvp, LK_EXCLUSIVE | LK_RETRY, p);
                    655:                        if (err2 != 0) {
                    656:                                *vpp = NULLVP;
                    657:                                return (err2);
                    658:                        }
                    659:                        cnp->cn_flags &= ~PDIRUNLOCK;
                    660:                }
                    661:
                    662:                err2 = VOP_ACCESS(dvp, VEXEC, cnp->cn_cred, cnp->cn_proc);
                    663:                if (err2 != 0) {
                    664:                        if (error == 0) {
                    665:                                if (*vpp != dvp)
                    666:                                        vput(*vpp);
                    667:                                else
                    668:                                        vrele(*vpp);
                    669:                        }
                    670:                        *vpp = NULLVP;
                    671:                        return (err2);
                    672:                }
                    673:
                    674:                if (error == ENOENT) {
                    675:                        if (!VOP_GETATTR(dvp, &vattr, cnp->cn_cred,
                    676:                            cnp->cn_proc) && vattr.va_mtime.tv_sec ==
                    677:                            VTONFS(dvp)->n_ctime)
                    678:                                return (ENOENT);
                    679:                        cache_purge(dvp);
                    680:                        np->n_ctime = 0;
                    681:                        goto dorpc;
                    682:                }
                    683:
                    684:                newvp = *vpp;
                    685:                if (!VOP_GETATTR(newvp, &vattr, cnp->cn_cred, cnp->cn_proc)
                    686:                        && vattr.va_ctime.tv_sec == VTONFS(newvp)->n_ctime)
                    687:                {
                    688:                        nfsstats.lookupcache_hits++;
                    689:                        if (cnp->cn_nameiop != LOOKUP && (flags & ISLASTCN))
                    690:                                cnp->cn_flags |= SAVENAME;
                    691:                        if ((!lockparent || !(flags & ISLASTCN)) &&
                    692:                             newvp != dvp)
                    693:                                VOP_UNLOCK(dvp, 0, p);
                    694:                        return (0);
                    695:                }
                    696:                cache_purge(newvp);
                    697:                if (newvp != dvp)
                    698:                        vput(newvp);
                    699:                else
                    700:                        vrele(newvp);
                    701:                *vpp = NULLVP;
                    702:        }
                    703: dorpc:
                    704:        error = 0;
                    705:        newvp = NULLVP;
                    706:        nfsstats.lookupcache_misses++;
                    707:        nfsstats.rpccnt[NFSPROC_LOOKUP]++;
                    708:        len = cnp->cn_namelen;
                    709:        nfsm_reqhead(dvp, NFSPROC_LOOKUP,
                    710:                NFSX_FH(v3) + NFSX_UNSIGNED + nfsm_rndup(len));
                    711:        nfsm_fhtom(dvp, v3);
                    712:        nfsm_strtom(cnp->cn_nameptr, len, NFS_MAXNAMLEN);
                    713:        nfsm_request(dvp, NFSPROC_LOOKUP, cnp->cn_proc, cnp->cn_cred);
                    714:        if (error) {
                    715:                nfsm_postop_attr(dvp, attrflag);
                    716:                m_freem(mrep);
                    717:                goto nfsmout;
                    718:        }
                    719:        nfsm_getfh(fhp, fhsize, v3);
                    720:
                    721:        /*
                    722:         * Handle RENAME case...
                    723:         */
                    724:        if (cnp->cn_nameiop == RENAME && wantparent && (flags & ISLASTCN)) {
                    725:                if (NFS_CMPFH(np, fhp, fhsize)) {
                    726:                        m_freem(mrep);
                    727:                        return (EISDIR);
                    728:                }
                    729:                error = nfs_nget(dvp->v_mount, fhp, fhsize, &np);
                    730:                if (error) {
                    731:                        m_freem(mrep);
                    732:                        return (error);
                    733:                }
                    734:                newvp = NFSTOV(np);
                    735:                if (v3) {
                    736:                        nfsm_postop_attr(newvp, attrflag);
                    737:                        nfsm_postop_attr(dvp, attrflag);
                    738:                } else
                    739:                        nfsm_loadattr(newvp, (struct vattr *)0);
                    740:                *vpp = newvp;
                    741:                m_freem(mrep);
                    742:                cnp->cn_flags |= SAVENAME;
                    743:                if (!lockparent) {
                    744:                        VOP_UNLOCK(dvp, 0, p);
                    745:                        cnp->cn_flags |= PDIRUNLOCK;
                    746:                }
                    747:                return (0);
                    748:        }
                    749:
                    750:        /*
                    751:         * The postop attr handling is duplicated for each if case,
                    752:         * because it should be done while dvp is locked (unlocking
                    753:         * dvp is different for each case).
                    754:         */
                    755:
                    756:        if (NFS_CMPFH(np, fhp, fhsize)) {
                    757:                VREF(dvp);
                    758:                newvp = dvp;
                    759:                if (v3) {
                    760:                        nfsm_postop_attr(newvp, attrflag);
                    761:                        nfsm_postop_attr(dvp, attrflag);
                    762:                } else
                    763:                        nfsm_loadattr(newvp, (struct vattr *)0);
                    764:        } else if (flags & ISDOTDOT) {
                    765:                VOP_UNLOCK(dvp, 0, p);
                    766:                cnp->cn_flags |= PDIRUNLOCK;
                    767:
                    768:                error = nfs_nget(dvp->v_mount, fhp, fhsize, &np);
                    769:                if (error) {
                    770:                        if (vn_lock(dvp, LK_EXCLUSIVE | LK_RETRY, p) == 0)
                    771:                                cnp->cn_flags &= ~PDIRUNLOCK;
                    772:                        m_freem(mrep);
                    773:                        return (error);
                    774:                }
                    775:                newvp = NFSTOV(np);
                    776:
                    777:                if (v3) {
                    778:                        nfsm_postop_attr(newvp, attrflag);
                    779:                        nfsm_postop_attr(dvp, attrflag);
                    780:                } else
                    781:                        nfsm_loadattr(newvp, (struct vattr *)0);
                    782:
                    783:                if (lockparent && (flags & ISLASTCN)) {
                    784:                        if ((error = vn_lock(dvp, LK_EXCLUSIVE, p))) {
                    785:                                m_freem(mrep);
                    786:                                vput(newvp);
                    787:                                return error;
                    788:                        }
                    789:                        cnp->cn_flags &= ~PDIRUNLOCK;
                    790:                }
                    791:
                    792:        } else {
                    793:                error = nfs_nget(dvp->v_mount, fhp, fhsize, &np);
                    794:                if (error) {
                    795:                        m_freem(mrep);
                    796:                        return error;
                    797:                }
                    798:                newvp = NFSTOV(np);
                    799:                if (v3) {
                    800:                        nfsm_postop_attr(newvp, attrflag);
                    801:                        nfsm_postop_attr(dvp, attrflag);
                    802:                } else
                    803:                        nfsm_loadattr(newvp, (struct vattr *)0);
                    804:                if (!lockparent || !(flags & ISLASTCN)) {
                    805:                        VOP_UNLOCK(dvp, 0, p);
                    806:                        cnp->cn_flags |= PDIRUNLOCK;
                    807:                }
                    808:        }
                    809:        if (cnp->cn_nameiop != LOOKUP && (flags & ISLASTCN))
                    810:                cnp->cn_flags |= SAVENAME;
                    811:        if ((cnp->cn_flags & MAKEENTRY) &&
                    812:            (cnp->cn_nameiop != DELETE || !(flags & ISLASTCN))) {
                    813:                np->n_ctime = np->n_vattr.va_ctime.tv_sec;
                    814:                cache_enter(dvp, newvp, cnp);
                    815:        }
                    816:        *vpp = newvp;
                    817:        nfsm_reqdone;
                    818:        if (error) {
                    819:                /*
                    820:                 * We get here only because of errors returned by
                    821:                 * the RPC. Otherwise we'll have returned above
                    822:                 * (the nfsm_* macros will jump to nfsm_reqdone
                    823:                 * on error).
                    824:                 */
                    825:                if (error == ENOENT && (cnp->cn_flags & MAKEENTRY) &&
                    826:                    cnp->cn_nameiop != CREATE) {
                    827:                        if (VTONFS(dvp)->n_ctime == 0)
                    828:                                VTONFS(dvp)->n_ctime =
                    829:                                    VTONFS(dvp)->n_vattr.va_mtime.tv_sec;
                    830:                        cache_enter(dvp, NULL, cnp);
                    831:                }
                    832:                if (newvp != NULLVP) {
                    833:                        vrele(newvp);
                    834:                        if (newvp != dvp)
                    835:                                VOP_UNLOCK(newvp, 0, p);
                    836:                }
                    837:                if ((cnp->cn_nameiop == CREATE || cnp->cn_nameiop == RENAME) &&
                    838:                    (flags & ISLASTCN) && error == ENOENT) {
                    839:                        if (dvp->v_mount->mnt_flag & MNT_RDONLY)
                    840:                                error = EROFS;
                    841:                        else
                    842:                                error = EJUSTRETURN;
                    843:                }
                    844:                if (cnp->cn_nameiop != LOOKUP && (flags & ISLASTCN))
                    845:                        cnp->cn_flags |= SAVENAME;
                    846:                *vpp = NULL;
                    847:        }
                    848:        return (error);
                    849: }
                    850:
                    851: /*
                    852:  * nfs read call.
                    853:  * Just call nfs_bioread() to do the work.
                    854:  */
                    855: int
                    856: nfs_read(v)
                    857:        void *v;
                    858: {
                    859:        struct vop_read_args *ap = v;
                    860:        struct vnode *vp = ap->a_vp;
                    861:
                    862:        if (vp->v_type != VREG)
                    863:                return (EPERM);
                    864:        return (nfs_bioread(vp, ap->a_uio, ap->a_ioflag, ap->a_cred));
                    865: }
                    866:
                    867: /*
                    868:  * nfs readlink call
                    869:  */
                    870: int
                    871: nfs_readlink(v)
                    872:        void *v;
                    873: {
                    874:        struct vop_readlink_args *ap = v;
                    875:        struct vnode *vp = ap->a_vp;
                    876:
                    877:        if (vp->v_type != VLNK)
                    878:                return (EPERM);
                    879:        return (nfs_bioread(vp, ap->a_uio, 0, ap->a_cred));
                    880: }
                    881:
                    882: /*
                    883:  * Do a readlink rpc.
                    884:  * Called by nfs_doio() from below the buffer cache.
                    885:  */
                    886: int
                    887: nfs_readlinkrpc(vp, uiop, cred)
                    888:        struct vnode *vp;
                    889:        struct uio *uiop;
                    890:        struct ucred *cred;
                    891: {
                    892:        u_int32_t *tl;
                    893:        caddr_t cp;
                    894:        int32_t t1, t2;
                    895:        caddr_t bpos, dpos, cp2;
                    896:        int error = 0, len, attrflag;
                    897:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                    898:        int v3 = NFS_ISV3(vp);
                    899:
                    900:        nfsstats.rpccnt[NFSPROC_READLINK]++;
                    901:        nfsm_reqhead(vp, NFSPROC_READLINK, NFSX_FH(v3));
                    902:        nfsm_fhtom(vp, v3);
                    903:        nfsm_request(vp, NFSPROC_READLINK, uiop->uio_procp, cred);
                    904:        if (v3)
                    905:                nfsm_postop_attr(vp, attrflag);
                    906:        if (!error) {
                    907:                nfsm_strsiz(len, NFS_MAXPATHLEN);
                    908:                nfsm_mtouio(uiop, len);
                    909:        }
                    910:        nfsm_reqdone;
                    911:        return (error);
                    912: }
                    913:
                    914: /*
                    915:  * nfs read rpc call
                    916:  * Ditto above
                    917:  */
                    918: int
                    919: nfs_readrpc(vp, uiop)
                    920:        struct vnode *vp;
                    921:        struct uio *uiop;
                    922: {
                    923:        u_int32_t *tl;
                    924:        caddr_t cp;
                    925:        int32_t t1, t2;
                    926:        caddr_t bpos, dpos, cp2;
                    927:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                    928:        struct nfsmount *nmp;
                    929:        int error = 0, len, retlen, tsiz, eof, attrflag;
                    930:        int v3 = NFS_ISV3(vp);
                    931:
                    932: #ifndef nolint
                    933:        eof = 0;
                    934: #endif
                    935:        nmp = VFSTONFS(vp->v_mount);
                    936:        tsiz = uiop->uio_resid;
                    937:        if (uiop->uio_offset + tsiz > 0xffffffff && !v3)
                    938:                return (EFBIG);
                    939:        while (tsiz > 0) {
                    940:                nfsstats.rpccnt[NFSPROC_READ]++;
                    941:                len = (tsiz > nmp->nm_rsize) ? nmp->nm_rsize : tsiz;
                    942:                nfsm_reqhead(vp, NFSPROC_READ, NFSX_FH(v3) + NFSX_UNSIGNED * 3);
                    943:                nfsm_fhtom(vp, v3);
                    944:                nfsm_build(tl, u_int32_t *, NFSX_UNSIGNED * 3);
                    945:                if (v3) {
                    946:                        txdr_hyper(uiop->uio_offset, tl);
                    947:                        *(tl + 2) = txdr_unsigned(len);
                    948:                } else {
                    949:                        *tl++ = txdr_unsigned(uiop->uio_offset);
                    950:                        *tl++ = txdr_unsigned(len);
                    951:                        *tl = 0;
                    952:                }
                    953:                nfsm_request(vp, NFSPROC_READ, uiop->uio_procp,
                    954:                    VTONFS(vp)->n_rcred);
                    955:                if (v3) {
                    956:                        nfsm_postop_attr(vp, attrflag);
                    957:                        if (error) {
                    958:                                m_freem(mrep);
                    959:                                goto nfsmout;
                    960:                        }
                    961:                        nfsm_dissect(tl, u_int32_t *, 2 * NFSX_UNSIGNED);
                    962:                        eof = fxdr_unsigned(int, *(tl + 1));
                    963:                } else
                    964:                        nfsm_loadattr(vp, (struct vattr *)0);
                    965:                nfsm_strsiz(retlen, nmp->nm_rsize);
                    966:                nfsm_mtouio(uiop, retlen);
                    967:                m_freem(mrep);
                    968:                tsiz -= retlen;
                    969:                if (v3) {
                    970:                        if (eof || retlen == 0)
                    971:                                tsiz = 0;
                    972:                } else if (retlen < len)
                    973:                        tsiz = 0;
                    974:        }
                    975: nfsmout:
                    976:        return (error);
                    977: }
                    978:
                    979: /*
                    980:  * nfs write call
                    981:  */
                    982: int
                    983: nfs_writerpc(vp, uiop, iomode, must_commit)
                    984:        struct vnode *vp;
                    985:        struct uio *uiop;
                    986:        int *iomode, *must_commit;
                    987: {
                    988:        u_int32_t *tl;
                    989:        caddr_t cp;
                    990:        int32_t t1, t2, backup;
                    991:        caddr_t bpos, dpos, cp2;
                    992:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                    993:        struct nfsmount *nmp = VFSTONFS(vp->v_mount);
                    994:        int error = 0, len, tsiz, wccflag = NFSV3_WCCRATTR, rlen, commit;
                    995:        int v3 = NFS_ISV3(vp), committed = NFSV3WRITE_FILESYNC;
                    996:
                    997: #ifndef DIAGNOSTIC
                    998:        if (uiop->uio_iovcnt != 1)
                    999:                panic("nfs: writerpc iovcnt > 1");
                   1000: #endif
                   1001:        *must_commit = 0;
                   1002:        tsiz = uiop->uio_resid;
                   1003:        if (uiop->uio_offset + tsiz > 0xffffffff && !v3)
                   1004:                return (EFBIG);
                   1005:        while (tsiz > 0) {
                   1006:                nfsstats.rpccnt[NFSPROC_WRITE]++;
                   1007:                len = (tsiz > nmp->nm_wsize) ? nmp->nm_wsize : tsiz;
                   1008:                nfsm_reqhead(vp, NFSPROC_WRITE,
                   1009:                        NFSX_FH(v3) + 5 * NFSX_UNSIGNED + nfsm_rndup(len));
                   1010:                nfsm_fhtom(vp, v3);
                   1011:                if (v3) {
                   1012:                        nfsm_build(tl, u_int32_t *, 5 * NFSX_UNSIGNED);
                   1013:                        txdr_hyper(uiop->uio_offset, tl);
                   1014:                        tl += 2;
                   1015:                        *tl++ = txdr_unsigned(len);
                   1016:                        *tl++ = txdr_unsigned(*iomode);
                   1017:                        *tl = txdr_unsigned(len);
                   1018:                } else {
                   1019:                        u_int32_t x;
                   1020:
                   1021:                        nfsm_build(tl, u_int32_t *, 4 * NFSX_UNSIGNED);
                   1022:                        /* Set both "begin" and "current" to non-garbage. */
                   1023:                        x = txdr_unsigned((u_int32_t)uiop->uio_offset);
                   1024:                        *tl++ = x;      /* "begin offset" */
                   1025:                        *tl++ = x;      /* "current offset" */
                   1026:                        x = txdr_unsigned(len);
                   1027:                        *tl++ = x;      /* total to this offset */
                   1028:                        *tl = x;        /* size of this write */
                   1029:
                   1030:                }
                   1031:                nfsm_uiotom(uiop, len);
                   1032:                nfsm_request(vp, NFSPROC_WRITE, uiop->uio_procp,
                   1033:                    VTONFS(vp)->n_wcred);
                   1034:                if (v3) {
                   1035:                        wccflag = NFSV3_WCCCHK;
                   1036:                        nfsm_wcc_data(vp, wccflag);
                   1037:                        if (!error) {
                   1038:                                nfsm_dissect(tl, u_int32_t *, 2 * NFSX_UNSIGNED
                   1039:                                        + NFSX_V3WRITEVERF);
                   1040:                                rlen = fxdr_unsigned(int, *tl++);
                   1041:                                if (rlen == 0) {
                   1042:                                        error = NFSERR_IO;
                   1043:                                        break;
                   1044:                                } else if (rlen < len) {
                   1045:                                        backup = len - rlen;
                   1046:                                        (char *)uiop->uio_iov->iov_base -= backup;
                   1047:                                        uiop->uio_iov->iov_len += backup;
                   1048:                                        uiop->uio_offset -= backup;
                   1049:                                        uiop->uio_resid += backup;
                   1050:                                        len = rlen;
                   1051:                                }
                   1052:                                commit = fxdr_unsigned(int, *tl++);
                   1053:
                   1054:                                /*
                   1055:                                 * Return the lowest committment level
                   1056:                                 * obtained by any of the RPCs.
                   1057:                                 */
                   1058:                                if (committed == NFSV3WRITE_FILESYNC)
                   1059:                                        committed = commit;
                   1060:                                else if (committed == NFSV3WRITE_DATASYNC &&
                   1061:                                        commit == NFSV3WRITE_UNSTABLE)
                   1062:                                        committed = commit;
                   1063:                                if ((nmp->nm_flag & NFSMNT_HASWRITEVERF) == 0) {
                   1064:                                    bcopy((caddr_t)tl, (caddr_t)nmp->nm_verf,
                   1065:                                        NFSX_V3WRITEVERF);
                   1066:                                    nmp->nm_flag |= NFSMNT_HASWRITEVERF;
                   1067:                                } else if (bcmp((caddr_t)tl,
                   1068:                                    (caddr_t)nmp->nm_verf, NFSX_V3WRITEVERF)) {
                   1069:                                    *must_commit = 1;
                   1070:                                    bcopy((caddr_t)tl, (caddr_t)nmp->nm_verf,
                   1071:                                        NFSX_V3WRITEVERF);
                   1072:                                }
                   1073:                        }
                   1074:                } else
                   1075:                    nfsm_loadattr(vp, (struct vattr *)0);
                   1076:                if (wccflag)
                   1077:                    VTONFS(vp)->n_mtime = VTONFS(vp)->n_vattr.va_mtime.tv_sec;
                   1078:                m_freem(mrep);
                   1079:                tsiz -= len;
                   1080:        }
                   1081: nfsmout:
                   1082:        *iomode = committed;
                   1083:        if (error)
                   1084:                uiop->uio_resid = tsiz;
                   1085:        return (error);
                   1086: }
                   1087:
                   1088: /*
                   1089:  * nfs mknod rpc
                   1090:  * For NFS v2 this is a kludge. Use a create rpc but with the IFMT bits of the
                   1091:  * mode set to specify the file type and the size field for rdev.
                   1092:  */
                   1093: int
                   1094: nfs_mknodrpc(dvp, vpp, cnp, vap)
                   1095:        struct vnode *dvp;
                   1096:        struct vnode **vpp;
                   1097:        struct componentname *cnp;
                   1098:        struct vattr *vap;
                   1099: {
                   1100:        struct nfsv2_sattr *sp;
                   1101:        u_int32_t *tl;
                   1102:        caddr_t cp;
                   1103:        int32_t t1, t2;
                   1104:        struct vnode *newvp = (struct vnode *)0;
                   1105:        struct nfsnode *np;
                   1106:        char *cp2;
                   1107:        caddr_t bpos, dpos;
                   1108:        int error = 0, wccflag = NFSV3_WCCRATTR, gotvp = 0;
                   1109:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                   1110:        u_int32_t rdev;
                   1111:        int v3 = NFS_ISV3(dvp);
                   1112:
                   1113:        if (vap->va_type == VCHR || vap->va_type == VBLK)
                   1114:                rdev = txdr_unsigned(vap->va_rdev);
                   1115:        else if (vap->va_type == VFIFO || vap->va_type == VSOCK)
                   1116:                rdev = nfs_xdrneg1;
                   1117:        else {
                   1118:                VOP_ABORTOP(dvp, cnp);
                   1119:                vput(dvp);
                   1120:                return (EOPNOTSUPP);
                   1121:        }
                   1122:        nfsstats.rpccnt[NFSPROC_MKNOD]++;
                   1123:        nfsm_reqhead(dvp, NFSPROC_MKNOD, NFSX_FH(v3) + 4 * NFSX_UNSIGNED +
                   1124:                + nfsm_rndup(cnp->cn_namelen) + NFSX_SATTR(v3));
                   1125:        nfsm_fhtom(dvp, v3);
                   1126:        nfsm_strtom(cnp->cn_nameptr, cnp->cn_namelen, NFS_MAXNAMLEN);
                   1127:        if (v3) {
                   1128:                nfsm_build(tl, u_int32_t *, NFSX_UNSIGNED);
                   1129:                *tl++ = vtonfsv3_type(vap->va_type);
                   1130:                nfsm_v3attrbuild(vap, FALSE);
                   1131:                if (vap->va_type == VCHR || vap->va_type == VBLK) {
                   1132:                        nfsm_build(tl, u_int32_t *, 2 * NFSX_UNSIGNED);
                   1133:                        *tl++ = txdr_unsigned(major(vap->va_rdev));
                   1134:                        *tl = txdr_unsigned(minor(vap->va_rdev));
                   1135:                }
                   1136:        } else {
                   1137:                nfsm_build(sp, struct nfsv2_sattr *, NFSX_V2SATTR);
                   1138:                sp->sa_mode = vtonfsv2_mode(vap->va_type, vap->va_mode);
                   1139:                sp->sa_uid = nfs_xdrneg1;
                   1140:                sp->sa_gid = nfs_xdrneg1;
                   1141:                sp->sa_size = rdev;
                   1142:                txdr_nfsv2time(&vap->va_atime, &sp->sa_atime);
                   1143:                txdr_nfsv2time(&vap->va_mtime, &sp->sa_mtime);
                   1144:        }
                   1145:        nfsm_request(dvp, NFSPROC_MKNOD, cnp->cn_proc, cnp->cn_cred);
                   1146:        if (!error) {
                   1147:                nfsm_mtofh(dvp, newvp, v3, gotvp);
                   1148:                if (!gotvp) {
                   1149:                        if (newvp) {
                   1150:                                vrele(newvp);
                   1151:                                newvp = (struct vnode *)0;
                   1152:                        }
                   1153:                        error = nfs_lookitup(dvp, cnp->cn_nameptr,
                   1154:                            cnp->cn_namelen, cnp->cn_cred, cnp->cn_proc, &np);
                   1155:                        if (!error)
                   1156:                                newvp = NFSTOV(np);
                   1157:                }
                   1158:        }
                   1159:        if (v3)
                   1160:                nfsm_wcc_data(dvp, wccflag);
                   1161:        nfsm_reqdone;
                   1162:        if (error) {
                   1163:                if (newvp)
                   1164:                        vrele(newvp);
                   1165:        } else {
                   1166:                if (cnp->cn_flags & MAKEENTRY)
                   1167:                        cache_enter(dvp, newvp, cnp);
                   1168:                *vpp = newvp;
                   1169:        }
                   1170:        pool_put(&namei_pool, cnp->cn_pnbuf);
                   1171:        VTONFS(dvp)->n_flag |= NMODIFIED;
                   1172:        if (!wccflag)
                   1173:                VTONFS(dvp)->n_attrstamp = 0;
                   1174:        vrele(dvp);
                   1175:        return (error);
                   1176: }
                   1177:
                   1178: /*
                   1179:  * nfs mknod vop
                   1180:  * just call nfs_mknodrpc() to do the work.
                   1181:  */
                   1182: /* ARGSUSED */
                   1183: int
                   1184: nfs_mknod(v)
                   1185:        void *v;
                   1186: {
                   1187:        struct vop_mknod_args *ap = v;
                   1188:        struct vnode *newvp;
                   1189:        int error;
                   1190:
                   1191:        error = nfs_mknodrpc(ap->a_dvp, &newvp, ap->a_cnp, ap->a_vap);
                   1192:        if (!error)
                   1193:                vrele(newvp);
                   1194:
                   1195:        VN_KNOTE(ap->a_dvp, NOTE_WRITE);
                   1196:
                   1197:        return (error);
                   1198: }
                   1199:
                   1200: static u_long create_verf;
                   1201: /*
                   1202:  * nfs file create call
                   1203:  */
                   1204: int
                   1205: nfs_create(v)
                   1206:        void *v;
                   1207: {
                   1208:        struct vop_create_args *ap = v;
                   1209:        struct vnode *dvp = ap->a_dvp;
                   1210:        struct vattr *vap = ap->a_vap;
                   1211:        struct componentname *cnp = ap->a_cnp;
                   1212:        struct nfsv2_sattr *sp;
                   1213:        u_int32_t *tl;
                   1214:        caddr_t cp;
                   1215:        int32_t t1, t2;
                   1216:        struct nfsnode *np = (struct nfsnode *)0;
                   1217:        struct vnode *newvp = (struct vnode *)0;
                   1218:        caddr_t bpos, dpos, cp2;
                   1219:        int error = 0, wccflag = NFSV3_WCCRATTR, gotvp = 0, fmode = 0;
                   1220:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                   1221:        int v3 = NFS_ISV3(dvp);
                   1222:
                   1223:        /*
                   1224:         * Oops, not for me..
                   1225:         */
                   1226:        if (vap->va_type == VSOCK)
                   1227:                return (nfs_mknodrpc(dvp, ap->a_vpp, cnp, vap));
                   1228:
                   1229: #ifdef VA_EXCLUSIVE
                   1230:        if (vap->va_vaflags & VA_EXCLUSIVE)
                   1231:                fmode |= O_EXCL;
                   1232: #endif
                   1233: again:
                   1234:        nfsstats.rpccnt[NFSPROC_CREATE]++;
                   1235:        nfsm_reqhead(dvp, NFSPROC_CREATE, NFSX_FH(v3) + 2 * NFSX_UNSIGNED +
                   1236:                nfsm_rndup(cnp->cn_namelen) + NFSX_SATTR(v3));
                   1237:        nfsm_fhtom(dvp, v3);
                   1238:        nfsm_strtom(cnp->cn_nameptr, cnp->cn_namelen, NFS_MAXNAMLEN);
                   1239:        if (v3) {
                   1240:                nfsm_build(tl, u_int32_t *, NFSX_UNSIGNED);
                   1241:                if (fmode & O_EXCL) {
                   1242:                        *tl = txdr_unsigned(NFSV3CREATE_EXCLUSIVE);
                   1243:                        nfsm_build(tl, u_int32_t *, NFSX_V3CREATEVERF);
                   1244:                        if (TAILQ_FIRST(&in_ifaddr))
                   1245:                                *tl++ = TAILQ_FIRST(&in_ifaddr)->ia_addr.sin_addr.s_addr;
                   1246:                        else
                   1247:                                *tl++ = create_verf;
                   1248:                        *tl = ++create_verf;
                   1249:                } else {
                   1250:                        *tl = txdr_unsigned(NFSV3CREATE_UNCHECKED);
                   1251:                        nfsm_v3attrbuild(vap, FALSE);
                   1252:                }
                   1253:        } else {
                   1254:                nfsm_build(sp, struct nfsv2_sattr *, NFSX_V2SATTR);
                   1255:                sp->sa_mode = vtonfsv2_mode(vap->va_type, vap->va_mode);
                   1256:                sp->sa_uid = nfs_xdrneg1;
                   1257:                sp->sa_gid = nfs_xdrneg1;
                   1258:                sp->sa_size = 0;
                   1259:                txdr_nfsv2time(&vap->va_atime, &sp->sa_atime);
                   1260:                txdr_nfsv2time(&vap->va_mtime, &sp->sa_mtime);
                   1261:        }
                   1262:        nfsm_request(dvp, NFSPROC_CREATE, cnp->cn_proc, cnp->cn_cred);
                   1263:        if (!error) {
                   1264:                nfsm_mtofh(dvp, newvp, v3, gotvp);
                   1265:                if (!gotvp) {
                   1266:                        if (newvp) {
                   1267:                                vrele(newvp);
                   1268:                                newvp = (struct vnode *)0;
                   1269:                        }
                   1270:                        error = nfs_lookitup(dvp, cnp->cn_nameptr,
                   1271:                            cnp->cn_namelen, cnp->cn_cred, cnp->cn_proc, &np);
                   1272:                        if (!error)
                   1273:                                newvp = NFSTOV(np);
                   1274:                }
                   1275:        }
                   1276:        if (v3)
                   1277:                nfsm_wcc_data(dvp, wccflag);
                   1278:        nfsm_reqdone;
                   1279:        if (error) {
                   1280:                if (v3 && (fmode & O_EXCL) && error == NFSERR_NOTSUPP) {
                   1281:                        fmode &= ~O_EXCL;
                   1282:                        goto again;
                   1283:                }
                   1284:                if (newvp)
                   1285:                        vrele(newvp);
                   1286:        } else if (v3 && (fmode & O_EXCL))
                   1287:                error = nfs_setattrrpc(newvp, vap, cnp->cn_cred, cnp->cn_proc);
                   1288:        if (!error) {
                   1289:                if (cnp->cn_flags & MAKEENTRY)
                   1290:                        cache_enter(dvp, newvp, cnp);
                   1291:                *ap->a_vpp = newvp;
                   1292:        }
                   1293:        pool_put(&namei_pool, cnp->cn_pnbuf);
                   1294:        VTONFS(dvp)->n_flag |= NMODIFIED;
                   1295:        if (!wccflag)
                   1296:                VTONFS(dvp)->n_attrstamp = 0;
                   1297:        VN_KNOTE(ap->a_dvp, NOTE_WRITE);
                   1298:        vrele(dvp);
                   1299:        return (error);
                   1300: }
                   1301:
                   1302: /*
                   1303:  * nfs file remove call
                   1304:  * To try and make nfs semantics closer to ufs semantics, a file that has
                   1305:  * other processes using the vnode is renamed instead of removed and then
                   1306:  * removed later on the last close.
                   1307:  * - If v_usecount > 1
                   1308:  *       If a rename is not already in the works
                   1309:  *          call nfs_sillyrename() to set it up
                   1310:  *     else
                   1311:  *       do the remove rpc
                   1312:  */
                   1313: int
                   1314: nfs_remove(v)
                   1315:        void *v;
                   1316: {
                   1317:        struct vop_remove_args *ap = v;
                   1318:        struct vnode *vp = ap->a_vp;
                   1319:        struct vnode *dvp = ap->a_dvp;
                   1320:        struct componentname *cnp = ap->a_cnp;
                   1321:        struct nfsnode *np = VTONFS(vp);
                   1322:        int error = 0;
                   1323:        struct vattr vattr;
                   1324:
                   1325: #ifndef DIAGNOSTIC
                   1326:        if ((cnp->cn_flags & HASBUF) == 0)
                   1327:                panic("nfs_remove: no name");
                   1328:        if (vp->v_usecount < 1)
                   1329:                panic("nfs_remove: bad v_usecount");
                   1330: #endif
                   1331:        if (vp->v_type == VDIR)
                   1332:                error = EPERM;
                   1333:        else if (vp->v_usecount == 1 || (np->n_sillyrename &&
                   1334:            VOP_GETATTR(vp, &vattr, cnp->cn_cred, cnp->cn_proc) == 0 &&
                   1335:            vattr.va_nlink > 1)) {
                   1336:                /*
                   1337:                 * Purge the name cache so that the chance of a lookup for
                   1338:                 * the name succeeding while the remove is in progress is
                   1339:                 * minimized. Without node locking it can still happen, such
                   1340:                 * that an I/O op returns ESTALE, but since you get this if
                   1341:                 * another host removes the file..
                   1342:                 */
                   1343:                cache_purge(vp);
                   1344:                /*
                   1345:                 * throw away biocache buffers, mainly to avoid
                   1346:                 * unnecessary delayed writes later.
                   1347:                 */
                   1348:                error = nfs_vinvalbuf(vp, 0, cnp->cn_cred, cnp->cn_proc, 1);
                   1349:                /* Do the rpc */
                   1350:                if (error != EINTR)
                   1351:                        error = nfs_removerpc(dvp, cnp->cn_nameptr,
                   1352:                                cnp->cn_namelen, cnp->cn_cred, cnp->cn_proc);
                   1353:                /*
                   1354:                 * Kludge City: If the first reply to the remove rpc is lost..
                   1355:                 *   the reply to the retransmitted request will be ENOENT
                   1356:                 *   since the file was in fact removed
                   1357:                 *   Therefore, we cheat and return success.
                   1358:                 */
                   1359:                if (error == ENOENT)
                   1360:                        error = 0;
                   1361:        } else if (!np->n_sillyrename)
                   1362:                error = nfs_sillyrename(dvp, vp, cnp);
                   1363:        pool_put(&namei_pool, cnp->cn_pnbuf);
                   1364:        np->n_attrstamp = 0;
                   1365:        vrele(dvp);
                   1366:        vrele(vp);
                   1367:
                   1368:        VN_KNOTE(vp, NOTE_DELETE);
                   1369:        VN_KNOTE(dvp, NOTE_WRITE);
                   1370:
                   1371:        return (error);
                   1372: }
                   1373:
                   1374: /*
                   1375:  * nfs file remove rpc called from nfs_inactive
                   1376:  */
                   1377: int
                   1378: nfs_removeit(sp)
                   1379:        struct sillyrename *sp;
                   1380: {
                   1381:
                   1382:        return (nfs_removerpc(sp->s_dvp, sp->s_name, sp->s_namlen, sp->s_cred,
                   1383:                (struct proc *)0));
                   1384: }
                   1385:
                   1386: /*
                   1387:  * Nfs remove rpc, called from nfs_remove() and nfs_removeit().
                   1388:  */
                   1389: int
                   1390: nfs_removerpc(dvp, name, namelen, cred, proc)
                   1391:        struct vnode *dvp;
                   1392:        char *name;
                   1393:        int namelen;
                   1394:        struct ucred *cred;
                   1395:        struct proc *proc;
                   1396: {
                   1397:        u_int32_t *tl;
                   1398:        caddr_t cp;
                   1399:        int32_t t1, t2;
                   1400:        caddr_t bpos, dpos, cp2;
                   1401:        int error = 0, wccflag = NFSV3_WCCRATTR;
                   1402:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                   1403:        int v3 = NFS_ISV3(dvp);
                   1404:
                   1405:        nfsstats.rpccnt[NFSPROC_REMOVE]++;
                   1406:        nfsm_reqhead(dvp, NFSPROC_REMOVE,
                   1407:                NFSX_FH(v3) + NFSX_UNSIGNED + nfsm_rndup(namelen));
                   1408:        nfsm_fhtom(dvp, v3);
                   1409:        nfsm_strtom(name, namelen, NFS_MAXNAMLEN);
                   1410:        nfsm_request(dvp, NFSPROC_REMOVE, proc, cred);
                   1411:        if (v3)
                   1412:                nfsm_wcc_data(dvp, wccflag);
                   1413:        nfsm_reqdone;
                   1414:        VTONFS(dvp)->n_flag |= NMODIFIED;
                   1415:        if (!wccflag)
                   1416:                VTONFS(dvp)->n_attrstamp = 0;
                   1417:        return (error);
                   1418: }
                   1419:
                   1420: /*
                   1421:  * nfs file rename call
                   1422:  */
                   1423: int
                   1424: nfs_rename(v)
                   1425:        void *v;
                   1426: {
                   1427:        struct vop_rename_args  *ap = v;
                   1428:        struct vnode *fvp = ap->a_fvp;
                   1429:        struct vnode *tvp = ap->a_tvp;
                   1430:        struct vnode *fdvp = ap->a_fdvp;
                   1431:        struct vnode *tdvp = ap->a_tdvp;
                   1432:        struct componentname *tcnp = ap->a_tcnp;
                   1433:        struct componentname *fcnp = ap->a_fcnp;
                   1434:        int error;
                   1435:
                   1436: #ifndef DIAGNOSTIC
                   1437:        if ((tcnp->cn_flags & HASBUF) == 0 ||
                   1438:            (fcnp->cn_flags & HASBUF) == 0)
                   1439:                panic("nfs_rename: no name");
                   1440: #endif
                   1441:        /* Check for cross-device rename */
                   1442:        if ((fvp->v_mount != tdvp->v_mount) ||
                   1443:            (tvp && (fvp->v_mount != tvp->v_mount))) {
                   1444:                error = EXDEV;
                   1445:                goto out;
                   1446:        }
                   1447:
                   1448:        /*
                   1449:         * If the tvp exists and is in use, sillyrename it before doing the
                   1450:         * rename of the new file over it.
                   1451:         */
                   1452:        if (tvp && tvp->v_usecount > 1 && !VTONFS(tvp)->n_sillyrename &&
                   1453:            tvp->v_type != VDIR && !nfs_sillyrename(tdvp, tvp, tcnp)) {
                   1454:                VN_KNOTE(tvp, NOTE_DELETE);
                   1455:                vrele(tvp);
                   1456:                tvp = NULL;
                   1457:        }
                   1458:
                   1459:        error = nfs_renamerpc(fdvp, fcnp->cn_nameptr, fcnp->cn_namelen,
                   1460:                tdvp, tcnp->cn_nameptr, tcnp->cn_namelen, tcnp->cn_cred,
                   1461:                tcnp->cn_proc);
                   1462:
                   1463:        VN_KNOTE(fdvp, NOTE_WRITE);
                   1464:        VN_KNOTE(tdvp, NOTE_WRITE);
                   1465:
                   1466:        if (fvp->v_type == VDIR) {
                   1467:                if (tvp != NULL && tvp->v_type == VDIR)
                   1468:                        cache_purge(tdvp);
                   1469:                cache_purge(fdvp);
                   1470:        }
                   1471: out:
                   1472:        if (tdvp == tvp)
                   1473:                vrele(tdvp);
                   1474:        else
                   1475:                vput(tdvp);
                   1476:        if (tvp)
                   1477:                vput(tvp);
                   1478:        vrele(fdvp);
                   1479:        vrele(fvp);
                   1480:        /*
                   1481:         * Kludge: Map ENOENT => 0 assuming that it is a reply to a retry.
                   1482:         */
                   1483:        if (error == ENOENT)
                   1484:                error = 0;
                   1485:        return (error);
                   1486: }
                   1487:
                   1488: /*
                   1489:  * nfs file rename rpc called from nfs_remove() above
                   1490:  */
                   1491: int
                   1492: nfs_renameit(sdvp, scnp, sp)
                   1493:        struct vnode *sdvp;
                   1494:        struct componentname *scnp;
                   1495:        struct sillyrename *sp;
                   1496: {
                   1497:        return (nfs_renamerpc(sdvp, scnp->cn_nameptr, scnp->cn_namelen,
                   1498:                sdvp, sp->s_name, sp->s_namlen, scnp->cn_cred, scnp->cn_proc));
                   1499: }
                   1500:
                   1501: /*
                   1502:  * Do an nfs rename rpc. Called from nfs_rename() and nfs_renameit().
                   1503:  */
                   1504: int
                   1505: nfs_renamerpc(fdvp, fnameptr, fnamelen, tdvp, tnameptr, tnamelen, cred, proc)
                   1506:        struct vnode *fdvp;
                   1507:        char *fnameptr;
                   1508:        int fnamelen;
                   1509:        struct vnode *tdvp;
                   1510:        char *tnameptr;
                   1511:        int tnamelen;
                   1512:        struct ucred *cred;
                   1513:        struct proc *proc;
                   1514: {
                   1515:        u_int32_t *tl;
                   1516:        caddr_t cp;
                   1517:        int32_t t1, t2;
                   1518:        caddr_t bpos, dpos, cp2;
                   1519:        int error = 0, fwccflag = NFSV3_WCCRATTR, twccflag = NFSV3_WCCRATTR;
                   1520:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                   1521:        int v3 = NFS_ISV3(fdvp);
                   1522:
                   1523:        nfsstats.rpccnt[NFSPROC_RENAME]++;
                   1524:        nfsm_reqhead(fdvp, NFSPROC_RENAME,
                   1525:                (NFSX_FH(v3) + NFSX_UNSIGNED)*2 + nfsm_rndup(fnamelen) +
                   1526:                nfsm_rndup(tnamelen));
                   1527:        nfsm_fhtom(fdvp, v3);
                   1528:        nfsm_strtom(fnameptr, fnamelen, NFS_MAXNAMLEN);
                   1529:        nfsm_fhtom(tdvp, v3);
                   1530:        nfsm_strtom(tnameptr, tnamelen, NFS_MAXNAMLEN);
                   1531:        nfsm_request(fdvp, NFSPROC_RENAME, proc, cred);
                   1532:        if (v3) {
                   1533:                nfsm_wcc_data(fdvp, fwccflag);
                   1534:                nfsm_wcc_data(tdvp, twccflag);
                   1535:        }
                   1536:        nfsm_reqdone;
                   1537:        VTONFS(fdvp)->n_flag |= NMODIFIED;
                   1538:        VTONFS(tdvp)->n_flag |= NMODIFIED;
                   1539:        if (!fwccflag)
                   1540:                VTONFS(fdvp)->n_attrstamp = 0;
                   1541:        if (!twccflag)
                   1542:                VTONFS(tdvp)->n_attrstamp = 0;
                   1543:        return (error);
                   1544: }
                   1545:
                   1546: /*
                   1547:  * nfs hard link create call
                   1548:  */
                   1549: int
                   1550: nfs_link(v)
                   1551:        void *v;
                   1552: {
                   1553:        struct vop_link_args *ap = v;
                   1554:        struct vnode *vp = ap->a_vp;
                   1555:        struct vnode *dvp = ap->a_dvp;
                   1556:        struct componentname *cnp = ap->a_cnp;
                   1557:        u_int32_t *tl;
                   1558:        caddr_t cp;
                   1559:        int32_t t1, t2;
                   1560:        caddr_t bpos, dpos, cp2;
                   1561:        int error = 0, wccflag = NFSV3_WCCRATTR, attrflag = 0;
                   1562:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                   1563:        int v3;
                   1564:
                   1565:        if (dvp->v_mount != vp->v_mount) {
                   1566:                pool_put(&namei_pool, cnp->cn_pnbuf);
                   1567:                if (vp == dvp)
                   1568:                        vrele(dvp);
                   1569:                else
                   1570:                        vput(dvp);
                   1571:                return (EXDEV);
                   1572:        }
                   1573:
                   1574:        /*
                   1575:         * Push all writes to the server, so that the attribute cache
                   1576:         * doesn't get "out of sync" with the server.
                   1577:         * XXX There should be a better way!
                   1578:         */
                   1579:        VOP_FSYNC(vp, cnp->cn_cred, MNT_WAIT, cnp->cn_proc);
                   1580:
                   1581:        v3 = NFS_ISV3(vp);
                   1582:        nfsstats.rpccnt[NFSPROC_LINK]++;
                   1583:        nfsm_reqhead(vp, NFSPROC_LINK,
                   1584:                NFSX_FH(v3)*2 + NFSX_UNSIGNED + nfsm_rndup(cnp->cn_namelen));
                   1585:        nfsm_fhtom(vp, v3);
                   1586:        nfsm_fhtom(dvp, v3);
                   1587:        nfsm_strtom(cnp->cn_nameptr, cnp->cn_namelen, NFS_MAXNAMLEN);
                   1588:        nfsm_request(vp, NFSPROC_LINK, cnp->cn_proc, cnp->cn_cred);
                   1589:        if (v3) {
                   1590:                nfsm_postop_attr(vp, attrflag);
                   1591:                nfsm_wcc_data(dvp, wccflag);
                   1592:        }
                   1593:        nfsm_reqdone;
                   1594:        pool_put(&namei_pool, cnp->cn_pnbuf);
                   1595:        VTONFS(dvp)->n_flag |= NMODIFIED;
                   1596:        if (!attrflag)
                   1597:                VTONFS(vp)->n_attrstamp = 0;
                   1598:        if (!wccflag)
                   1599:                VTONFS(dvp)->n_attrstamp = 0;
                   1600:
                   1601:        VN_KNOTE(vp, NOTE_LINK);
                   1602:        VN_KNOTE(dvp, NOTE_WRITE);
                   1603:        vput(dvp);
                   1604:        /*
                   1605:         * Kludge: Map EEXIST => 0 assuming that it is a reply to a retry.
                   1606:         */
                   1607:        if (error == EEXIST)
                   1608:                error = 0;
                   1609:        return (error);
                   1610: }
                   1611:
                   1612: /*
                   1613:  * nfs symbolic link create call
                   1614:  */
                   1615: int
                   1616: nfs_symlink(v)
                   1617:        void *v;
                   1618: {
                   1619:        struct vop_symlink_args *ap = v;
                   1620:        struct vnode *dvp = ap->a_dvp;
                   1621:        struct vattr *vap = ap->a_vap;
                   1622:        struct componentname *cnp = ap->a_cnp;
                   1623:        struct nfsv2_sattr *sp;
                   1624:        u_int32_t *tl;
                   1625:        caddr_t cp;
                   1626:        int32_t t1, t2;
                   1627:        caddr_t bpos, dpos, cp2;
                   1628:        int slen, error = 0, wccflag = NFSV3_WCCRATTR, gotvp;
                   1629:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                   1630:        struct vnode *newvp = (struct vnode *)0;
                   1631:        int v3 = NFS_ISV3(dvp);
                   1632:
                   1633:        nfsstats.rpccnt[NFSPROC_SYMLINK]++;
                   1634:        slen = strlen(ap->a_target);
                   1635:        nfsm_reqhead(dvp, NFSPROC_SYMLINK, NFSX_FH(v3) + 2*NFSX_UNSIGNED +
                   1636:            nfsm_rndup(cnp->cn_namelen) + nfsm_rndup(slen) + NFSX_SATTR(v3));
                   1637:        nfsm_fhtom(dvp, v3);
                   1638:        nfsm_strtom(cnp->cn_nameptr, cnp->cn_namelen, NFS_MAXNAMLEN);
                   1639:        if (v3)
                   1640:                nfsm_v3attrbuild(vap, FALSE);
                   1641:        nfsm_strtom(ap->a_target, slen, NFS_MAXPATHLEN);
                   1642:        if (!v3) {
                   1643:                nfsm_build(sp, struct nfsv2_sattr *, NFSX_V2SATTR);
                   1644:                sp->sa_mode = vtonfsv2_mode(VLNK, vap->va_mode);
                   1645:                sp->sa_uid = nfs_xdrneg1;
                   1646:                sp->sa_gid = nfs_xdrneg1;
                   1647:                sp->sa_size = nfs_xdrneg1;
                   1648:                txdr_nfsv2time(&vap->va_atime, &sp->sa_atime);
                   1649:                txdr_nfsv2time(&vap->va_mtime, &sp->sa_mtime);
                   1650:        }
                   1651:        nfsm_request(dvp, NFSPROC_SYMLINK, cnp->cn_proc, cnp->cn_cred);
                   1652:        if (v3) {
                   1653:                if (!error)
                   1654:                        nfsm_mtofh(dvp, newvp, v3, gotvp);
                   1655:                nfsm_wcc_data(dvp, wccflag);
                   1656:        }
                   1657:        nfsm_reqdone;
                   1658:        if (newvp)
                   1659:                vrele(newvp);
                   1660:        pool_put(&namei_pool, cnp->cn_pnbuf);
                   1661:        VTONFS(dvp)->n_flag |= NMODIFIED;
                   1662:        if (!wccflag)
                   1663:                VTONFS(dvp)->n_attrstamp = 0;
                   1664:        VN_KNOTE(dvp, NOTE_WRITE);
                   1665:        vrele(dvp);
                   1666:        /*
                   1667:         * Kludge: Map EEXIST => 0 assuming that it is a reply to a retry.
                   1668:         */
                   1669:        if (error == EEXIST)
                   1670:                error = 0;
                   1671:        return (error);
                   1672: }
                   1673:
                   1674: /*
                   1675:  * nfs make dir call
                   1676:  */
                   1677: int
                   1678: nfs_mkdir(v)
                   1679:        void *v;
                   1680: {
                   1681:        struct vop_mkdir_args *ap = v;
                   1682:        struct vnode *dvp = ap->a_dvp;
                   1683:        struct vattr *vap = ap->a_vap;
                   1684:        struct componentname *cnp = ap->a_cnp;
                   1685:        struct nfsv2_sattr *sp;
                   1686:        u_int32_t *tl;
                   1687:        caddr_t cp;
                   1688:        int32_t t1, t2;
                   1689:        int len;
                   1690:        struct nfsnode *np = (struct nfsnode *)0;
                   1691:        struct vnode *newvp = (struct vnode *)0;
                   1692:        caddr_t bpos, dpos, cp2;
                   1693:        int error = 0, wccflag = NFSV3_WCCRATTR;
                   1694:        int gotvp = 0;
                   1695:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                   1696:        int v3 = NFS_ISV3(dvp);
                   1697:
                   1698:        len = cnp->cn_namelen;
                   1699:        nfsstats.rpccnt[NFSPROC_MKDIR]++;
                   1700:        nfsm_reqhead(dvp, NFSPROC_MKDIR,
                   1701:          NFSX_FH(v3) + NFSX_UNSIGNED + nfsm_rndup(len) + NFSX_SATTR(v3));
                   1702:        nfsm_fhtom(dvp, v3);
                   1703:        nfsm_strtom(cnp->cn_nameptr, len, NFS_MAXNAMLEN);
                   1704:        if (v3) {
                   1705:                nfsm_v3attrbuild(vap, FALSE);
                   1706:        } else {
                   1707:                nfsm_build(sp, struct nfsv2_sattr *, NFSX_V2SATTR);
                   1708:                sp->sa_mode = vtonfsv2_mode(VDIR, vap->va_mode);
                   1709:                sp->sa_uid = nfs_xdrneg1;
                   1710:                sp->sa_gid = nfs_xdrneg1;
                   1711:                sp->sa_size = nfs_xdrneg1;
                   1712:                txdr_nfsv2time(&vap->va_atime, &sp->sa_atime);
                   1713:                txdr_nfsv2time(&vap->va_mtime, &sp->sa_mtime);
                   1714:        }
                   1715:        nfsm_request(dvp, NFSPROC_MKDIR, cnp->cn_proc, cnp->cn_cred);
                   1716:        if (!error)
                   1717:                nfsm_mtofh(dvp, newvp, v3, gotvp);
                   1718:        if (v3)
                   1719:                nfsm_wcc_data(dvp, wccflag);
                   1720:        nfsm_reqdone;
                   1721:        VTONFS(dvp)->n_flag |= NMODIFIED;
                   1722:        if (!wccflag)
                   1723:                VTONFS(dvp)->n_attrstamp = 0;
                   1724:        /*
                   1725:         * Kludge: Map EEXIST => 0 assuming that you have a reply to a retry
                   1726:         * if we can succeed in looking up the directory.
                   1727:         */
                   1728:        if (error == EEXIST || (!error && !gotvp)) {
                   1729:                if (newvp) {
                   1730:                        vrele(newvp);
                   1731:                        newvp = (struct vnode *)0;
                   1732:                }
                   1733:                error = nfs_lookitup(dvp, cnp->cn_nameptr, len, cnp->cn_cred,
                   1734:                        cnp->cn_proc, &np);
                   1735:                if (!error) {
                   1736:                        newvp = NFSTOV(np);
                   1737:                        if (newvp->v_type != VDIR)
                   1738:                                error = EEXIST;
                   1739:                }
                   1740:        }
                   1741:        if (error) {
                   1742:                if (newvp)
                   1743:                        vrele(newvp);
                   1744:        } else {
                   1745:                VN_KNOTE(dvp, NOTE_WRITE|NOTE_LINK);
                   1746:                *ap->a_vpp = newvp;
                   1747:        }
                   1748:        pool_put(&namei_pool, cnp->cn_pnbuf);
                   1749:        vrele(dvp);
                   1750:        return (error);
                   1751: }
                   1752:
                   1753: /*
                   1754:  * nfs remove directory call
                   1755:  */
                   1756: int
                   1757: nfs_rmdir(v)
                   1758:        void *v;
                   1759: {
                   1760:        struct vop_rmdir_args *ap = v;
                   1761:        struct vnode *vp = ap->a_vp;
                   1762:        struct vnode *dvp = ap->a_dvp;
                   1763:        struct componentname *cnp = ap->a_cnp;
                   1764:        u_int32_t *tl;
                   1765:        caddr_t cp;
                   1766:        int32_t t1, t2;
                   1767:        caddr_t bpos, dpos, cp2;
                   1768:        int error = 0, wccflag = NFSV3_WCCRATTR;
                   1769:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                   1770:        int v3 = NFS_ISV3(dvp);
                   1771:
                   1772:        if (dvp == vp) {
                   1773:                vrele(dvp);
                   1774:                vrele(dvp);
                   1775:                pool_put(&namei_pool, cnp->cn_pnbuf);
                   1776:                return (EINVAL);
                   1777:        }
                   1778:        nfsstats.rpccnt[NFSPROC_RMDIR]++;
                   1779:        nfsm_reqhead(dvp, NFSPROC_RMDIR,
                   1780:                NFSX_FH(v3) + NFSX_UNSIGNED + nfsm_rndup(cnp->cn_namelen));
                   1781:        nfsm_fhtom(dvp, v3);
                   1782:        nfsm_strtom(cnp->cn_nameptr, cnp->cn_namelen, NFS_MAXNAMLEN);
                   1783:        nfsm_request(dvp, NFSPROC_RMDIR, cnp->cn_proc, cnp->cn_cred);
                   1784:        if (v3)
                   1785:                nfsm_wcc_data(dvp, wccflag);
                   1786:        nfsm_reqdone;
                   1787:        pool_put(&namei_pool, cnp->cn_pnbuf);
                   1788:        VTONFS(dvp)->n_flag |= NMODIFIED;
                   1789:        if (!wccflag)
                   1790:                VTONFS(dvp)->n_attrstamp = 0;
                   1791:
                   1792:        VN_KNOTE(dvp, NOTE_WRITE|NOTE_LINK);
                   1793:        VN_KNOTE(vp, NOTE_DELETE);
                   1794:
                   1795:        cache_purge(dvp);
                   1796:        cache_purge(vp);
                   1797:        vrele(vp);
                   1798:        vrele(dvp);
                   1799:        /*
                   1800:         * Kludge: Map ENOENT => 0 assuming that you have a reply to a retry.
                   1801:         */
                   1802:        if (error == ENOENT)
                   1803:                error = 0;
                   1804:        return (error);
                   1805: }
                   1806:
                   1807:
                   1808: /*
                   1809:  * The readdir logic below has a big design bug. It stores the NFS cookie in
                   1810:  * the returned uio->uio_offset but does not store the verifier (it cannot).
                   1811:  * Instead, the code stores the verifier in the nfsnode and applies that
                   1812:  * verifies to all cookies, no matter what verifier was originally with
                   1813:  * the cookie.
                   1814:  *
                   1815:  * From a practical standpoint, this is not a problem since almost all
                   1816:  * NFS servers do not change the validity of cookies across deletes
                   1817:  * and inserts.
                   1818:  */
                   1819:
                   1820: struct nfs_dirent {
                   1821:        u_int32_t cookie[2];
                   1822:        struct dirent dirent;
                   1823: };
                   1824:
                   1825: #define        NFS_DIRHDSIZ    (sizeof (struct nfs_dirent) - (MAXNAMLEN + 1))
                   1826: #define NFS_DIRENT_OVERHEAD  offsetof(struct nfs_dirent, dirent)
                   1827:
                   1828: /*
                   1829:  * nfs readdir call
                   1830:  */
                   1831: int
                   1832: nfs_readdir(v)
                   1833:        void *v;
                   1834: {
                   1835:        struct vop_readdir_args *ap = v;
                   1836:        struct vnode *vp = ap->a_vp;
                   1837:        struct nfsnode *np = VTONFS(vp);
                   1838:        struct uio *uio = ap->a_uio;
                   1839:        int tresid, error;
                   1840:        struct vattr vattr;
                   1841:        u_long *cookies = NULL;
                   1842:        int ncookies = 0, cnt;
                   1843:        u_int64_t  newoff = uio->uio_offset;
                   1844:        struct nfsmount *nmp = VFSTONFS(vp->v_mount);
                   1845:        struct uio readdir_uio;
                   1846:        struct iovec readdir_iovec;
                   1847:        struct proc * p = uio->uio_procp;
                   1848:        int done = 0, eof = 0;
                   1849:        struct ucred *cred = ap->a_cred;
                   1850:        void *data;
                   1851:
                   1852:        if (vp->v_type != VDIR)
                   1853:                return (EPERM);
                   1854:        /*
                   1855:         * First, check for hit on the EOF offset cache
                   1856:         */
                   1857:        if (np->n_direofoffset != 0 &&
                   1858:            uio->uio_offset == np->n_direofoffset) {
                   1859:                if (VOP_GETATTR(vp, &vattr, ap->a_cred, uio->uio_procp) == 0 &&
                   1860:                    np->n_mtime == vattr.va_mtime.tv_sec) {
                   1861:                        nfsstats.direofcache_hits++;
                   1862:                        *ap->a_eofflag = 1;
                   1863:                        return (0);
                   1864:                }
                   1865:        }
                   1866:
                   1867:        if (uio->uio_resid < NFS_FABLKSIZE)
                   1868:                return (EINVAL);
                   1869:
                   1870:        tresid = uio->uio_resid;
                   1871:
                   1872:        if (uio->uio_rw != UIO_READ)
                   1873:                return (EINVAL);
                   1874:
                   1875:        if (ap->a_cookies) {
                   1876:                ncookies = uio->uio_resid / 20;
                   1877:
                   1878:                MALLOC(cookies, u_long *, sizeof(*cookies) * ncookies,
                   1879:                       M_TEMP, M_WAITOK);
                   1880:                *ap->a_ncookies = ncookies;
                   1881:                *ap->a_cookies = cookies;
                   1882:        }
                   1883:
                   1884:        if ((nmp->nm_flag & (NFSMNT_NFSV3 | NFSMNT_GOTFSINFO)) == NFSMNT_NFSV3)
                   1885:                (void)nfs_fsinfo(nmp, vp, cred, p);
                   1886:
                   1887:        cnt = 5;
                   1888:
                   1889:        MALLOC(data, void *, NFS_DIRBLKSIZ, M_TEMP,
                   1890:            M_WAITOK);
                   1891:
                   1892:        do {
                   1893:                struct nfs_dirent *ndp = data;
                   1894:
                   1895:                readdir_iovec.iov_len = NFS_DIRBLKSIZ;
                   1896:                readdir_iovec.iov_base = data;
                   1897:                readdir_uio.uio_offset = newoff;
                   1898:                readdir_uio.uio_iov = &readdir_iovec;
                   1899:                readdir_uio.uio_iovcnt = 1;
                   1900:                readdir_uio.uio_segflg = UIO_SYSSPACE;
                   1901:                readdir_uio.uio_rw = UIO_READ;
                   1902:                readdir_uio.uio_resid = NFS_DIRBLKSIZ;
                   1903:                readdir_uio.uio_procp = curproc;
                   1904:
                   1905:                if (nmp->nm_flag & NFSMNT_RDIRPLUS) {
                   1906:                        error = nfs_readdirplusrpc(vp, &readdir_uio, cred,
                   1907:                            &eof);
                   1908:                        if (error == NFSERR_NOTSUPP)
                   1909:                                nmp->nm_flag &= ~NFSMNT_RDIRPLUS;
                   1910:                }
                   1911:                if ((nmp->nm_flag & NFSMNT_RDIRPLUS) == 0)
                   1912:                        error = nfs_readdirrpc(vp, &readdir_uio, cred, &eof);
                   1913:
                   1914:                if (error == NFSERR_BAD_COOKIE)
                   1915:                        error = EINVAL;
                   1916:
                   1917:                while (error == 0 &&
                   1918:                    (ap->a_cookies == NULL || ncookies != 0) &&
                   1919:                    ndp < (struct nfs_dirent *)readdir_iovec.iov_base) {
                   1920:                        struct dirent *dp = &ndp->dirent;
                   1921:                        int reclen = dp->d_reclen;
                   1922:
                   1923:                        dp->d_reclen -= NFS_DIRENT_OVERHEAD;
                   1924:
                   1925:                        if (uio->uio_resid < dp->d_reclen) {
                   1926:                                eof = 0;
                   1927:                                done = 1;
                   1928:                                break;
                   1929:                        }
                   1930:
                   1931:                        error = uiomove((caddr_t)dp, dp->d_reclen, uio);
                   1932:                        if (error)
                   1933:                                break;
                   1934:
                   1935:                        newoff = fxdr_hyper(&ndp->cookie[0]);
                   1936:
                   1937:                        if (ap->a_cookies != NULL) {
                   1938:                                *cookies = newoff;
                   1939:                                cookies++;
                   1940:                                ncookies--;
                   1941:                        }
                   1942:
                   1943:                        ndp = (struct nfs_dirent *)((u_int8_t *)ndp + reclen);
                   1944:                }
                   1945:        } while (!error && !done && !eof && cnt--);
                   1946:
                   1947:        FREE(data, M_TEMP);
                   1948:        data = NULL;
                   1949:
                   1950:        if (ap->a_cookies) {
                   1951:                if (error) {
                   1952:                        FREE(*ap->a_cookies, M_TEMP);
                   1953:                        *ap->a_cookies = NULL;
                   1954:                        *ap->a_ncookies = 0;
                   1955:                } else {
                   1956:                        *ap->a_ncookies -= ncookies;
                   1957:                }
                   1958:        }
                   1959:
                   1960:        if (!error)
                   1961:                uio->uio_offset = newoff;
                   1962:
                   1963:        if (!error && (eof || uio->uio_resid == tresid)) {
                   1964:                nfsstats.direofcache_misses++;
                   1965:                *ap->a_eofflag = 1;
                   1966:                return (0);
                   1967:        }
                   1968:
                   1969:        *ap->a_eofflag = 0;
                   1970:        return (error);
                   1971: }
                   1972:
                   1973:
                   1974: /*
                   1975:  * The function below stuff the cookies in after the name
                   1976:  */
                   1977:
                   1978: /*
                   1979:  * Readdir rpc call.
                   1980:  */
                   1981: int
                   1982: nfs_readdirrpc(struct vnode *vp,
                   1983:     struct uio *uiop,
                   1984:     struct ucred *cred,
                   1985:     int *end_of_directory)
                   1986: {
                   1987:        int len, left;
                   1988:        struct nfs_dirent *ndp = NULL;
                   1989:        struct dirent *dp = NULL;
                   1990:        u_int32_t *tl;
                   1991:        caddr_t cp;
                   1992:        int32_t t1, t2;
                   1993:        caddr_t bpos, dpos, cp2;
                   1994:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                   1995:        nfsuint64 cookie;
                   1996:        struct nfsmount *nmp = VFSTONFS(vp->v_mount);
                   1997:        struct nfsnode *dnp = VTONFS(vp);
                   1998:        u_quad_t fileno;
                   1999:        int error = 0, tlen, more_dirs = 1, blksiz = 0, bigenough = 1;
                   2000:        int attrflag;
                   2001:        int v3 = NFS_ISV3(vp);
                   2002:
                   2003: #ifndef DIAGNOSTIC
                   2004:        if (uiop->uio_iovcnt != 1 ||
                   2005:                (uiop->uio_resid & (NFS_DIRBLKSIZ - 1)))
                   2006:                panic("nfs readdirrpc bad uio");
                   2007: #endif
                   2008:
                   2009:        txdr_hyper(uiop->uio_offset, &cookie.nfsuquad[0]);
                   2010:
                   2011:        /*
                   2012:         * Loop around doing readdir rpc's of size nm_readdirsize
                   2013:         * truncated to a multiple of NFS_READDIRBLKSIZ.
                   2014:         * The stopping criteria is EOF or buffer full.
                   2015:         */
                   2016:        while (more_dirs && bigenough) {
                   2017:                nfsstats.rpccnt[NFSPROC_READDIR]++;
                   2018:                nfsm_reqhead(vp, NFSPROC_READDIR, NFSX_FH(v3) +
                   2019:                        NFSX_READDIR(v3));
                   2020:                nfsm_fhtom(vp, v3);
                   2021:                if (v3) {
                   2022:                        nfsm_build(tl, u_int32_t *, 5 * NFSX_UNSIGNED);
                   2023:                        *tl++ = cookie.nfsuquad[0];
                   2024:                        *tl++ = cookie.nfsuquad[1];
                   2025:                        if (cookie.nfsuquad[0] == 0 &&
                   2026:                            cookie.nfsuquad[1] == 0) {
                   2027:                                *tl++ = 0;
                   2028:                                *tl++ = 0;
                   2029:                        } else {
                   2030:                                *tl++ = dnp->n_cookieverf.nfsuquad[0];
                   2031:                                *tl++ = dnp->n_cookieverf.nfsuquad[1];
                   2032:                        }
                   2033:                } else {
                   2034:                        nfsm_build(tl, u_int32_t *, 2 * NFSX_UNSIGNED);
                   2035:                        *tl++ = cookie.nfsuquad[1];
                   2036:                }
                   2037:                *tl = txdr_unsigned(nmp->nm_readdirsize);
                   2038:                nfsm_request(vp, NFSPROC_READDIR, uiop->uio_procp, cred);
                   2039:                if (v3) {
                   2040:                        nfsm_postop_attr(vp, attrflag);
                   2041:                        if (!error) {
                   2042:                                nfsm_dissect(tl, u_int32_t *,
                   2043:                                    2 * NFSX_UNSIGNED);
                   2044:                                dnp->n_cookieverf.nfsuquad[0] = *tl++;
                   2045:                                dnp->n_cookieverf.nfsuquad[1] = *tl;
                   2046:                        } else {
                   2047:                                m_freem(mrep);
                   2048:                                goto nfsmout;
                   2049:                        }
                   2050:                }
                   2051:                nfsm_dissect(tl, u_int32_t *, NFSX_UNSIGNED);
                   2052:                more_dirs = fxdr_unsigned(int, *tl);
                   2053:
                   2054:                /* loop thru the dir entries, doctoring them to 4bsd form */
                   2055:                while (more_dirs && bigenough) {
                   2056:                        if (v3) {
                   2057:                                nfsm_dissect(tl, u_int32_t *,
                   2058:                                    3 * NFSX_UNSIGNED);
                   2059:                                fileno = fxdr_hyper(tl);
                   2060:                                len = fxdr_unsigned(int, *(tl + 2));
                   2061:                        } else {
                   2062:                                nfsm_dissect(tl, u_int32_t *,
                   2063:                                    2 * NFSX_UNSIGNED);
                   2064:                                fileno = fxdr_unsigned(u_quad_t, *tl++);
                   2065:                                len = fxdr_unsigned(int, *tl);
                   2066:                        }
                   2067:                        if (len <= 0 || len > NFS_MAXNAMLEN) {
                   2068:                                error = EBADRPC;
                   2069:                                m_freem(mrep);
                   2070:                                goto nfsmout;
                   2071:                        }
                   2072:                        tlen = nfsm_rndup(len + 1);
                   2073:                        left = NFS_READDIRBLKSIZ - blksiz;
                   2074:                        if ((tlen + NFS_DIRHDSIZ) > left) {
                   2075:                                dp->d_reclen += left;
                   2076:                                uiop->uio_iov->iov_base += left;
                   2077:                                uiop->uio_iov->iov_len -= left;
                   2078:                                uiop->uio_resid -= left;
                   2079:                                blksiz = 0;
                   2080:                        }
                   2081:                        if ((tlen + NFS_DIRHDSIZ) > uiop->uio_resid)
                   2082:                                bigenough = 0;
                   2083:                        if (bigenough) {
                   2084:                                ndp = (struct nfs_dirent *)
                   2085:                                    uiop->uio_iov->iov_base;
                   2086:                                dp = &ndp->dirent;
                   2087:                                dp->d_fileno = (int)fileno;
                   2088:                                dp->d_namlen = len;
                   2089:                                dp->d_reclen = tlen + NFS_DIRHDSIZ;
                   2090:                                dp->d_type = DT_UNKNOWN;
                   2091:                                blksiz += dp->d_reclen;
                   2092:                                if (blksiz == NFS_READDIRBLKSIZ)
                   2093:                                        blksiz = 0;
                   2094:                                uiop->uio_resid -= NFS_DIRHDSIZ;
                   2095:                                (char *)uiop->uio_iov->iov_base += NFS_DIRHDSIZ;
                   2096:                                uiop->uio_iov->iov_len -= NFS_DIRHDSIZ;
                   2097:                                nfsm_mtouio(uiop, len);
                   2098:                                cp = uiop->uio_iov->iov_base;
                   2099:                                tlen -= len;
                   2100:                                *cp = '\0';     /* null terminate */
                   2101:                                uiop->uio_iov->iov_base += tlen;
                   2102:                                uiop->uio_iov->iov_len -= tlen;
                   2103:                                uiop->uio_resid -= tlen;
                   2104:                        } else
                   2105:                                nfsm_adv(nfsm_rndup(len));
                   2106:                        if (v3) {
                   2107:                                nfsm_dissect(tl, u_int32_t *,
                   2108:                                    3 * NFSX_UNSIGNED);
                   2109:                        } else {
                   2110:                                nfsm_dissect(tl, u_int32_t *,
                   2111:                                    2 * NFSX_UNSIGNED);
                   2112:                        }
                   2113:                        if (bigenough) {
                   2114:                                if (v3) {
                   2115:                                        ndp->cookie[0] = cookie.nfsuquad[0] =
                   2116:                                            *tl++;
                   2117:                                } else
                   2118:                                        ndp->cookie[0] = 0;
                   2119:
                   2120:                                ndp->cookie[1] = cookie.nfsuquad[1] = *tl++;
                   2121:                        } else if (v3)
                   2122:                                tl += 2;
                   2123:                        else
                   2124:                                tl++;
                   2125:                        more_dirs = fxdr_unsigned(int, *tl);
                   2126:                }
                   2127:                /*
                   2128:                 * If at end of rpc data, get the eof boolean
                   2129:                 */
                   2130:                if (!more_dirs) {
                   2131:                        nfsm_dissect(tl, u_int32_t *, NFSX_UNSIGNED);
                   2132:                        more_dirs = (fxdr_unsigned(int, *tl) == 0);
                   2133:                }
                   2134:                m_freem(mrep);
                   2135:        }
                   2136:        /*
                   2137:         * Fill last record, iff any, out to a multiple of NFS_READDIRBLKSIZ
                   2138:         * by increasing d_reclen for the last record.
                   2139:         */
                   2140:        if (blksiz > 0) {
                   2141:                left = NFS_READDIRBLKSIZ - blksiz;
                   2142:                dp->d_reclen += left;
                   2143:                (char *)uiop->uio_iov->iov_base += left;
                   2144:                uiop->uio_iov->iov_len -= left;
                   2145:                uiop->uio_resid -= left;
                   2146:        }
                   2147:
                   2148:        /*
                   2149:         * We are now either at the end of the directory or have filled the
                   2150:         * block.
                   2151:         */
                   2152:        if (bigenough) {
                   2153:                dnp->n_direofoffset = fxdr_hyper(&cookie.nfsuquad[0]);
                   2154:                if (end_of_directory) *end_of_directory = 1;
                   2155:        } else {
                   2156:                if (uiop->uio_resid > 0)
                   2157:                        printf("EEK! readdirrpc resid > 0\n");
                   2158:        }
                   2159:
                   2160: nfsmout:
                   2161:        return (error);
                   2162: }
                   2163:
                   2164: /*
                   2165:  * NFS V3 readdir plus RPC. Used in place of nfs_readdirrpc().
                   2166:  */
                   2167: int
                   2168: nfs_readdirplusrpc(struct vnode *vp, struct uio *uiop, struct ucred *cred,
                   2169:     int *end_of_directory)
                   2170: {
                   2171:        int len, left;
                   2172:        struct nfs_dirent *ndirp = NULL;
                   2173:        struct dirent *dp = NULL;
                   2174:        u_int32_t *tl;
                   2175:        caddr_t cp;
                   2176:        int32_t t1, t2;
                   2177:        struct vnode *newvp;
                   2178:        caddr_t bpos, dpos, cp2, dpossav1, dpossav2;
                   2179:        struct mbuf *mreq, *mrep, *md, *mb, *mb2, *mdsav1, *mdsav2;
                   2180:        struct nameidata nami, *ndp = &nami;
                   2181:        struct componentname *cnp = &ndp->ni_cnd;
                   2182:        nfsuint64 cookie;
                   2183:        struct nfsmount *nmp = VFSTONFS(vp->v_mount);
                   2184:        struct nfsnode *dnp = VTONFS(vp), *np;
                   2185:        nfsfh_t *fhp;
                   2186:        u_quad_t fileno;
                   2187:        int error = 0, tlen, more_dirs = 1, blksiz = 0, doit, bigenough = 1, i;
                   2188:        int attrflag, fhsize;
                   2189:
                   2190: #ifndef DIAGNOSTIC
                   2191:        if (uiop->uio_iovcnt != 1 ||
                   2192:                (uiop->uio_resid & (NFS_DIRBLKSIZ - 1)))
                   2193:                panic("nfs readdirplusrpc bad uio");
                   2194: #endif
                   2195:        ndp->ni_dvp = vp;
                   2196:        newvp = NULLVP;
                   2197:
                   2198:        txdr_hyper(uiop->uio_offset, &cookie.nfsuquad[0]);
                   2199:
                   2200:         /*
                   2201:         * Loop around doing readdir rpc's of size nm_readdirsize
                   2202:         * truncated to a multiple of NFS_READDIRBLKSIZ.
                   2203:         * The stopping criteria is EOF or buffer full.
                   2204:         */
                   2205:        while (more_dirs && bigenough) {
                   2206:                nfsstats.rpccnt[NFSPROC_READDIRPLUS]++;
                   2207:                nfsm_reqhead(vp, NFSPROC_READDIRPLUS,
                   2208:                        NFSX_FH(1) + 6 * NFSX_UNSIGNED);
                   2209:                nfsm_fhtom(vp, 1);
                   2210:                nfsm_build(tl, u_int32_t *, 6 * NFSX_UNSIGNED);
                   2211:                *tl++ = cookie.nfsuquad[0];
                   2212:                *tl++ = cookie.nfsuquad[1];
                   2213:                if (cookie.nfsuquad[0] == 0 &&
                   2214:                    cookie.nfsuquad[1] == 0) {
                   2215:                        *tl++ = 0;
                   2216:                        *tl++ = 0;
                   2217:                } else {
                   2218:                        *tl++ = dnp->n_cookieverf.nfsuquad[0];
                   2219:                        *tl++ = dnp->n_cookieverf.nfsuquad[1];
                   2220:                }
                   2221:                *tl++ = txdr_unsigned(nmp->nm_readdirsize);
                   2222:                *tl = txdr_unsigned(nmp->nm_rsize);
                   2223:                nfsm_request(vp, NFSPROC_READDIRPLUS, uiop->uio_procp, cred);
                   2224:                nfsm_postop_attr(vp, attrflag);
                   2225:                if (error) {
                   2226:                        m_freem(mrep);
                   2227:                        goto nfsmout;
                   2228:                }
                   2229:                nfsm_dissect(tl, u_int32_t *, 3 * NFSX_UNSIGNED);
                   2230:                dnp->n_cookieverf.nfsuquad[0] = *tl++;
                   2231:                dnp->n_cookieverf.nfsuquad[1] = *tl++;
                   2232:                more_dirs = fxdr_unsigned(int, *tl);
                   2233:
                   2234:                /* loop thru the dir entries, doctoring them to 4bsd form */
                   2235:                while (more_dirs && bigenough) {
                   2236:                        nfsm_dissect(tl, u_int32_t *, 3 * NFSX_UNSIGNED);
                   2237:                        fileno = fxdr_hyper(tl);
                   2238:                        len = fxdr_unsigned(int, *(tl + 2));
                   2239:                        if (len <= 0 || len > NFS_MAXNAMLEN) {
                   2240:                                error = EBADRPC;
                   2241:                                m_freem(mrep);
                   2242:                                goto nfsmout;
                   2243:                        }
                   2244:                        tlen = nfsm_rndup(len + 1);
                   2245:                        left = NFS_READDIRBLKSIZ - blksiz;
                   2246:                        if ((tlen + NFS_DIRHDSIZ) > left) {
                   2247:                                dp->d_reclen += left;
                   2248:                                (char *)uiop->uio_iov->iov_base += left;
                   2249:                                uiop->uio_iov->iov_len -= left;
                   2250:                                uiop->uio_resid -= left;
                   2251:                                blksiz = 0;
                   2252:                        }
                   2253:                        if ((tlen + NFS_DIRHDSIZ) > uiop->uio_resid)
                   2254:                                bigenough = 0;
                   2255:                        if (bigenough) {
                   2256:                                ndirp = (struct nfs_dirent *)
                   2257:                                    uiop->uio_iov->iov_base;
                   2258:                                dp = &ndirp->dirent;
                   2259:                                dp->d_fileno = (int)fileno;
                   2260:                                dp->d_namlen = len;
                   2261:                                dp->d_reclen = tlen + NFS_DIRHDSIZ;
                   2262:                                dp->d_type = DT_UNKNOWN;
                   2263:                                blksiz += dp->d_reclen;
                   2264:                                if (blksiz == NFS_READDIRBLKSIZ)
                   2265:                                        blksiz = 0;
                   2266:                                uiop->uio_resid -= NFS_DIRHDSIZ;
                   2267:                                (char *)uiop->uio_iov->iov_base += NFS_DIRHDSIZ;
                   2268:                                uiop->uio_iov->iov_len -= NFS_DIRHDSIZ;
                   2269:                                cnp->cn_nameptr = uiop->uio_iov->iov_base;
                   2270:                                cnp->cn_namelen = len;
                   2271:                                nfsm_mtouio(uiop, len);
                   2272:                                cp = uiop->uio_iov->iov_base;
                   2273:                                tlen -= len;
                   2274:                                *cp = '\0';
                   2275:                                uiop->uio_iov->iov_base += tlen;
                   2276:                                uiop->uio_iov->iov_len -= tlen;
                   2277:                                uiop->uio_resid -= tlen;
                   2278:                        } else
                   2279:                                nfsm_adv(nfsm_rndup(len));
                   2280:                        nfsm_dissect(tl, u_int32_t *, 3 * NFSX_UNSIGNED);
                   2281:                        if (bigenough) {
                   2282:                                ndirp->cookie[0] = cookie.nfsuquad[0] = *tl++;
                   2283:                                ndirp->cookie[1] = cookie.nfsuquad[1] = *tl++;
                   2284:                        } else
                   2285:                                tl += 2;
                   2286:
                   2287:                        /*
                   2288:                         * Since the attributes are before the file handle
                   2289:                         * (sigh), we must skip over the attributes and then
                   2290:                         * come back and get them.
                   2291:                         */
                   2292:                        attrflag = fxdr_unsigned(int, *tl);
                   2293:                        if (attrflag) {
                   2294:                            dpossav1 = dpos;
                   2295:                            mdsav1 = md;
                   2296:                            nfsm_adv(NFSX_V3FATTR);
                   2297:                            nfsm_dissect(tl, u_int32_t *, NFSX_UNSIGNED);
                   2298:                            doit = fxdr_unsigned(int, *tl);
                   2299:                            if (doit) {
                   2300:                                nfsm_getfh(fhp, fhsize, 1);
                   2301:                                if (NFS_CMPFH(dnp, fhp, fhsize)) {
                   2302:                                    VREF(vp);
                   2303:                                    newvp = vp;
                   2304:                                    np = dnp;
                   2305:                                } else {
                   2306:                                    error = nfs_nget(vp->v_mount, fhp,
                   2307:                                        fhsize, &np);
                   2308:                                    if (error)
                   2309:                                        doit = 0;
                   2310:                                    else
                   2311:                                        newvp = NFSTOV(np);
                   2312:                                }
                   2313:                            }
                   2314:                            if (doit && bigenough) {
                   2315:                                dpossav2 = dpos;
                   2316:                                dpos = dpossav1;
                   2317:                                mdsav2 = md;
                   2318:                                md = mdsav1;
                   2319:                                nfsm_loadattr(newvp, (struct vattr *)0);
                   2320:                                dpos = dpossav2;
                   2321:                                md = mdsav2;
                   2322:                                dp->d_type =
                   2323:                                    IFTODT(VTTOIF(np->n_vattr.va_type));
                   2324:                                if (cnp->cn_namelen <= NCHNAMLEN) {
                   2325:                                        ndp->ni_vp = newvp;
                   2326:                                        cnp->cn_hash =
                   2327:                                            hash32_str(cnp->cn_nameptr,
                   2328:                                                HASHINIT);
                   2329:                                        cache_enter(ndp->ni_dvp, ndp->ni_vp,
                   2330:                                            cnp);
                   2331:                                }
                   2332:                            }
                   2333:                        } else {
                   2334:                            /* Just skip over the file handle */
                   2335:                            nfsm_dissect(tl, u_int32_t *, NFSX_UNSIGNED);
                   2336:                            i = fxdr_unsigned(int, *tl);
                   2337:                            nfsm_adv(nfsm_rndup(i));
                   2338:                        }
                   2339:                        if (newvp != NULLVP) {
                   2340:                            vrele(newvp);
                   2341:                            newvp = NULLVP;
                   2342:                        }
                   2343:                        nfsm_dissect(tl, u_int32_t *, NFSX_UNSIGNED);
                   2344:                        more_dirs = fxdr_unsigned(int, *tl);
                   2345:                }
                   2346:                /*
                   2347:                 * If at end of rpc data, get the eof boolean
                   2348:                 */
                   2349:                if (!more_dirs) {
                   2350:                        nfsm_dissect(tl, u_int32_t *, NFSX_UNSIGNED);
                   2351:                        more_dirs = (fxdr_unsigned(int, *tl) == 0);
                   2352:                }
                   2353:                m_freem(mrep);
                   2354:        }
                   2355:        /*
                   2356:         * Fill last record, iff any, out to a multiple of NFS_READDIRBLKSIZ
                   2357:         * by increasing d_reclen for the last record.
                   2358:         */
                   2359:        if (blksiz > 0) {
                   2360:                left = NFS_READDIRBLKSIZ - blksiz;
                   2361:                dp->d_reclen += left;
                   2362:                (char *)uiop->uio_iov->iov_base += left;
                   2363:                uiop->uio_iov->iov_len -= left;
                   2364:                uiop->uio_resid -= left;
                   2365:        }
                   2366:
                   2367:        /*
                   2368:         * We are now either at the end of the directory or have filled the
                   2369:         * block.
                   2370:         */
                   2371:        if (bigenough) {
                   2372:                dnp->n_direofoffset = fxdr_hyper(&cookie.nfsuquad[0]);
                   2373:                if (end_of_directory) *end_of_directory = 1;
                   2374:        } else {
                   2375:                if (uiop->uio_resid > 0)
                   2376:                        printf("EEK! readdirplusrpc resid > 0\n");
                   2377:        }
                   2378:
                   2379: nfsmout:
                   2380:        if (newvp != NULLVP)
                   2381:                vrele(newvp);
                   2382:        return (error);
                   2383: }
                   2384:
                   2385: /*
                   2386:  * Silly rename. To make the NFS filesystem that is stateless look a little
                   2387:  * more like the "ufs" a remove of an active vnode is translated to a rename
                   2388:  * to a funny looking filename that is removed by nfs_inactive on the
                   2389:  * nfsnode. There is the potential for another process on a different client
                   2390:  * to create the same funny name between the nfs_lookitup() fails and the
                   2391:  * nfs_rename() completes, but...
                   2392:  */
                   2393: int
                   2394: nfs_sillyrename(dvp, vp, cnp)
                   2395:        struct vnode *dvp, *vp;
                   2396:        struct componentname *cnp;
                   2397: {
                   2398:        struct sillyrename *sp;
                   2399:        struct nfsnode *np;
                   2400:        int error;
                   2401:
                   2402:        cache_purge(dvp);
                   2403:        np = VTONFS(vp);
                   2404:        MALLOC(sp, struct sillyrename *, sizeof (struct sillyrename),
                   2405:                M_NFSREQ, M_WAITOK);
                   2406:        sp->s_cred = crdup(cnp->cn_cred);
                   2407:        sp->s_dvp = dvp;
                   2408:        VREF(dvp);
                   2409:
                   2410:        if (vp->v_type == VDIR) {
                   2411: #ifdef DIAGNOSTIC
                   2412:                printf("nfs: sillyrename dir\n");
                   2413: #endif
                   2414:                error = EINVAL;
                   2415:                goto bad;
                   2416:        }
                   2417:
                   2418:        /* Fudge together a funny name */
                   2419:        sp->s_namlen = snprintf(sp->s_name, sizeof sp->s_name,
                   2420:            ".nfsA%05x4.4", cnp->cn_proc->p_pid);
                   2421:        if (sp->s_namlen > sizeof sp->s_name)
                   2422:                sp->s_namlen = strlen(sp->s_name);
                   2423:
                   2424:        /* Try lookitups until we get one that isn't there */
                   2425:        while (nfs_lookitup(dvp, sp->s_name, sp->s_namlen, sp->s_cred,
                   2426:                cnp->cn_proc, (struct nfsnode **)0) == 0) {
                   2427:                sp->s_name[4]++;
                   2428:                if (sp->s_name[4] > 'z') {
                   2429:                        error = EINVAL;
                   2430:                        goto bad;
                   2431:                }
                   2432:        }
                   2433:        error = nfs_renameit(dvp, cnp, sp);
                   2434:        if (error)
                   2435:                goto bad;
                   2436:        error = nfs_lookitup(dvp, sp->s_name, sp->s_namlen, sp->s_cred,
                   2437:                cnp->cn_proc, &np);
                   2438:        np->n_sillyrename = sp;
                   2439:        return (0);
                   2440: bad:
                   2441:        vrele(sp->s_dvp);
                   2442:        crfree(sp->s_cred);
                   2443:        FREE((caddr_t)sp, M_NFSREQ);
                   2444:        return (error);
                   2445: }
                   2446:
                   2447: /*
                   2448:  * Look up a file name and optionally either update the file handle or
                   2449:  * allocate an nfsnode, depending on the value of npp.
                   2450:  * npp == NULL --> just do the lookup
                   2451:  * *npp == NULL --> allocate a new nfsnode and make sure attributes are
                   2452:  *                     handled too
                   2453:  * *npp != NULL --> update the file handle in the vnode
                   2454:  */
                   2455: int
                   2456: nfs_lookitup(dvp, name, len, cred, procp, npp)
                   2457:        struct vnode *dvp;
                   2458:        char *name;
                   2459:        int len;
                   2460:        struct ucred *cred;
                   2461:        struct proc *procp;
                   2462:        struct nfsnode **npp;
                   2463: {
                   2464:        u_int32_t *tl;
                   2465:        caddr_t cp;
                   2466:        int32_t t1, t2;
                   2467:        struct vnode *newvp = (struct vnode *)0;
                   2468:        struct nfsnode *np, *dnp = VTONFS(dvp);
                   2469:        caddr_t bpos, dpos, cp2;
                   2470:        int error = 0, fhlen, attrflag;
                   2471:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                   2472:        nfsfh_t *nfhp;
                   2473:        int v3 = NFS_ISV3(dvp);
                   2474:
                   2475:        nfsstats.rpccnt[NFSPROC_LOOKUP]++;
                   2476:        nfsm_reqhead(dvp, NFSPROC_LOOKUP,
                   2477:                NFSX_FH(v3) + NFSX_UNSIGNED + nfsm_rndup(len));
                   2478:        nfsm_fhtom(dvp, v3);
                   2479:        nfsm_strtom(name, len, NFS_MAXNAMLEN);
                   2480:        nfsm_request(dvp, NFSPROC_LOOKUP, procp, cred);
                   2481:        if (npp && !error) {
                   2482:                nfsm_getfh(nfhp, fhlen, v3);
                   2483:                if (*npp) {
                   2484:                    np = *npp;
                   2485:                    if (np->n_fhsize > NFS_SMALLFH && fhlen <= NFS_SMALLFH) {
                   2486:                        free((caddr_t)np->n_fhp, M_NFSBIGFH);
                   2487:                        np->n_fhp = &np->n_fh;
                   2488:                    } else if (np->n_fhsize <= NFS_SMALLFH && fhlen>NFS_SMALLFH)
                   2489:                        np->n_fhp =(nfsfh_t *)malloc(fhlen,M_NFSBIGFH,M_WAITOK);
                   2490:                    bcopy((caddr_t)nfhp, (caddr_t)np->n_fhp, fhlen);
                   2491:                    np->n_fhsize = fhlen;
                   2492:                    newvp = NFSTOV(np);
                   2493:                } else if (NFS_CMPFH(dnp, nfhp, fhlen)) {
                   2494:                    VREF(dvp);
                   2495:                    newvp = dvp;
                   2496:                } else {
                   2497:                    error = nfs_nget(dvp->v_mount, nfhp, fhlen, &np);
                   2498:                    if (error) {
                   2499:                        m_freem(mrep);
                   2500:                        return (error);
                   2501:                    }
                   2502:                    newvp = NFSTOV(np);
                   2503:                }
                   2504:                if (v3) {
                   2505:                        nfsm_postop_attr(newvp, attrflag);
                   2506:                        if (!attrflag && *npp == NULL) {
                   2507:                                m_freem(mrep);
                   2508:                                vrele(newvp);
                   2509:                                return (ENOENT);
                   2510:                        }
                   2511:                } else
                   2512:                        nfsm_loadattr(newvp, (struct vattr *)0);
                   2513:        }
                   2514:        nfsm_reqdone;
                   2515:        if (npp && *npp == NULL) {
                   2516:                if (error) {
                   2517:                        if (newvp)
                   2518:                                vrele(newvp);
                   2519:                } else
                   2520:                        *npp = np;
                   2521:        }
                   2522:        return (error);
                   2523: }
                   2524:
                   2525: /*
                   2526:  * Nfs Version 3 commit rpc
                   2527:  */
                   2528: int
                   2529: nfs_commit(vp, offset, cnt, procp)
                   2530:        struct vnode *vp;
                   2531:        u_quad_t offset;
                   2532:        int cnt;
                   2533:        struct proc *procp;
                   2534: {
                   2535:        caddr_t cp;
                   2536:        u_int32_t *tl;
                   2537:        int32_t t1, t2;
                   2538:        struct nfsmount *nmp = VFSTONFS(vp->v_mount);
                   2539:        caddr_t bpos, dpos, cp2;
                   2540:        int error = 0, wccflag = NFSV3_WCCRATTR;
                   2541:        struct mbuf *mreq, *mrep, *md, *mb, *mb2;
                   2542:
                   2543:        if ((nmp->nm_flag & NFSMNT_HASWRITEVERF) == 0)
                   2544:                return (0);
                   2545:        nfsstats.rpccnt[NFSPROC_COMMIT]++;
                   2546:        nfsm_reqhead(vp, NFSPROC_COMMIT, NFSX_FH(1));
                   2547:        nfsm_fhtom(vp, 1);
                   2548:        nfsm_build(tl, u_int32_t *, 3 * NFSX_UNSIGNED);
                   2549:        txdr_hyper(offset, tl);
                   2550:        tl += 2;
                   2551:        *tl = txdr_unsigned(cnt);
                   2552:        nfsm_request(vp, NFSPROC_COMMIT, procp, VTONFS(vp)->n_wcred);
                   2553:        nfsm_wcc_data(vp, wccflag);
                   2554:        if (!error) {
                   2555:                nfsm_dissect(tl, u_int32_t *, NFSX_V3WRITEVERF);
                   2556:                if (bcmp((caddr_t)nmp->nm_verf, (caddr_t)tl,
                   2557:                        NFSX_V3WRITEVERF)) {
                   2558:                        bcopy((caddr_t)tl, (caddr_t)nmp->nm_verf,
                   2559:                                NFSX_V3WRITEVERF);
                   2560:                        error = NFSERR_STALEWRITEVERF;
                   2561:                }
                   2562:        }
                   2563:        nfsm_reqdone;
                   2564:        return (error);
                   2565: }
                   2566:
                   2567: /*
                   2568:  * Kludge City..
                   2569:  * - make nfs_bmap() essentially a no-op that does no translation
                   2570:  * - do nfs_strategy() by doing I/O with nfs_readrpc/nfs_writerpc
                   2571:  *   (Maybe I could use the process's page mapping, but I was concerned that
                   2572:  *    Kernel Write might not be enabled and also figured copyout() would do
                   2573:  *    a lot more work than bcopy() and also it currently happens in the
                   2574:  *    context of the swapper process (2).
                   2575:  */
                   2576: int
                   2577: nfs_bmap(v)
                   2578:        void *v;
                   2579: {
                   2580:        struct vop_bmap_args *ap = v;
                   2581:        struct vnode *vp = ap->a_vp;
                   2582:
                   2583:        if (ap->a_vpp != NULL)
                   2584:                *ap->a_vpp = vp;
                   2585:        if (ap->a_bnp != NULL)
                   2586:                *ap->a_bnp = ap->a_bn * btodb(vp->v_mount->mnt_stat.f_iosize);
                   2587:        return (0);
                   2588: }
                   2589:
                   2590: /*
                   2591:  * Strategy routine.
                   2592:  * For async requests when nfsiod(s) are running, queue the request by
                   2593:  * calling nfs_asyncio(), otherwise just all nfs_doio() to do the
                   2594:  * request.
                   2595:  */
                   2596: int
                   2597: nfs_strategy(v)
                   2598:        void *v;
                   2599: {
                   2600:        struct vop_strategy_args *ap = v;
                   2601:        struct buf *bp = ap->a_bp;
                   2602:        struct proc *p;
                   2603:        int error = 0;
                   2604:
                   2605:        if ((bp->b_flags & (B_PHYS|B_ASYNC)) == (B_PHYS|B_ASYNC))
                   2606:                panic("nfs physio/async");
                   2607:        if (bp->b_flags & B_ASYNC)
                   2608:                p = NULL;
                   2609:        else
                   2610:                p = curproc;    /* XXX */
                   2611:        /*
                   2612:         * If the op is asynchronous and an i/o daemon is waiting
                   2613:         * queue the request, wake it up and wait for completion
                   2614:         * otherwise just do it ourselves.
                   2615:         */
                   2616:        if ((bp->b_flags & B_ASYNC) == 0 || nfs_asyncio(bp))
                   2617:                error = nfs_doio(bp, p);
                   2618:        return (error);
                   2619: }
                   2620:
                   2621: /*
                   2622:  * fsync vnode op. Just call nfs_flush() with commit == 1.
                   2623:  */
                   2624: /* ARGSUSED */
                   2625: int
                   2626: nfs_fsync(v)
                   2627:        void *v;
                   2628: {
                   2629:        struct vop_fsync_args *ap = v;
                   2630:
                   2631:        return (nfs_flush(ap->a_vp, ap->a_cred, ap->a_waitfor, ap->a_p, 1));
                   2632: }
                   2633:
                   2634: /*
                   2635:  * Flush all the blocks associated with a vnode.
                   2636:  *     Walk through the buffer pool and push any dirty pages
                   2637:  *     associated with the vnode.
                   2638:  */
                   2639: int
                   2640: nfs_flush(vp, cred, waitfor, p, commit)
                   2641:        struct vnode *vp;
                   2642:        struct ucred *cred;
                   2643:        int waitfor;
                   2644:        struct proc *p;
                   2645:        int commit;
                   2646: {
                   2647:        struct nfsnode *np = VTONFS(vp);
                   2648:        struct buf *bp;
                   2649:        int i;
                   2650:        struct buf *nbp;
                   2651:        struct nfsmount *nmp = VFSTONFS(vp->v_mount);
                   2652:        int s, error = 0, slptimeo = 0, slpflag = 0, retv, bvecpos;
                   2653:        int passone = 1;
                   2654:        u_quad_t off = (u_quad_t)-1, endoff = 0, toff;
                   2655: #ifndef NFS_COMMITBVECSIZ
                   2656: #define NFS_COMMITBVECSIZ      20
                   2657: #endif
                   2658:        struct buf *bvec[NFS_COMMITBVECSIZ];
                   2659:
                   2660:        if (nmp->nm_flag & NFSMNT_INT)
                   2661:                slpflag = PCATCH;
                   2662:        if (!commit)
                   2663:                passone = 0;
                   2664:        /*
                   2665:         * A b_flags == (B_DELWRI | B_NEEDCOMMIT) block has been written to the
                   2666:         * server, but nas not been committed to stable storage on the server
                   2667:         * yet. On the first pass, the byte range is worked out and the commit
                   2668:         * rpc is done. On the second pass, nfs_writebp() is called to do the
                   2669:         * job.
                   2670:         */
                   2671: again:
                   2672:        bvecpos = 0;
                   2673:        if (NFS_ISV3(vp) && commit) {
                   2674:                s = splbio();
                   2675:                for (bp = LIST_FIRST(&vp->v_dirtyblkhd); bp != NULL; bp = nbp) {
                   2676:                        nbp = LIST_NEXT(bp, b_vnbufs);
                   2677:                        if (bvecpos >= NFS_COMMITBVECSIZ)
                   2678:                                break;
                   2679:                        if ((bp->b_flags & (B_BUSY | B_DELWRI | B_NEEDCOMMIT))
                   2680:                                != (B_DELWRI | B_NEEDCOMMIT))
                   2681:                                continue;
                   2682:                        bremfree(bp);
                   2683:                        bp->b_flags |= (B_BUSY | B_WRITEINPROG);
                   2684:                        /*
                   2685:                         * A list of these buffers is kept so that the
                   2686:                         * second loop knows which buffers have actually
                   2687:                         * been committed. This is necessary, since there
                   2688:                         * may be a race between the commit rpc and new
                   2689:                         * uncommitted writes on the file.
                   2690:                         */
                   2691:                        bvec[bvecpos++] = bp;
                   2692:                        toff = ((u_quad_t)bp->b_blkno) * DEV_BSIZE +
                   2693:                                bp->b_dirtyoff;
                   2694:                        if (toff < off)
                   2695:                                off = toff;
                   2696:                        toff += (u_quad_t)(bp->b_dirtyend - bp->b_dirtyoff);
                   2697:                        if (toff > endoff)
                   2698:                                endoff = toff;
                   2699:                }
                   2700:                splx(s);
                   2701:        }
                   2702:        if (bvecpos > 0) {
                   2703:                /*
                   2704:                 * Commit data on the server, as required.
                   2705:                 */
                   2706:                retv = nfs_commit(vp, off, (int)(endoff - off), p);
                   2707:                if (retv == NFSERR_STALEWRITEVERF)
                   2708:                        nfs_clearcommit(vp->v_mount);
                   2709:                /*
                   2710:                 * Now, either mark the blocks I/O done or mark the
                   2711:                 * blocks dirty, depending on whether the commit
                   2712:                 * succeeded.
                   2713:                 */
                   2714:                for (i = 0; i < bvecpos; i++) {
                   2715:                        bp = bvec[i];
                   2716:                        bp->b_flags &= ~(B_NEEDCOMMIT | B_WRITEINPROG);
                   2717:                        if (retv)
                   2718:                            brelse(bp);
                   2719:                        else {
                   2720:                            s = splbio();
                   2721:                            buf_undirty(bp);
                   2722:                            vp->v_numoutput++;
                   2723:                            bp->b_flags |= B_ASYNC;
                   2724:                            bp->b_flags &= ~(B_READ|B_DONE|B_ERROR);
                   2725:                            bp->b_dirtyoff = bp->b_dirtyend = 0;
                   2726:                            biodone(bp);
                   2727:                            splx(s);
                   2728:                        }
                   2729:                }
                   2730:        }
                   2731:
                   2732:        /*
                   2733:         * Start/do any write(s) that are required.
                   2734:         */
                   2735: loop:
                   2736:        s = splbio();
                   2737:        for (bp = LIST_FIRST(&vp->v_dirtyblkhd); bp != NULL; bp = nbp) {
                   2738:                nbp = LIST_NEXT(bp, b_vnbufs);
                   2739:                if (bp->b_flags & B_BUSY) {
                   2740:                        if (waitfor != MNT_WAIT || passone)
                   2741:                                continue;
                   2742:                        bp->b_flags |= B_WANTED;
                   2743:                        error = tsleep((caddr_t)bp, slpflag | (PRIBIO + 1),
                   2744:                                "nfsfsync", slptimeo);
                   2745:                        splx(s);
                   2746:                        if (error) {
                   2747:                            if (nfs_sigintr(nmp, (struct nfsreq *)0, p))
                   2748:                                return (EINTR);
                   2749:                            if (slpflag == PCATCH) {
                   2750:                                slpflag = 0;
                   2751:                                slptimeo = 2 * hz;
                   2752:                            }
                   2753:                        }
                   2754:                        goto loop;
                   2755:                }
                   2756:                if ((bp->b_flags & B_DELWRI) == 0)
                   2757:                        panic("nfs_fsync: not dirty");
                   2758:                if ((passone || !commit) && (bp->b_flags & B_NEEDCOMMIT))
                   2759:                        continue;
                   2760:                bremfree(bp);
                   2761:                if (passone || !commit)
                   2762:                    bp->b_flags |= (B_BUSY|B_ASYNC);
                   2763:                else
                   2764:                    bp->b_flags |= (B_BUSY|B_ASYNC|B_WRITEINPROG|B_NEEDCOMMIT);
                   2765:                splx(s);
                   2766:                VOP_BWRITE(bp);
                   2767:                goto loop;
                   2768:        }
                   2769:        splx(s);
                   2770:        if (passone) {
                   2771:                passone = 0;
                   2772:                goto again;
                   2773:        }
                   2774:        if (waitfor == MNT_WAIT) {
                   2775:  loop2:
                   2776:                s = splbio();
                   2777:                error = vwaitforio(vp, slpflag, "nfs_fsync", slptimeo);
                   2778:                splx(s);
                   2779:                if (error) {
                   2780:                        if (nfs_sigintr(nmp, (struct nfsreq *)0, p))
                   2781:                                return (EINTR);
                   2782:                        if (slpflag == PCATCH) {
                   2783:                                slpflag = 0;
                   2784:                                slptimeo = 2 * hz;
                   2785:                        }
                   2786:                        goto loop2;
                   2787:                }
                   2788:
                   2789:                if (LIST_FIRST(&vp->v_dirtyblkhd) && commit) {
                   2790: #if 0
                   2791:                        vprint("nfs_fsync: dirty", vp);
                   2792: #endif
                   2793:                        goto loop;
                   2794:                }
                   2795:        }
                   2796:        if (np->n_flag & NWRITEERR) {
                   2797:                error = np->n_error;
                   2798:                np->n_flag &= ~NWRITEERR;
                   2799:        }
                   2800:        return (error);
                   2801: }
                   2802:
                   2803: /*
                   2804:  * Return POSIX pathconf information applicable to nfs.
                   2805:  *
                   2806:  * The NFS V2 protocol doesn't support this, so just return EINVAL
                   2807:  * for V2.
                   2808:  */
                   2809: /* ARGSUSED */
                   2810: int
                   2811: nfs_pathconf(v)
                   2812:        void *v;
                   2813: {
                   2814: #if 0
                   2815:        struct vop_pathconf_args *ap = v;
                   2816: #endif
                   2817:
                   2818:        return (EINVAL);
                   2819: }
                   2820:
                   2821: /*
                   2822:  * NFS advisory byte-level locks.
                   2823:  */
                   2824: int
                   2825: nfs_advlock(v)
                   2826:        void *v;
                   2827: {
                   2828:        struct vop_advlock_args *ap = v;
                   2829:        struct nfsnode *np = VTONFS(ap->a_vp);
                   2830:
                   2831:        return (lf_advlock(&np->n_lockf, np->n_size, ap->a_id, ap->a_op,
                   2832:            ap->a_fl, ap->a_flags));
                   2833: }
                   2834:
                   2835: /*
                   2836:  * Print out the contents of an nfsnode.
                   2837:  */
                   2838: int
                   2839: nfs_print(v)
                   2840:        void *v;
                   2841: {
                   2842:        struct vop_print_args *ap = v;
                   2843:        struct vnode *vp = ap->a_vp;
                   2844:        struct nfsnode *np = VTONFS(vp);
                   2845:
                   2846:        printf("tag VT_NFS, fileid %ld fsid 0x%lx",
                   2847:                np->n_vattr.va_fileid, np->n_vattr.va_fsid);
                   2848: #ifdef FIFO
                   2849:        if (vp->v_type == VFIFO)
                   2850:                fifo_printinfo(vp);
                   2851: #endif
                   2852:        printf("\n");
                   2853:        return (0);
                   2854: }
                   2855:
                   2856: /*
                   2857:  * Just call nfs_writebp() with the force argument set to 1.
                   2858:  */
                   2859: int
                   2860: nfs_bwrite(v)
                   2861:        void *v;
                   2862: {
                   2863:        struct vop_bwrite_args *ap = v;
                   2864:
                   2865:        return (nfs_writebp(ap->a_bp, 1));
                   2866: }
                   2867:
                   2868: /*
                   2869:  * This is a clone of vop_generic_bwrite(), except that B_WRITEINPROG isn't set unless
                   2870:  * the force flag is one and it also handles the B_NEEDCOMMIT flag.
                   2871:  */
                   2872: int
                   2873: nfs_writebp(bp, force)
                   2874:        struct buf *bp;
                   2875:        int force;
                   2876: {
                   2877:        int oldflags = bp->b_flags, retv = 1;
                   2878:        struct proc *p = curproc;       /* XXX */
                   2879:        off_t off;
                   2880:        size_t cnt;
                   2881:        int   s;
                   2882:        struct vnode *vp;
                   2883:        struct nfsnode *np;
                   2884:
                   2885:        if(!(bp->b_flags & B_BUSY))
                   2886:                panic("bwrite: buffer is not busy???");
                   2887:
                   2888:        vp = bp->b_vp;
                   2889:        np = VTONFS(vp);
                   2890:
                   2891:        bp->b_flags &= ~(B_READ|B_DONE|B_ERROR);
                   2892:
                   2893:        s = splbio();
                   2894:        buf_undirty(bp);
                   2895:
                   2896:        if ((oldflags & B_ASYNC) && !(oldflags & B_DELWRI) && p)
                   2897:                ++p->p_stats->p_ru.ru_oublock;
                   2898:
                   2899:        bp->b_vp->v_numoutput++;
                   2900:        splx(s);
                   2901:
                   2902:        /*
                   2903:         * If B_NEEDCOMMIT is set, a commit rpc may do the trick. If not
                   2904:         * an actual write will have to be scheduled via. VOP_STRATEGY().
                   2905:         * If B_WRITEINPROG is already set, then push it with a write anyhow.
                   2906:         */
                   2907:        if ((oldflags & (B_NEEDCOMMIT | B_WRITEINPROG)) == B_NEEDCOMMIT) {
                   2908:                off = ((u_quad_t)bp->b_blkno) * DEV_BSIZE + bp->b_dirtyoff;
                   2909:                cnt = bp->b_dirtyend - bp->b_dirtyoff;
                   2910:
                   2911:                rw_enter_write(&np->n_commitlock);
                   2912:                if (!(bp->b_flags & B_NEEDCOMMIT)) {
                   2913:                        rw_exit_write(&np->n_commitlock);
                   2914:                        return (0);
                   2915:                }
                   2916:
                   2917:                /*
                   2918:                 * If it's already been commited by somebody else,
                   2919:                 * bail.
                   2920:                 */
                   2921:                if (!nfs_in_committed_range(vp, bp)) {
                   2922:                        int pushedrange = 0;
                   2923:                        /*
                   2924:                         * Since we're going to do this, push as much
                   2925:                         * as we can.
                   2926:                         */
                   2927:
                   2928:                        if (nfs_in_tobecommitted_range(vp, bp)) {
                   2929:                                pushedrange = 1;
                   2930:                                off = np->n_pushlo;
                   2931:                                cnt = np->n_pushhi - np->n_pushlo;
                   2932:                        }
                   2933:
                   2934:                        bp->b_flags |= B_WRITEINPROG;
                   2935:                        retv = nfs_commit(bp->b_vp, off, cnt, bp->b_proc);
                   2936:                        bp->b_flags &= ~B_WRITEINPROG;
                   2937:
                   2938:                        if (retv == 0) {
                   2939:                                if (pushedrange)
                   2940:                                        nfs_merge_commit_ranges(vp);
                   2941:                                else
                   2942:                                        nfs_add_committed_range(vp, bp);
                   2943:                        }
                   2944:                } else
                   2945:                        retv = 0; /* It has already been commited. */
                   2946:
                   2947:                rw_exit_write(&np->n_commitlock);
                   2948:                if (!retv) {
                   2949:                        bp->b_dirtyoff = bp->b_dirtyend = 0;
                   2950:                        bp->b_flags &= ~B_NEEDCOMMIT;
                   2951:                        s = splbio();
                   2952:                        biodone(bp);
                   2953:                        splx(s);
                   2954:                } else if (retv == NFSERR_STALEWRITEVERF)
                   2955:                        nfs_clearcommit(bp->b_vp->v_mount);
                   2956:        }
                   2957:        if (retv) {
                   2958:                if (force)
                   2959:                        bp->b_flags |= B_WRITEINPROG;
                   2960:                VOP_STRATEGY(bp);
                   2961:        }
                   2962:
                   2963:        if( (oldflags & B_ASYNC) == 0) {
                   2964:                int rtval = biowait(bp);
                   2965:                if (!(oldflags & B_DELWRI) && p) {
                   2966:                        ++p->p_stats->p_ru.ru_oublock;
                   2967:                }
                   2968:                brelse(bp);
                   2969:                return (rtval);
                   2970:        }
                   2971:
                   2972:        return (0);
                   2973: }
                   2974:
                   2975: /*
                   2976:  * nfs special file access vnode op.
                   2977:  * Essentially just get vattr and then imitate iaccess() since the device is
                   2978:  * local to the client.
                   2979:  */
                   2980: int
                   2981: nfsspec_access(v)
                   2982:        void *v;
                   2983: {
                   2984:        struct vop_access_args *ap = v;
                   2985:        struct vattr va;
                   2986:        struct vnode *vp = ap->a_vp;
                   2987:        int error;
                   2988:
                   2989:         /*
                   2990:         * Disallow write attempts on filesystems mounted read-only;
                   2991:         * unless the file is a socket, fifo, or a block or character
                   2992:         * device resident on the filesystem.
                   2993:         */
                   2994:        if ((ap->a_mode & VWRITE) && (vp->v_mount->mnt_flag & MNT_RDONLY)) {
                   2995:                switch (vp->v_type) {
                   2996:                case VREG:
                   2997:                case VDIR:
                   2998:                case VLNK:
                   2999:                        return (EROFS);
                   3000:                default:
                   3001:                        break;
                   3002:                }
                   3003:        }
                   3004:
                   3005:        error = VOP_GETATTR(vp, &va, ap->a_cred, ap->a_p);
                   3006:        if (error)
                   3007:                return (error);
                   3008:
                   3009:        return (vaccess(va.va_mode, va.va_uid, va.va_gid, ap->a_mode,
                   3010:            ap->a_cred));
                   3011: }
                   3012:
                   3013: /* ARGSUSED */
                   3014: int
                   3015: nfs_poll(v)
                   3016:         void *v;
                   3017: {
                   3018:        struct vop_poll_args *ap = v;
                   3019:
                   3020:        /*
                   3021:         * We should really check to see if I/O is possible.
                   3022:         */
                   3023:        return (ap->a_events & (POLLIN | POLLOUT | POLLRDNORM | POLLWRNORM));
                   3024: }
                   3025:
                   3026: /*
                   3027:  * Read wrapper for special devices.
                   3028:  */
                   3029: int
                   3030: nfsspec_read(v)
                   3031:        void *v;
                   3032: {
                   3033:        struct vop_read_args *ap = v;
                   3034:        struct nfsnode *np = VTONFS(ap->a_vp);
                   3035:
                   3036:        /*
                   3037:         * Set access flag.
                   3038:         */
                   3039:        np->n_flag |= NACC;
                   3040:        getnanotime(&np->n_atim);
                   3041:        return (VOCALL(spec_vnodeop_p, VOFFSET(vop_read), ap));
                   3042: }
                   3043:
                   3044: /*
                   3045:  * Write wrapper for special devices.
                   3046:  */
                   3047: int
                   3048: nfsspec_write(v)
                   3049:        void *v;
                   3050: {
                   3051:        struct vop_write_args *ap = v;
                   3052:        struct nfsnode *np = VTONFS(ap->a_vp);
                   3053:
                   3054:        /*
                   3055:         * Set update flag.
                   3056:         */
                   3057:        np->n_flag |= NUPD;
                   3058:        getnanotime(&np->n_mtim);
                   3059:        return (VOCALL(spec_vnodeop_p, VOFFSET(vop_write), ap));
                   3060: }
                   3061:
                   3062: /*
                   3063:  * Close wrapper for special devices.
                   3064:  *
                   3065:  * Update the times on the nfsnode then do device close.
                   3066:  */
                   3067: int
                   3068: nfsspec_close(v)
                   3069:        void *v;
                   3070: {
                   3071:        struct vop_close_args *ap = v;
                   3072:        struct vnode *vp = ap->a_vp;
                   3073:        struct nfsnode *np = VTONFS(vp);
                   3074:        struct vattr vattr;
                   3075:
                   3076:        if (np->n_flag & (NACC | NUPD)) {
                   3077:                np->n_flag |= NCHG;
                   3078:                if (vp->v_usecount == 1 &&
                   3079:                    (vp->v_mount->mnt_flag & MNT_RDONLY) == 0) {
                   3080:                        VATTR_NULL(&vattr);
                   3081:                        if (np->n_flag & NACC)
                   3082:                                vattr.va_atime = np->n_atim;
                   3083:                        if (np->n_flag & NUPD)
                   3084:                                vattr.va_mtime = np->n_mtim;
                   3085:                        (void)VOP_SETATTR(vp, &vattr, ap->a_cred, ap->a_p);
                   3086:                }
                   3087:        }
                   3088:        return (VOCALL(spec_vnodeop_p, VOFFSET(vop_close), ap));
                   3089: }
                   3090:
                   3091: #ifdef FIFO
                   3092: /*
                   3093:  * Read wrapper for fifos.
                   3094:  */
                   3095: int
                   3096: nfsfifo_read(v)
                   3097:        void *v;
                   3098: {
                   3099:        struct vop_read_args *ap = v;
                   3100:        extern int (**fifo_vnodeop_p)(void *);
                   3101:        struct nfsnode *np = VTONFS(ap->a_vp);
                   3102:
                   3103:        /*
                   3104:         * Set access flag.
                   3105:         */
                   3106:        np->n_flag |= NACC;
                   3107:        getnanotime(&np->n_atim);
                   3108:        return (VOCALL(fifo_vnodeop_p, VOFFSET(vop_read), ap));
                   3109: }
                   3110:
                   3111: /*
                   3112:  * Write wrapper for fifos.
                   3113:  */
                   3114: int
                   3115: nfsfifo_write(v)
                   3116:        void *v;
                   3117: {
                   3118:        struct vop_write_args *ap = v;
                   3119:        extern int (**fifo_vnodeop_p)(void *);
                   3120:        struct nfsnode *np = VTONFS(ap->a_vp);
                   3121:
                   3122:        /*
                   3123:         * Set update flag.
                   3124:         */
                   3125:        np->n_flag |= NUPD;
                   3126:        getnanotime(&np->n_mtim);
                   3127:        return (VOCALL(fifo_vnodeop_p, VOFFSET(vop_write), ap));
                   3128: }
                   3129:
                   3130: /*
                   3131:  * Close wrapper for fifos.
                   3132:  *
                   3133:  * Update the times on the nfsnode then do fifo close.
                   3134:  */
                   3135: int
                   3136: nfsfifo_close(v)
                   3137:        void *v;
                   3138: {
                   3139:        struct vop_close_args *ap = v;
                   3140:        struct vnode *vp = ap->a_vp;
                   3141:        struct nfsnode *np = VTONFS(vp);
                   3142:        struct vattr vattr;
                   3143:        extern int (**fifo_vnodeop_p)(void *);
                   3144:
                   3145:        if (np->n_flag & (NACC | NUPD)) {
                   3146:                if (np->n_flag & NACC) {
                   3147:                        getnanotime(&np->n_atim);
                   3148:                }
                   3149:                if (np->n_flag & NUPD) {
                   3150:                        getnanotime(&np->n_mtim);
                   3151:                }
                   3152:                np->n_flag |= NCHG;
                   3153:                if (vp->v_usecount == 1 &&
                   3154:                    (vp->v_mount->mnt_flag & MNT_RDONLY) == 0) {
                   3155:                        VATTR_NULL(&vattr);
                   3156:                        if (np->n_flag & NACC)
                   3157:                                vattr.va_atime = np->n_atim;
                   3158:                        if (np->n_flag & NUPD)
                   3159:                                vattr.va_mtime = np->n_mtim;
                   3160:                        (void)VOP_SETATTR(vp, &vattr, ap->a_cred, ap->a_p);
                   3161:                }
                   3162:        }
                   3163:        return (VOCALL(fifo_vnodeop_p, VOFFSET(vop_close), ap));
                   3164: }
                   3165:
                   3166: int
                   3167: nfsfifo_reclaim(void *v)
                   3168: {
                   3169:        fifo_reclaim(v);
                   3170:        return (nfs_reclaim(v));
                   3171: }
                   3172: #endif /* ! FIFO */

CVSweb