138420Smckusick /* 238420Smckusick * Copyright (c) 1989 The Regents of the University of California. 338420Smckusick * All rights reserved. 438420Smckusick * 538420Smckusick * This code is derived from software contributed to Berkeley by 638420Smckusick * Rick Macklem at The University of Guelph. 738420Smckusick * 844513Sbostic * %sccs.include.redist.c% 938420Smckusick * 10*54445Smckusick * @(#)nfs_subs.c 7.55 (Berkeley) 06/25/92 1138420Smckusick */ 1238420Smckusick 1338420Smckusick /* 1438420Smckusick * These functions support the macros and help fiddle mbuf chains for 1538420Smckusick * the nfs op functions. They do things like create the rpc header and 1638420Smckusick * copy data between mbuf chains and uio lists. 1738420Smckusick */ 1853322Smckusick #include <sys/param.h> 1953322Smckusick #include <sys/proc.h> 2053322Smckusick #include <sys/filedesc.h> 2153322Smckusick #include <sys/systm.h> 2253322Smckusick #include <sys/kernel.h> 2353322Smckusick #include <sys/mount.h> 2453322Smckusick #include <sys/file.h> 2553322Smckusick #include <sys/vnode.h> 2653322Smckusick #include <sys/namei.h> 2753322Smckusick #include <sys/mbuf.h> 2853322Smckusick #include <sys/map.h> 2953322Smckusick #include <sys/socket.h> 30*54445Smckusick #include <sys/stat.h> 3147573Skarels 3253322Smckusick #include <vm/vm.h> 3347573Skarels 3453322Smckusick #include <ufs/ufs/quota.h> 3553322Smckusick #include <ufs/ufs/inode.h> 3653322Smckusick #include <ufs/ufs/ufsmount.h> 3738420Smckusick 3853322Smckusick #include <nfs/rpcv2.h> 3953322Smckusick #include <nfs/nfsv2.h> 4053322Smckusick #include <nfs/nfsnode.h> 4153322Smckusick #include <nfs/nfs.h> 4253322Smckusick #include <nfs/xdr_subs.h> 4353322Smckusick #include <nfs/nfsm_subs.h> 4453322Smckusick #include <nfs/nfsmount.h> 4553322Smckusick #include <nfs/nqnfs.h> 4653322Smckusick #include <nfs/nfsrtt.h> 4753322Smckusick 4838420Smckusick #define TRUE 1 4938420Smckusick #define FALSE 0 5038420Smckusick 5138420Smckusick /* 5238420Smckusick * Data items converted to xdr at startup, since they are constant 5338420Smckusick * This is kinda hokey, but may save a little time doing byte swaps 5438420Smckusick */ 5538420Smckusick u_long nfs_procids[NFS_NPROCS]; 5638420Smckusick u_long nfs_xdrneg1; 5752196Smckusick u_long rpc_call, rpc_vers, rpc_reply, rpc_msgdenied, rpc_autherr, 5852196Smckusick rpc_mismatch, rpc_auth_unix, rpc_msgaccepted, rpc_rejectedcred, 5952196Smckusick rpc_auth_kerb; 6038420Smckusick u_long nfs_vers, nfs_prog, nfs_true, nfs_false; 6152196Smckusick 6238420Smckusick /* And other global data */ 6352196Smckusick static u_long nfs_xid = 0; 6442244Smckusick enum vtype ntov_type[7] = { VNON, VREG, VDIR, VBLK, VCHR, VLNK, VNON }; 6541902Smckusick extern struct proc *nfs_iodwant[NFS_MAXASYNCDAEMON]; 6641902Smckusick extern struct nfsreq nfsreqh; 6752196Smckusick extern int nqnfs_piggy[NFS_NPROCS]; 6852196Smckusick extern struct nfsrtt nfsrtt; 6952196Smckusick extern union nqsrvthead nqthead; 7052196Smckusick extern union nqsrvthead nqfhead[NQLCHSZ]; 7152196Smckusick extern time_t nqnfsstarttime; 7252196Smckusick extern u_long nqnfs_prog, nqnfs_vers; 7352196Smckusick extern int nqsrv_clockskew; 7452196Smckusick extern int nqsrv_writeslack; 7552196Smckusick extern int nqsrv_maxlease; 7638420Smckusick 7738420Smckusick /* 7838420Smckusick * Create the header for an rpc request packet 7938420Smckusick * The hsiz is the size of the rest of the nfs request header. 8038420Smckusick * (just used to decide if a cluster is a good idea) 8138420Smckusick */ 8252196Smckusick struct mbuf * 8352196Smckusick nfsm_reqh(vp, procid, hsiz, bposp) 8452196Smckusick struct vnode *vp; 8539494Smckusick u_long procid; 8638420Smckusick int hsiz; 8752196Smckusick caddr_t *bposp; 8838420Smckusick { 8952196Smckusick register struct mbuf *mb; 9048049Smckusick register u_long *tl; 9152196Smckusick register caddr_t bpos; 9252196Smckusick struct mbuf *mb2; 9352196Smckusick struct nfsmount *nmp; 9452196Smckusick int nqflag; 9538420Smckusick 9652196Smckusick MGET(mb, M_WAIT, MT_DATA); 9752196Smckusick if (hsiz >= MINCLSIZE) 9852196Smckusick MCLGET(mb, M_WAIT); 9952196Smckusick mb->m_len = 0; 10052196Smckusick bpos = mtod(mb, caddr_t); 10152196Smckusick 10252196Smckusick /* 10352196Smckusick * For NQNFS, add lease request. 10452196Smckusick */ 10552196Smckusick if (vp) { 10652196Smckusick nmp = VFSTONFS(vp->v_mount); 10752196Smckusick if (nmp->nm_flag & NFSMNT_NQNFS) { 10852196Smckusick nqflag = NQNFS_NEEDLEASE(vp, procid); 10952196Smckusick if (nqflag) { 11052196Smckusick nfsm_build(tl, u_long *, 2*NFSX_UNSIGNED); 11152196Smckusick *tl++ = txdr_unsigned(nqflag); 11252196Smckusick *tl = txdr_unsigned(nmp->nm_leaseterm); 11352196Smckusick } else { 11452196Smckusick nfsm_build(tl, u_long *, NFSX_UNSIGNED); 11552196Smckusick *tl = 0; 11652196Smckusick } 11752196Smckusick } 11852196Smckusick } 11952196Smckusick /* Finally, return values */ 12052196Smckusick *bposp = bpos; 12152196Smckusick return (mb); 12252196Smckusick } 12338420Smckusick 12452196Smckusick /* 12552196Smckusick * Build the RPC header and fill in the authorization info. 12652196Smckusick * The authorization string argument is only used when the credentials 12752196Smckusick * come from outside of the kernel. 12852196Smckusick * Returns the head of the mbuf list. 12952196Smckusick */ 13052196Smckusick struct mbuf * 13152196Smckusick nfsm_rpchead(cr, nqnfs, procid, auth_type, auth_len, auth_str, mrest, 13252196Smckusick mrest_len, mbp, xidp) 13352196Smckusick register struct ucred *cr; 13452196Smckusick int nqnfs; 13552196Smckusick int procid; 13652196Smckusick int auth_type; 13752196Smckusick int auth_len; 13852196Smckusick char *auth_str; 13952196Smckusick struct mbuf *mrest; 14052196Smckusick int mrest_len; 14152196Smckusick struct mbuf **mbp; 14252196Smckusick u_long *xidp; 14352196Smckusick { 14452196Smckusick register struct mbuf *mb; 14552196Smckusick register u_long *tl; 14652196Smckusick register caddr_t bpos; 14752196Smckusick register int i; 14852196Smckusick struct mbuf *mreq, *mb2; 14952196Smckusick int siz, grpsiz, authsiz; 15052196Smckusick 15152196Smckusick authsiz = nfsm_rndup(auth_len); 15252196Smckusick if (auth_type == RPCAUTH_NQNFS) 15352196Smckusick authsiz += 2 * NFSX_UNSIGNED; 15452196Smckusick MGETHDR(mb, M_WAIT, MT_DATA); 15552196Smckusick if ((authsiz + 10*NFSX_UNSIGNED) >= MINCLSIZE) { 15652196Smckusick MCLGET(mb, M_WAIT); 15752196Smckusick } else if ((authsiz + 10*NFSX_UNSIGNED) < MHLEN) { 15852196Smckusick MH_ALIGN(mb, authsiz + 10*NFSX_UNSIGNED); 15952196Smckusick } else { 16052196Smckusick MH_ALIGN(mb, 8*NFSX_UNSIGNED); 16152196Smckusick } 16252196Smckusick mb->m_len = 0; 16352196Smckusick mreq = mb; 16452196Smckusick bpos = mtod(mb, caddr_t); 16552196Smckusick 16638420Smckusick /* 16752196Smckusick * First the RPC header. 16838420Smckusick */ 16952196Smckusick nfsm_build(tl, u_long *, 8*NFSX_UNSIGNED); 17052196Smckusick if (++nfs_xid == 0) 17152196Smckusick nfs_xid++; 17252196Smckusick *tl++ = *xidp = txdr_unsigned(nfs_xid); 17348049Smckusick *tl++ = rpc_call; 17448049Smckusick *tl++ = rpc_vers; 17552196Smckusick if (nqnfs) { 17652196Smckusick *tl++ = txdr_unsigned(NQNFS_PROG); 17752196Smckusick *tl++ = txdr_unsigned(NQNFS_VER1); 17852196Smckusick } else { 17952196Smckusick *tl++ = txdr_unsigned(NFS_PROG); 18052196Smckusick *tl++ = txdr_unsigned(NFS_VER2); 18152196Smckusick } 18252196Smckusick *tl++ = txdr_unsigned(procid); 18338420Smckusick 18452196Smckusick /* 18552196Smckusick * And then the authorization cred. 18652196Smckusick */ 18752196Smckusick *tl++ = txdr_unsigned(auth_type); 18852196Smckusick *tl = txdr_unsigned(authsiz); 18952196Smckusick switch (auth_type) { 19052196Smckusick case RPCAUTH_UNIX: 19152196Smckusick nfsm_build(tl, u_long *, auth_len); 19252196Smckusick *tl++ = 0; /* stamp ?? */ 19352196Smckusick *tl++ = 0; /* NULL hostname */ 19452196Smckusick *tl++ = txdr_unsigned(cr->cr_uid); 19552196Smckusick *tl++ = txdr_unsigned(cr->cr_groups[0]); 19652196Smckusick grpsiz = (auth_len >> 2) - 5; 19752196Smckusick *tl++ = txdr_unsigned(grpsiz); 19852196Smckusick for (i = 1; i <= grpsiz; i++) 19952196Smckusick *tl++ = txdr_unsigned(cr->cr_groups[i]); 20052196Smckusick break; 20152196Smckusick case RPCAUTH_NQNFS: 20252196Smckusick nfsm_build(tl, u_long *, 2*NFSX_UNSIGNED); 20352196Smckusick *tl++ = txdr_unsigned(cr->cr_uid); 20452196Smckusick *tl = txdr_unsigned(auth_len); 20552196Smckusick siz = auth_len; 20652196Smckusick while (siz > 0) { 20752196Smckusick if (M_TRAILINGSPACE(mb) == 0) { 20852196Smckusick MGET(mb2, M_WAIT, MT_DATA); 20952196Smckusick if (siz >= MINCLSIZE) 21052196Smckusick MCLGET(mb2, M_WAIT); 21152196Smckusick mb->m_next = mb2; 21252196Smckusick mb = mb2; 21352196Smckusick mb->m_len = 0; 21452196Smckusick bpos = mtod(mb, caddr_t); 21552196Smckusick } 21652196Smckusick i = MIN(siz, M_TRAILINGSPACE(mb)); 21752196Smckusick bcopy(auth_str, bpos, i); 21852196Smckusick mb->m_len += i; 21952196Smckusick auth_str += i; 22052196Smckusick bpos += i; 22152196Smckusick siz -= i; 22238420Smckusick } 22352196Smckusick if ((siz = nfsm_rndup(auth_len) - auth_len) > 0) { 22452196Smckusick for (i = 0; i < siz; i++) 22552196Smckusick *bpos++ = '\0'; 22652196Smckusick mb->m_len += siz; 22752196Smckusick } 22852196Smckusick break; 22952196Smckusick }; 23052196Smckusick nfsm_build(tl, u_long *, 2*NFSX_UNSIGNED); 23152196Smckusick *tl++ = txdr_unsigned(RPCAUTH_NULL); 23252196Smckusick *tl = 0; 23352196Smckusick mb->m_next = mrest; 23452196Smckusick mreq->m_pkthdr.len = authsiz + 10*NFSX_UNSIGNED + mrest_len; 23552196Smckusick mreq->m_pkthdr.rcvif = (struct ifnet *)0; 23652196Smckusick *mbp = mb; 23738420Smckusick return (mreq); 23838420Smckusick } 23938420Smckusick 24038420Smckusick /* 24138420Smckusick * copies mbuf chain to the uio scatter/gather list 24238420Smckusick */ 24338420Smckusick nfsm_mbuftouio(mrep, uiop, siz, dpos) 24438420Smckusick struct mbuf **mrep; 24543353Smckusick register struct uio *uiop; 24638420Smckusick int siz; 24738420Smckusick caddr_t *dpos; 24838420Smckusick { 24943353Smckusick register char *mbufcp, *uiocp; 25038420Smckusick register int xfer, left, len; 25138420Smckusick register struct mbuf *mp; 25238420Smckusick long uiosiz, rem; 25341902Smckusick int error = 0; 25438420Smckusick 25538420Smckusick mp = *mrep; 25638420Smckusick mbufcp = *dpos; 25738420Smckusick len = mtod(mp, caddr_t)+mp->m_len-mbufcp; 25838420Smckusick rem = nfsm_rndup(siz)-siz; 25938420Smckusick while (siz > 0) { 26038420Smckusick if (uiop->uio_iovcnt <= 0 || uiop->uio_iov == NULL) 26141902Smckusick return (EFBIG); 26238420Smckusick left = uiop->uio_iov->iov_len; 26338420Smckusick uiocp = uiop->uio_iov->iov_base; 26438420Smckusick if (left > siz) 26538420Smckusick left = siz; 26638420Smckusick uiosiz = left; 26738420Smckusick while (left > 0) { 26838420Smckusick while (len == 0) { 26938420Smckusick mp = mp->m_next; 27038420Smckusick if (mp == NULL) 27138420Smckusick return (EBADRPC); 27238420Smckusick mbufcp = mtod(mp, caddr_t); 27338420Smckusick len = mp->m_len; 27438420Smckusick } 27538420Smckusick xfer = (left > len) ? len : left; 27638420Smckusick #ifdef notdef 27738420Smckusick /* Not Yet.. */ 27838420Smckusick if (uiop->uio_iov->iov_op != NULL) 27938420Smckusick (*(uiop->uio_iov->iov_op)) 28038420Smckusick (mbufcp, uiocp, xfer); 28138420Smckusick else 28238420Smckusick #endif 28338420Smckusick if (uiop->uio_segflg == UIO_SYSSPACE) 28438420Smckusick bcopy(mbufcp, uiocp, xfer); 28538420Smckusick else 28638420Smckusick copyout(mbufcp, uiocp, xfer); 28738420Smckusick left -= xfer; 28838420Smckusick len -= xfer; 28938420Smckusick mbufcp += xfer; 29038420Smckusick uiocp += xfer; 29139585Smckusick uiop->uio_offset += xfer; 29238420Smckusick uiop->uio_resid -= xfer; 29338420Smckusick } 29438420Smckusick if (uiop->uio_iov->iov_len <= siz) { 29538420Smckusick uiop->uio_iovcnt--; 29638420Smckusick uiop->uio_iov++; 29738420Smckusick } else { 29838420Smckusick uiop->uio_iov->iov_base += uiosiz; 29938420Smckusick uiop->uio_iov->iov_len -= uiosiz; 30038420Smckusick } 30138420Smckusick siz -= uiosiz; 30238420Smckusick } 30338420Smckusick *dpos = mbufcp; 30438420Smckusick *mrep = mp; 30541902Smckusick if (rem > 0) { 30641902Smckusick if (len < rem) 30741902Smckusick error = nfs_adv(mrep, dpos, rem, len); 30841902Smckusick else 30941902Smckusick *dpos += rem; 31041902Smckusick } 31141902Smckusick return (error); 31238420Smckusick } 31338420Smckusick 31438420Smckusick /* 31538420Smckusick * copies a uio scatter/gather list to an mbuf chain... 31638420Smckusick */ 31738420Smckusick nfsm_uiotombuf(uiop, mq, siz, bpos) 31838420Smckusick register struct uio *uiop; 31938420Smckusick struct mbuf **mq; 32038420Smckusick int siz; 32138420Smckusick caddr_t *bpos; 32238420Smckusick { 32343353Smckusick register char *uiocp; 32443353Smckusick register struct mbuf *mp, *mp2; 32552196Smckusick register int xfer, left, mlen; 32643353Smckusick int uiosiz, clflg, rem; 32743353Smckusick char *cp; 32838420Smckusick 32938420Smckusick if (siz > MLEN) /* or should it >= MCLBYTES ?? */ 33038420Smckusick clflg = 1; 33138420Smckusick else 33238420Smckusick clflg = 0; 33338420Smckusick rem = nfsm_rndup(siz)-siz; 33452196Smckusick mp = mp2 = *mq; 33538420Smckusick while (siz > 0) { 33638420Smckusick if (uiop->uio_iovcnt <= 0 || uiop->uio_iov == NULL) 33741902Smckusick return (EINVAL); 33838420Smckusick left = uiop->uio_iov->iov_len; 33938420Smckusick uiocp = uiop->uio_iov->iov_base; 34038420Smckusick if (left > siz) 34138420Smckusick left = siz; 34238420Smckusick uiosiz = left; 34338420Smckusick while (left > 0) { 34452196Smckusick mlen = M_TRAILINGSPACE(mp); 34552196Smckusick if (mlen == 0) { 34652196Smckusick MGET(mp, M_WAIT, MT_DATA); 34752196Smckusick if (clflg) 34852196Smckusick MCLGET(mp, M_WAIT); 34952196Smckusick mp->m_len = 0; 35052196Smckusick mp2->m_next = mp; 35152196Smckusick mp2 = mp; 35252196Smckusick mlen = M_TRAILINGSPACE(mp); 35352196Smckusick } 35452196Smckusick xfer = (left > mlen) ? mlen : left; 35538420Smckusick #ifdef notdef 35638420Smckusick /* Not Yet.. */ 35738420Smckusick if (uiop->uio_iov->iov_op != NULL) 35838420Smckusick (*(uiop->uio_iov->iov_op)) 35952196Smckusick (uiocp, mtod(mp, caddr_t)+mp->m_len, xfer); 36038420Smckusick else 36138420Smckusick #endif 36238420Smckusick if (uiop->uio_segflg == UIO_SYSSPACE) 36352196Smckusick bcopy(uiocp, mtod(mp, caddr_t)+mp->m_len, xfer); 36438420Smckusick else 36552196Smckusick copyin(uiocp, mtod(mp, caddr_t)+mp->m_len, xfer); 36652196Smckusick mp->m_len += xfer; 36738420Smckusick left -= xfer; 36838420Smckusick uiocp += xfer; 36939585Smckusick uiop->uio_offset += xfer; 37038420Smckusick uiop->uio_resid -= xfer; 37138420Smckusick } 37238420Smckusick if (uiop->uio_iov->iov_len <= siz) { 37338420Smckusick uiop->uio_iovcnt--; 37438420Smckusick uiop->uio_iov++; 37538420Smckusick } else { 37638420Smckusick uiop->uio_iov->iov_base += uiosiz; 37738420Smckusick uiop->uio_iov->iov_len -= uiosiz; 37838420Smckusick } 37938420Smckusick siz -= uiosiz; 38038420Smckusick } 38138420Smckusick if (rem > 0) { 38252196Smckusick if (rem > M_TRAILINGSPACE(mp)) { 38338420Smckusick MGET(mp, M_WAIT, MT_DATA); 38438420Smckusick mp->m_len = 0; 38538420Smckusick mp2->m_next = mp; 38638420Smckusick } 38738420Smckusick cp = mtod(mp, caddr_t)+mp->m_len; 38838420Smckusick for (left = 0; left < rem; left++) 38938420Smckusick *cp++ = '\0'; 39038420Smckusick mp->m_len += rem; 39138420Smckusick *bpos = cp; 39238420Smckusick } else 39338420Smckusick *bpos = mtod(mp, caddr_t)+mp->m_len; 39438420Smckusick *mq = mp; 39541902Smckusick return (0); 39638420Smckusick } 39738420Smckusick 39838420Smckusick /* 39938420Smckusick * Help break down an mbuf chain by setting the first siz bytes contiguous 40038420Smckusick * pointed to by returned val. 40138420Smckusick * If Updateflg == True we can overwrite the first part of the mbuf data 40252196Smckusick * (in this case it can never sleep, so it can be called from interrupt level) 40352196Smckusick * it may however block when Updateflg == False 40452196Smckusick * This is used by the macros nfsm_dissect and nfsm_dissecton for tough 40538420Smckusick * cases. (The macros use the vars. dpos and dpos2) 40638420Smckusick */ 40738420Smckusick nfsm_disct(mdp, dposp, siz, left, updateflg, cp2) 40838420Smckusick struct mbuf **mdp; 40938420Smckusick caddr_t *dposp; 41038420Smckusick int siz; 41138420Smckusick int left; 41238420Smckusick int updateflg; 41338420Smckusick caddr_t *cp2; 41438420Smckusick { 41538420Smckusick register struct mbuf *mp, *mp2; 41638420Smckusick register int siz2, xfer; 41752196Smckusick register caddr_t p; 41838420Smckusick 41938420Smckusick mp = *mdp; 42038420Smckusick while (left == 0) { 42138420Smckusick *mdp = mp = mp->m_next; 42238420Smckusick if (mp == NULL) 42341902Smckusick return (EBADRPC); 42438420Smckusick left = mp->m_len; 42538420Smckusick *dposp = mtod(mp, caddr_t); 42638420Smckusick } 42738420Smckusick if (left >= siz) { 42838420Smckusick *cp2 = *dposp; 42938420Smckusick *dposp += siz; 43038420Smckusick } else if (mp->m_next == NULL) { 43141902Smckusick return (EBADRPC); 43241902Smckusick } else if (siz > MHLEN) { 43338420Smckusick panic("nfs S too big"); 43438420Smckusick } else { 43538420Smckusick /* Iff update, you can overwrite, else must alloc new mbuf */ 43638420Smckusick if (updateflg) { 43738420Smckusick NFSMINOFF(mp); 43838420Smckusick } else { 43938420Smckusick MGET(mp2, M_WAIT, MT_DATA); 44038420Smckusick mp2->m_next = mp->m_next; 44138420Smckusick mp->m_next = mp2; 44238420Smckusick mp->m_len -= left; 44338420Smckusick mp = mp2; 44438420Smckusick } 44552196Smckusick *cp2 = p = mtod(mp, caddr_t); 44652196Smckusick bcopy(*dposp, p, left); /* Copy what was left */ 44738420Smckusick siz2 = siz-left; 44852196Smckusick p += left; 44938420Smckusick mp2 = mp->m_next; 45041902Smckusick /* Loop around copying up the siz2 bytes */ 45138420Smckusick while (siz2 > 0) { 45238420Smckusick if (mp2 == NULL) 45338420Smckusick return (EBADRPC); 45438420Smckusick xfer = (siz2 > mp2->m_len) ? mp2->m_len : siz2; 45541902Smckusick if (xfer > 0) { 45652196Smckusick bcopy(mtod(mp2, caddr_t), p, xfer); 45741902Smckusick NFSMADV(mp2, xfer); 45841902Smckusick mp2->m_len -= xfer; 45952196Smckusick p += xfer; 46041902Smckusick siz2 -= xfer; 46141902Smckusick } 46238420Smckusick if (siz2 > 0) 46338420Smckusick mp2 = mp2->m_next; 46438420Smckusick } 46538420Smckusick mp->m_len = siz; 46638420Smckusick *mdp = mp2; 46738420Smckusick *dposp = mtod(mp2, caddr_t); 46838420Smckusick } 46939494Smckusick return (0); 47038420Smckusick } 47138420Smckusick 47238420Smckusick /* 47341902Smckusick * Advance the position in the mbuf chain. 47438420Smckusick */ 47538420Smckusick nfs_adv(mdp, dposp, offs, left) 47638420Smckusick struct mbuf **mdp; 47738420Smckusick caddr_t *dposp; 47838420Smckusick int offs; 47938420Smckusick int left; 48038420Smckusick { 48138420Smckusick register struct mbuf *m; 48238420Smckusick register int s; 48338420Smckusick 48438420Smckusick m = *mdp; 48538420Smckusick s = left; 48638420Smckusick while (s < offs) { 48738420Smckusick offs -= s; 48838420Smckusick m = m->m_next; 48938420Smckusick if (m == NULL) 49041902Smckusick return (EBADRPC); 49138420Smckusick s = m->m_len; 49238420Smckusick } 49338420Smckusick *mdp = m; 49438420Smckusick *dposp = mtod(m, caddr_t)+offs; 49541902Smckusick return (0); 49638420Smckusick } 49738420Smckusick 49838420Smckusick /* 49938420Smckusick * Copy a string into mbufs for the hard cases... 50038420Smckusick */ 50138420Smckusick nfsm_strtmbuf(mb, bpos, cp, siz) 50238420Smckusick struct mbuf **mb; 50338420Smckusick char **bpos; 50438420Smckusick char *cp; 50538420Smckusick long siz; 50638420Smckusick { 50738420Smckusick register struct mbuf *m1, *m2; 50838420Smckusick long left, xfer, len, tlen; 50948049Smckusick u_long *tl; 51038420Smckusick int putsize; 51138420Smckusick 51238420Smckusick putsize = 1; 51338420Smckusick m2 = *mb; 51452196Smckusick left = M_TRAILINGSPACE(m2); 51538420Smckusick if (left > 0) { 51648049Smckusick tl = ((u_long *)(*bpos)); 51748049Smckusick *tl++ = txdr_unsigned(siz); 51838420Smckusick putsize = 0; 51938420Smckusick left -= NFSX_UNSIGNED; 52038420Smckusick m2->m_len += NFSX_UNSIGNED; 52138420Smckusick if (left > 0) { 52248049Smckusick bcopy(cp, (caddr_t) tl, left); 52338420Smckusick siz -= left; 52438420Smckusick cp += left; 52538420Smckusick m2->m_len += left; 52638420Smckusick left = 0; 52738420Smckusick } 52838420Smckusick } 52952196Smckusick /* Loop around adding mbufs */ 53038420Smckusick while (siz > 0) { 53138420Smckusick MGET(m1, M_WAIT, MT_DATA); 53238420Smckusick if (siz > MLEN) 53341902Smckusick MCLGET(m1, M_WAIT); 53438420Smckusick m1->m_len = NFSMSIZ(m1); 53538420Smckusick m2->m_next = m1; 53638420Smckusick m2 = m1; 53748049Smckusick tl = mtod(m1, u_long *); 53838420Smckusick tlen = 0; 53938420Smckusick if (putsize) { 54048049Smckusick *tl++ = txdr_unsigned(siz); 54138420Smckusick m1->m_len -= NFSX_UNSIGNED; 54238420Smckusick tlen = NFSX_UNSIGNED; 54338420Smckusick putsize = 0; 54438420Smckusick } 54538420Smckusick if (siz < m1->m_len) { 54638420Smckusick len = nfsm_rndup(siz); 54738420Smckusick xfer = siz; 54838420Smckusick if (xfer < len) 54948049Smckusick *(tl+(xfer>>2)) = 0; 55038420Smckusick } else { 55138420Smckusick xfer = len = m1->m_len; 55238420Smckusick } 55348049Smckusick bcopy(cp, (caddr_t) tl, xfer); 55438420Smckusick m1->m_len = len+tlen; 55538420Smckusick siz -= xfer; 55638420Smckusick cp += xfer; 55738420Smckusick } 55838420Smckusick *mb = m1; 55938420Smckusick *bpos = mtod(m1, caddr_t)+m1->m_len; 56041902Smckusick return (0); 56138420Smckusick } 56238420Smckusick 56338420Smckusick /* 56438420Smckusick * Called once to initialize data structures... 56538420Smckusick */ 56639444Smckusick nfs_init() 56738420Smckusick { 56838420Smckusick register int i; 56952196Smckusick union nqsrvthead *lhp; 57038420Smckusick 57152196Smckusick nfsrtt.pos = 0; 57238420Smckusick rpc_vers = txdr_unsigned(RPC_VER2); 57338420Smckusick rpc_call = txdr_unsigned(RPC_CALL); 57438420Smckusick rpc_reply = txdr_unsigned(RPC_REPLY); 57538420Smckusick rpc_msgdenied = txdr_unsigned(RPC_MSGDENIED); 57638420Smckusick rpc_msgaccepted = txdr_unsigned(RPC_MSGACCEPTED); 57738420Smckusick rpc_mismatch = txdr_unsigned(RPC_MISMATCH); 57852196Smckusick rpc_autherr = txdr_unsigned(RPC_AUTHERR); 57952196Smckusick rpc_rejectedcred = txdr_unsigned(AUTH_REJECTCRED); 58038420Smckusick rpc_auth_unix = txdr_unsigned(RPCAUTH_UNIX); 58152196Smckusick rpc_auth_kerb = txdr_unsigned(RPCAUTH_NQNFS); 58238420Smckusick nfs_vers = txdr_unsigned(NFS_VER2); 58338420Smckusick nfs_prog = txdr_unsigned(NFS_PROG); 58438420Smckusick nfs_true = txdr_unsigned(TRUE); 58538420Smckusick nfs_false = txdr_unsigned(FALSE); 58638420Smckusick /* Loop thru nfs procids */ 58738420Smckusick for (i = 0; i < NFS_NPROCS; i++) 58838420Smckusick nfs_procids[i] = txdr_unsigned(i); 58939345Smckusick /* Ensure async daemons disabled */ 59041902Smckusick for (i = 0; i < NFS_MAXASYNCDAEMON; i++) 59139345Smckusick nfs_iodwant[i] = (struct proc *)0; 59238420Smckusick nfs_xdrneg1 = txdr_unsigned(-1); 59338420Smckusick nfs_nhinit(); /* Init the nfsnode table */ 59452979Smckusick nfsrv_init(0); /* Init server data structures */ 59539755Smckusick nfsrv_initcache(); /* Init the server request cache */ 59641902Smckusick 59741902Smckusick /* 59852196Smckusick * Initialize the nqnfs server stuff. 59952196Smckusick */ 60052196Smckusick if (nqnfsstarttime == 0) { 60152196Smckusick nqnfsstarttime = boottime.tv_sec + nqsrv_maxlease 60252196Smckusick + nqsrv_clockskew + nqsrv_writeslack; 60352196Smckusick NQLOADNOVRAM(nqnfsstarttime); 60452196Smckusick nqnfs_prog = txdr_unsigned(NQNFS_PROG); 60552196Smckusick nqnfs_vers = txdr_unsigned(NQNFS_VER1); 60652196Smckusick nqthead.th_head[0] = &nqthead; 60752196Smckusick nqthead.th_head[1] = &nqthead; 60852196Smckusick for (i = 0; i < NQLCHSZ; i++) { 60952196Smckusick lhp = &nqfhead[i]; 61052196Smckusick lhp->th_head[0] = lhp; 61152196Smckusick lhp->th_head[1] = lhp; 61252196Smckusick } 61352196Smckusick } 61452196Smckusick 61552196Smckusick /* 61641902Smckusick * Initialize reply list and start timer 61741902Smckusick */ 61841902Smckusick nfsreqh.r_prev = nfsreqh.r_next = &nfsreqh; 61938420Smckusick nfs_timer(); 62038420Smckusick } 62138420Smckusick 62238420Smckusick /* 62338420Smckusick * Attribute cache routines. 62438420Smckusick * nfs_loadattrcache() - loads or updates the cache contents from attributes 62538420Smckusick * that are on the mbuf list 62638420Smckusick * nfs_getattrcache() - returns valid attributes if found in cache, returns 62738420Smckusick * error otherwise 62838420Smckusick */ 62938420Smckusick 63038420Smckusick /* 63139444Smckusick * Load the attribute cache (that lives in the nfsnode entry) with 63238420Smckusick * the values on the mbuf list and 63338420Smckusick * Iff vap not NULL 63438420Smckusick * copy the attributes to *vaper 63538420Smckusick */ 63639457Smckusick nfs_loadattrcache(vpp, mdp, dposp, vaper) 63739457Smckusick struct vnode **vpp; 63838420Smckusick struct mbuf **mdp; 63938420Smckusick caddr_t *dposp; 64038420Smckusick struct vattr *vaper; 64138420Smckusick { 64239457Smckusick register struct vnode *vp = *vpp; 64338420Smckusick register struct vattr *vap; 64438884Smacklem register struct nfsv2_fattr *fp; 64553553Sheideman extern int (**spec_nfsv2nodeop_p)(); 64653553Sheideman extern int (**spec_vnodeop_p)(); 64739457Smckusick register struct nfsnode *np; 64839494Smckusick register long t1; 64939494Smckusick caddr_t dpos, cp2; 65039494Smckusick int error = 0; 65139494Smckusick struct mbuf *md; 65252196Smckusick enum vtype vtyp; 65352196Smckusick u_short vmode; 65442244Smckusick long rdev; 65539444Smckusick struct timeval mtime; 65639444Smckusick struct vnode *nvp; 65738420Smckusick 65838420Smckusick md = *mdp; 65938420Smckusick dpos = *dposp; 66052196Smckusick t1 = (mtod(md, caddr_t) + md->m_len) - dpos; 66138420Smckusick if (error = nfsm_disct(&md, &dpos, NFSX_FATTR, t1, TRUE, &cp2)) 66238420Smckusick return (error); 66338884Smacklem fp = (struct nfsv2_fattr *)cp2; 66452196Smckusick vtyp = nfstov_type(fp->fa_type); 66552196Smckusick vmode = fxdr_unsigned(u_short, fp->fa_mode); 66653384Smckusick if (vtyp == VNON || vtyp == VREG) 66752196Smckusick vtyp = IFTOVT(vmode); 66842244Smckusick rdev = fxdr_unsigned(long, fp->fa_rdev); 66939444Smckusick fxdr_time(&fp->fa_mtime, &mtime); 67039444Smckusick /* 67139444Smckusick * If v_type == VNON it is a new node, so fill in the v_type, 67239444Smckusick * n_mtime fields. Check to see if it represents a special 67339444Smckusick * device, and if so, check for a possible alias. Once the 67439444Smckusick * correct vnode has been obtained, fill in the rest of the 67539444Smckusick * information. 67639444Smckusick */ 67738420Smckusick np = VTONFS(vp); 67839444Smckusick if (vp->v_type == VNON) { 67952196Smckusick if (vtyp == VCHR && rdev == 0xffffffff) 68052196Smckusick vp->v_type = vtyp = VFIFO; 68142244Smckusick else 68252196Smckusick vp->v_type = vtyp; 68340295Smckusick if (vp->v_type == VFIFO) { 68440295Smckusick #ifdef FIFO 68553553Sheideman extern int (**fifo_nfsv2nodeop_p)(); 68653553Sheideman vp->v_op = fifo_nfsv2nodeop_p; 68740295Smckusick #else 68840295Smckusick return (EOPNOTSUPP); 68940295Smckusick #endif /* FIFO */ 69040295Smckusick } 69139444Smckusick if (vp->v_type == VCHR || vp->v_type == VBLK) { 69253553Sheideman vp->v_op = spec_nfsv2nodeop_p; 69342244Smckusick if (nvp = checkalias(vp, (dev_t)rdev, vp->v_mount)) { 69439444Smckusick /* 69551984Smckusick * Discard unneeded vnode, but save its nfsnode. 69651984Smckusick */ 69751984Smckusick remque(np); 69851984Smckusick nvp->v_data = vp->v_data; 69951984Smckusick vp->v_data = NULL; 70053553Sheideman vp->v_op = spec_vnodeop_p; 70151984Smckusick vrele(vp); 70251984Smckusick vgone(vp); 70351984Smckusick /* 70439444Smckusick * Reinitialize aliased node. 70539444Smckusick */ 70639444Smckusick np->n_vnode = nvp; 70751984Smckusick insque(np, nfs_hash(&np->n_fh)); 70851984Smckusick *vpp = vp = nvp; 70939444Smckusick } 71039444Smckusick } 71152196Smckusick if ((VFSTONFS(vp->v_mount)->nm_flag & NFSMNT_NQNFS) == 0) 71252196Smckusick np->n_mtime = mtime.tv_sec; 71339444Smckusick } 71438420Smckusick vap = &np->n_vattr; 71552196Smckusick vap->va_type = vtyp; 71652196Smckusick vap->va_mode = (vmode & 07777); 71738884Smacklem vap->va_nlink = fxdr_unsigned(u_short, fp->fa_nlink); 71838884Smacklem vap->va_uid = fxdr_unsigned(uid_t, fp->fa_uid); 71938884Smacklem vap->va_gid = fxdr_unsigned(gid_t, fp->fa_gid); 72038884Smacklem vap->va_size = fxdr_unsigned(u_long, fp->fa_size); 72145716Smckusick if ((np->n_flag & NMODIFIED) == 0 || vap->va_size > np->n_size) { 72238884Smacklem np->n_size = vap->va_size; 72353322Smckusick vnode_pager_setsize(vp, (u_long)np->n_size); 72445716Smckusick } 72538884Smacklem vap->va_blocksize = fxdr_unsigned(long, fp->fa_blocksize); 72642244Smckusick vap->va_rdev = (dev_t)rdev; 72742244Smckusick vap->va_bytes = fxdr_unsigned(long, fp->fa_blocks) * NFS_FABLKSIZE; 72842878Smckusick vap->va_fsid = vp->v_mount->mnt_stat.f_fsid.val[0]; 72938884Smacklem vap->va_fileid = fxdr_unsigned(long, fp->fa_fileid); 73054106Smckusick vap->va_atime.ts_sec = fxdr_unsigned(long, fp->fa_atime.tv_sec); 73154106Smckusick vap->va_atime.ts_nsec = 0; 73239755Smckusick vap->va_flags = fxdr_unsigned(u_long, fp->fa_atime.tv_usec); 73354106Smckusick vap->va_mtime.ts_sec = mtime.tv_sec; 73454106Smckusick vap->va_mtime.ts_nsec = mtime.tv_usec * 1000; 73554106Smckusick vap->va_ctime.ts_sec = fxdr_unsigned(long, fp->fa_ctime.tv_sec); 73654106Smckusick vap->va_ctime.ts_nsec = 0; 73739755Smckusick vap->va_gen = fxdr_unsigned(u_long, fp->fa_ctime.tv_usec); 73838420Smckusick np->n_attrstamp = time.tv_sec; 73938420Smckusick *dposp = dpos; 74038420Smckusick *mdp = md; 74138884Smacklem if (vaper != NULL) { 74238420Smckusick bcopy((caddr_t)vap, (caddr_t)vaper, sizeof(*vap)); 74338884Smacklem if ((np->n_flag & NMODIFIED) && (np->n_size > vap->va_size)) 74438884Smacklem vaper->va_size = np->n_size; 74553628Smckusick if (np->n_flag & NCHG) { 74654106Smckusick if (np->n_flag & NACC) { 74754106Smckusick vaper->va_atime.ts_sec = np->n_atim.tv_sec; 74854106Smckusick vaper->va_atime.ts_nsec = 74954106Smckusick np->n_atim.tv_usec * 1000; 75054106Smckusick } 75154106Smckusick if (np->n_flag & NUPD) { 75254106Smckusick vaper->va_mtime.ts_sec = np->n_mtim.tv_sec; 75354106Smckusick vaper->va_mtime.ts_nsec = 75454106Smckusick np->n_mtim.tv_usec * 1000; 75554106Smckusick } 75653628Smckusick } 75738884Smacklem } 75838420Smckusick return (0); 75938420Smckusick } 76038420Smckusick 76138420Smckusick /* 76238420Smckusick * Check the time stamp 76338420Smckusick * If the cache is valid, copy contents to *vap and return 0 76438420Smckusick * otherwise return an error 76538420Smckusick */ 76638420Smckusick nfs_getattrcache(vp, vap) 76738420Smckusick register struct vnode *vp; 76838420Smckusick struct vattr *vap; 76938420Smckusick { 77038420Smckusick register struct nfsnode *np; 77138420Smckusick 77238420Smckusick np = VTONFS(vp); 77352196Smckusick if (VFSTONFS(vp->v_mount)->nm_flag & NFSMNT_NQNFS) { 77452196Smckusick if (!NQNFS_CKCACHABLE(vp, NQL_READ) || np->n_attrstamp == 0) { 77552196Smckusick nfsstats.attrcache_misses++; 77652196Smckusick return (ENOENT); 77752196Smckusick } 77852196Smckusick } else if ((time.tv_sec - np->n_attrstamp) >= NFS_ATTRTIMEO) { 77938420Smckusick nfsstats.attrcache_misses++; 78038420Smckusick return (ENOENT); 78138420Smckusick } 78252196Smckusick nfsstats.attrcache_hits++; 78352196Smckusick bcopy((caddr_t)&np->n_vattr,(caddr_t)vap,sizeof(struct vattr)); 78452196Smckusick if ((np->n_flag & NMODIFIED) == 0) { 78552196Smckusick np->n_size = vap->va_size; 78653322Smckusick vnode_pager_setsize(vp, (u_long)np->n_size); 78752196Smckusick } else if (np->n_size > vap->va_size) 78852196Smckusick vap->va_size = np->n_size; 78953628Smckusick if (np->n_flag & NCHG) { 79054106Smckusick if (np->n_flag & NACC) { 79154106Smckusick vap->va_atime.ts_sec = np->n_atim.tv_sec; 79254106Smckusick vap->va_atime.ts_nsec = np->n_atim.tv_usec * 1000; 79354106Smckusick } 79454106Smckusick if (np->n_flag & NUPD) { 79554106Smckusick vap->va_mtime.ts_sec = np->n_mtim.tv_sec; 79654106Smckusick vap->va_mtime.ts_nsec = np->n_mtim.tv_usec * 1000; 79754106Smckusick } 79853628Smckusick } 79952196Smckusick return (0); 80038420Smckusick } 80138420Smckusick 80238420Smckusick /* 80352196Smckusick * Set up nameidata for a lookup() call and do it 80438420Smckusick */ 80552196Smckusick nfs_namei(ndp, fhp, len, slp, nam, mdp, dposp, p) 80638420Smckusick register struct nameidata *ndp; 80738420Smckusick fhandle_t *fhp; 80838420Smckusick int len; 80952196Smckusick struct nfssvc_sock *slp; 81052196Smckusick struct mbuf *nam; 81138420Smckusick struct mbuf **mdp; 81238420Smckusick caddr_t *dposp; 81349739Smckusick struct proc *p; 81438420Smckusick { 81538420Smckusick register int i, rem; 81638420Smckusick register struct mbuf *md; 81749739Smckusick register char *fromcp, *tocp; 81846514Smckusick struct vnode *dp; 81952315Sheideman int error, rdonly; 82052315Sheideman struct componentname *cnp = &ndp->ni_cnd; 82138420Smckusick 82252315Sheideman MALLOC(cnp->cn_pnbuf, char *, len + 1, M_NAMEI, M_WAITOK); 82349739Smckusick /* 82449739Smckusick * Copy the name from the mbuf list to ndp->ni_pnbuf 82549739Smckusick * and set the various ndp fields appropriately. 82649739Smckusick */ 82749739Smckusick fromcp = *dposp; 82852315Sheideman tocp = cnp->cn_pnbuf; 82949739Smckusick md = *mdp; 83049739Smckusick rem = mtod(md, caddr_t) + md->m_len - fromcp; 83152315Sheideman cnp->cn_hash = 0; 83249739Smckusick for (i = 0; i < len; i++) { 83349739Smckusick while (rem == 0) { 83449739Smckusick md = md->m_next; 83549739Smckusick if (md == NULL) { 83649739Smckusick error = EBADRPC; 83749739Smckusick goto out; 83842244Smckusick } 83949739Smckusick fromcp = mtod(md, caddr_t); 84049739Smckusick rem = md->m_len; 84138420Smckusick } 84249739Smckusick if (*fromcp == '\0' || *fromcp == '/') { 84349739Smckusick error = EINVAL; 84449739Smckusick goto out; 84542244Smckusick } 84649739Smckusick if (*fromcp & 0200) 84752315Sheideman if ((*fromcp&0377) == ('/'|0200) || cnp->cn_nameiop != DELETE) { 84849739Smckusick error = EINVAL; 84949739Smckusick goto out; 85049739Smckusick } 85152315Sheideman cnp->cn_hash += (unsigned char)*fromcp; 85249739Smckusick *tocp++ = *fromcp++; 85349739Smckusick rem--; 85449739Smckusick } 85549739Smckusick *tocp = '\0'; 85649739Smckusick *mdp = md; 85749739Smckusick *dposp = fromcp; 85849739Smckusick len = nfsm_rndup(len)-len; 85949739Smckusick if (len > 0) { 86049739Smckusick if (rem >= len) 86149739Smckusick *dposp += len; 86249739Smckusick else if (error = nfs_adv(mdp, dposp, len, rem)) 86349739Smckusick goto out; 86449739Smckusick } 86552315Sheideman ndp->ni_pathlen = tocp - cnp->cn_pnbuf; 86652315Sheideman cnp->cn_nameptr = cnp->cn_pnbuf; 86746514Smckusick /* 86846514Smckusick * Extract and set starting directory. 86946514Smckusick */ 87052315Sheideman if (error = nfsrv_fhtovp(fhp, FALSE, &dp, ndp->ni_cnd.cn_cred, slp, 87152315Sheideman nam, &rdonly)) 87249739Smckusick goto out; 87338425Smckusick if (dp->v_type != VDIR) { 87441902Smckusick vrele(dp); 87549739Smckusick error = ENOTDIR; 87649739Smckusick goto out; 87738425Smckusick } 87846514Smckusick ndp->ni_startdir = dp; 87952196Smckusick if (rdonly) 88052315Sheideman cnp->cn_flags |= (NOCROSSMOUNT | RDONLY); 88152196Smckusick else 88252315Sheideman cnp->cn_flags |= NOCROSSMOUNT; 88339345Smckusick /* 88449739Smckusick * And call lookup() to do the real work 88538420Smckusick */ 88652315Sheideman cnp->cn_proc = p; 88752315Sheideman if (error = lookup(ndp)) 88849739Smckusick goto out; 88949739Smckusick /* 89049739Smckusick * Check for encountering a symbolic link 89149739Smckusick */ 89252315Sheideman if (cnp->cn_flags & ISSYMLINK) { 89352315Sheideman if ((cnp->cn_flags & LOCKPARENT) && ndp->ni_pathlen == 1) 89449739Smckusick vput(ndp->ni_dvp); 89549739Smckusick else 89649739Smckusick vrele(ndp->ni_dvp); 89749739Smckusick vput(ndp->ni_vp); 89849739Smckusick ndp->ni_vp = NULL; 89949739Smckusick error = EINVAL; 90049739Smckusick goto out; 90149739Smckusick } 90249739Smckusick /* 90349739Smckusick * Check for saved name request 90449739Smckusick */ 90552315Sheideman if (cnp->cn_flags & (SAVENAME | SAVESTART)) { 90652315Sheideman cnp->cn_flags |= HASBUF; 90749739Smckusick return (0); 90849739Smckusick } 90949739Smckusick out: 91052315Sheideman FREE(cnp->cn_pnbuf, M_NAMEI); 91138420Smckusick return (error); 91238420Smckusick } 91338420Smckusick 91438420Smckusick /* 91538420Smckusick * A fiddled version of m_adj() that ensures null fill to a long 91638420Smckusick * boundary and only trims off the back end 91738420Smckusick */ 91852196Smckusick void 91938420Smckusick nfsm_adj(mp, len, nul) 92038420Smckusick struct mbuf *mp; 92138420Smckusick register int len; 92238420Smckusick int nul; 92338420Smckusick { 92438420Smckusick register struct mbuf *m; 92538420Smckusick register int count, i; 92638420Smckusick register char *cp; 92738420Smckusick 92838420Smckusick /* 92938420Smckusick * Trim from tail. Scan the mbuf chain, 93038420Smckusick * calculating its length and finding the last mbuf. 93138420Smckusick * If the adjustment only affects this mbuf, then just 93238420Smckusick * adjust and return. Otherwise, rescan and truncate 93338420Smckusick * after the remaining size. 93438420Smckusick */ 93538420Smckusick count = 0; 93638420Smckusick m = mp; 93738420Smckusick for (;;) { 93838420Smckusick count += m->m_len; 93938420Smckusick if (m->m_next == (struct mbuf *)0) 94038420Smckusick break; 94138420Smckusick m = m->m_next; 94238420Smckusick } 94338579Smckusick if (m->m_len > len) { 94438420Smckusick m->m_len -= len; 94538420Smckusick if (nul > 0) { 94638420Smckusick cp = mtod(m, caddr_t)+m->m_len-nul; 94738420Smckusick for (i = 0; i < nul; i++) 94838420Smckusick *cp++ = '\0'; 94938420Smckusick } 95038420Smckusick return; 95138420Smckusick } 95238420Smckusick count -= len; 95338420Smckusick if (count < 0) 95438420Smckusick count = 0; 95538420Smckusick /* 95638420Smckusick * Correct length for chain is "count". 95738420Smckusick * Find the mbuf with last data, adjust its length, 95838420Smckusick * and toss data from remaining mbufs on chain. 95938420Smckusick */ 96038420Smckusick for (m = mp; m; m = m->m_next) { 96138420Smckusick if (m->m_len >= count) { 96238420Smckusick m->m_len = count; 96338420Smckusick if (nul > 0) { 96438420Smckusick cp = mtod(m, caddr_t)+m->m_len-nul; 96538420Smckusick for (i = 0; i < nul; i++) 96638420Smckusick *cp++ = '\0'; 96738420Smckusick } 96838420Smckusick break; 96938420Smckusick } 97038420Smckusick count -= m->m_len; 97138420Smckusick } 97238420Smckusick while (m = m->m_next) 97338420Smckusick m->m_len = 0; 97438420Smckusick } 97538420Smckusick 97638420Smckusick /* 97738420Smckusick * nfsrv_fhtovp() - convert a fh to a vnode ptr (optionally locked) 97838420Smckusick * - look up fsid in mount list (if not found ret error) 97938420Smckusick * - check that it is exported 98038420Smckusick * - get vp by calling VFS_FHTOVP() macro 98138420Smckusick * - if not lockflag unlock it with VOP_UNLOCK() 98252196Smckusick * - if cred->cr_uid == 0 or MNT_EXPORTANON set it to neth_anon 98338420Smckusick */ 98452196Smckusick nfsrv_fhtovp(fhp, lockflag, vpp, cred, slp, nam, rdonlyp) 98538420Smckusick fhandle_t *fhp; 98638420Smckusick int lockflag; 98738420Smckusick struct vnode **vpp; 98838420Smckusick struct ucred *cred; 98952196Smckusick struct nfssvc_sock *slp; 99052196Smckusick struct mbuf *nam; 99152196Smckusick int *rdonlyp; 99238420Smckusick { 99353553Sheideman USES_VOP_UNLOCK; 99438420Smckusick register struct mount *mp; 99552196Smckusick register struct netaddrhash *np; 99652196Smckusick register struct ufsmount *ump; 99752196Smckusick register struct nfsuid *uidp; 99852196Smckusick struct sockaddr *saddr; 99952196Smckusick int error; 100038420Smckusick 100152196Smckusick *vpp = (struct vnode *)0; 100238420Smckusick if ((mp = getvfs(&fhp->fh_fsid)) == NULL) 100338420Smckusick return (ESTALE); 100441398Smckusick if ((mp->mnt_flag & MNT_EXPORTED) == 0) 100538420Smckusick return (EACCES); 100645282Smckusick 100752196Smckusick /* 100852196Smckusick * Get the export permission structure for this <mp, client> tuple. 100952196Smckusick */ 101052196Smckusick ump = VFSTOUFS(mp); 101152196Smckusick if (nam) { 101245282Smckusick 101352196Smckusick /* 101452196Smckusick * First search for a network match. 101552196Smckusick */ 101652196Smckusick np = ump->um_netaddr[NETMASK_HASH]; 101752196Smckusick while (np) { 101852196Smckusick if (nfs_netaddr_match(np->neth_family, &np->neth_haddr, 101952196Smckusick &np->neth_hmask, nam)) 102052196Smckusick break; 102152196Smckusick np = np->neth_next; 102245282Smckusick } 102352196Smckusick 102452196Smckusick /* 102552196Smckusick * If not found, try for an address match. 102652196Smckusick */ 102752196Smckusick if (np == (struct netaddrhash *)0) { 102852196Smckusick saddr = mtod(nam, struct sockaddr *); 102952196Smckusick np = ump->um_netaddr[NETADDRHASH(saddr)]; 103052196Smckusick while (np) { 103152196Smckusick if (nfs_netaddr_match(np->neth_family, &np->neth_haddr, 103252196Smckusick (struct netaddrhash *)0, nam)) 103352196Smckusick break; 103452196Smckusick np = np->neth_next; 103552196Smckusick } 103645282Smckusick } 103752196Smckusick } else 103852196Smckusick np = (struct netaddrhash *)0; 103952196Smckusick if (np == (struct netaddrhash *)0) { 104052196Smckusick 104152196Smckusick /* 104252196Smckusick * If no address match, use the default if it exists. 104352196Smckusick */ 104452196Smckusick if ((mp->mnt_flag & MNT_DEFEXPORTED) == 0) 104552196Smckusick return (EACCES); 104652196Smckusick np = &ump->um_defexported; 104745282Smckusick } 104852196Smckusick 104952196Smckusick /* 105052196Smckusick * Check/setup credentials. 105152196Smckusick */ 105252196Smckusick if (np->neth_exflags & MNT_EXKERB) { 105352196Smckusick uidp = slp->ns_uidh[NUIDHASH(cred->cr_uid)]; 105452196Smckusick while (uidp) { 105552196Smckusick if (uidp->nu_uid == cred->cr_uid) 105652196Smckusick break; 105752196Smckusick uidp = uidp->nu_hnext; 105845282Smckusick } 105952196Smckusick if (uidp) { 106052196Smckusick if (cred->cr_ref != 1) 106152196Smckusick panic("nsrv fhtovp"); 106252196Smckusick *cred = uidp->nu_cr; 106352196Smckusick } else 106452196Smckusick return (NQNFS_AUTHERR); 106552196Smckusick } else if (cred->cr_uid == 0 || (np->neth_exflags & MNT_EXPORTANON)) 106652196Smckusick *cred = np->neth_anon; 1067*54445Smckusick if (error = VFS_FHTOVP(mp, &fhp->fh_fid, vpp)) 106852196Smckusick return (ESTALE); 106952196Smckusick if (np->neth_exflags & MNT_EXRDONLY) 107052196Smckusick *rdonlyp = 1; 107145282Smckusick else 107252196Smckusick *rdonlyp = 0; 107352196Smckusick if (!lockflag) 107452196Smckusick VOP_UNLOCK(*vpp); 107552196Smckusick return (0); 107645282Smckusick } 1077