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*55519Smckusick * @(#)nfs_subs.c 7.61 (Berkeley) 07/22/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/systm.h> 2153322Smckusick #include <sys/kernel.h> 2253322Smckusick #include <sys/mount.h> 2353322Smckusick #include <sys/vnode.h> 2453322Smckusick #include <sys/namei.h> 2553322Smckusick #include <sys/mbuf.h> 2653322Smckusick #include <sys/socket.h> 2754445Smckusick #include <sys/stat.h> 2847573Skarels 2953322Smckusick #include <nfs/rpcv2.h> 3053322Smckusick #include <nfs/nfsv2.h> 3153322Smckusick #include <nfs/nfsnode.h> 3253322Smckusick #include <nfs/nfs.h> 3353322Smckusick #include <nfs/xdr_subs.h> 3453322Smckusick #include <nfs/nfsm_subs.h> 3553322Smckusick #include <nfs/nfsmount.h> 3653322Smckusick #include <nfs/nqnfs.h> 3753322Smckusick #include <nfs/nfsrtt.h> 3853322Smckusick 3954988Smckusick #include <netinet/in.h> 4054988Smckusick #ifdef ISO 4154988Smckusick #include <netiso/iso.h> 4254988Smckusick #endif 4354988Smckusick 4438420Smckusick #define TRUE 1 4538420Smckusick #define FALSE 0 4638420Smckusick 4738420Smckusick /* 4838420Smckusick * Data items converted to xdr at startup, since they are constant 4938420Smckusick * This is kinda hokey, but may save a little time doing byte swaps 5038420Smckusick */ 5138420Smckusick u_long nfs_procids[NFS_NPROCS]; 5238420Smckusick u_long nfs_xdrneg1; 5352196Smckusick u_long rpc_call, rpc_vers, rpc_reply, rpc_msgdenied, rpc_autherr, 5452196Smckusick rpc_mismatch, rpc_auth_unix, rpc_msgaccepted, rpc_rejectedcred, 5552196Smckusick rpc_auth_kerb; 5638420Smckusick u_long nfs_vers, nfs_prog, nfs_true, nfs_false; 5752196Smckusick 5838420Smckusick /* And other global data */ 5952196Smckusick static u_long nfs_xid = 0; 6042244Smckusick enum vtype ntov_type[7] = { VNON, VREG, VDIR, VBLK, VCHR, VLNK, VNON }; 6141902Smckusick extern struct proc *nfs_iodwant[NFS_MAXASYNCDAEMON]; 6241902Smckusick extern struct nfsreq nfsreqh; 6352196Smckusick extern int nqnfs_piggy[NFS_NPROCS]; 6452196Smckusick extern struct nfsrtt nfsrtt; 6552196Smckusick extern time_t nqnfsstarttime; 6652196Smckusick extern u_long nqnfs_prog, nqnfs_vers; 6752196Smckusick extern int nqsrv_clockskew; 6852196Smckusick extern int nqsrv_writeslack; 6952196Smckusick extern int nqsrv_maxlease; 7038420Smckusick 7138420Smckusick /* 7238420Smckusick * Create the header for an rpc request packet 7338420Smckusick * The hsiz is the size of the rest of the nfs request header. 7438420Smckusick * (just used to decide if a cluster is a good idea) 7538420Smckusick */ 7652196Smckusick struct mbuf * 7752196Smckusick nfsm_reqh(vp, procid, hsiz, bposp) 7852196Smckusick struct vnode *vp; 7939494Smckusick u_long procid; 8038420Smckusick int hsiz; 8152196Smckusick caddr_t *bposp; 8238420Smckusick { 8352196Smckusick register struct mbuf *mb; 8448049Smckusick register u_long *tl; 8552196Smckusick register caddr_t bpos; 8652196Smckusick struct mbuf *mb2; 8752196Smckusick struct nfsmount *nmp; 8852196Smckusick int nqflag; 8938420Smckusick 9052196Smckusick MGET(mb, M_WAIT, MT_DATA); 9152196Smckusick if (hsiz >= MINCLSIZE) 9252196Smckusick MCLGET(mb, M_WAIT); 9352196Smckusick mb->m_len = 0; 9452196Smckusick bpos = mtod(mb, caddr_t); 9552196Smckusick 9652196Smckusick /* 9752196Smckusick * For NQNFS, add lease request. 9852196Smckusick */ 9952196Smckusick if (vp) { 10052196Smckusick nmp = VFSTONFS(vp->v_mount); 10152196Smckusick if (nmp->nm_flag & NFSMNT_NQNFS) { 10252196Smckusick nqflag = NQNFS_NEEDLEASE(vp, procid); 10352196Smckusick if (nqflag) { 10452196Smckusick nfsm_build(tl, u_long *, 2*NFSX_UNSIGNED); 10552196Smckusick *tl++ = txdr_unsigned(nqflag); 10652196Smckusick *tl = txdr_unsigned(nmp->nm_leaseterm); 10752196Smckusick } else { 10852196Smckusick nfsm_build(tl, u_long *, NFSX_UNSIGNED); 10952196Smckusick *tl = 0; 11052196Smckusick } 11152196Smckusick } 11252196Smckusick } 11352196Smckusick /* Finally, return values */ 11452196Smckusick *bposp = bpos; 11552196Smckusick return (mb); 11652196Smckusick } 11738420Smckusick 11852196Smckusick /* 11952196Smckusick * Build the RPC header and fill in the authorization info. 12052196Smckusick * The authorization string argument is only used when the credentials 12152196Smckusick * come from outside of the kernel. 12252196Smckusick * Returns the head of the mbuf list. 12352196Smckusick */ 12452196Smckusick struct mbuf * 12552196Smckusick nfsm_rpchead(cr, nqnfs, procid, auth_type, auth_len, auth_str, mrest, 12652196Smckusick mrest_len, mbp, xidp) 12752196Smckusick register struct ucred *cr; 12852196Smckusick int nqnfs; 12952196Smckusick int procid; 13052196Smckusick int auth_type; 13152196Smckusick int auth_len; 13252196Smckusick char *auth_str; 13352196Smckusick struct mbuf *mrest; 13452196Smckusick int mrest_len; 13552196Smckusick struct mbuf **mbp; 13652196Smckusick u_long *xidp; 13752196Smckusick { 13852196Smckusick register struct mbuf *mb; 13952196Smckusick register u_long *tl; 14052196Smckusick register caddr_t bpos; 14152196Smckusick register int i; 14252196Smckusick struct mbuf *mreq, *mb2; 14352196Smckusick int siz, grpsiz, authsiz; 14452196Smckusick 14552196Smckusick authsiz = nfsm_rndup(auth_len); 14652196Smckusick if (auth_type == RPCAUTH_NQNFS) 14752196Smckusick authsiz += 2 * NFSX_UNSIGNED; 14852196Smckusick MGETHDR(mb, M_WAIT, MT_DATA); 14952196Smckusick if ((authsiz + 10*NFSX_UNSIGNED) >= MINCLSIZE) { 15052196Smckusick MCLGET(mb, M_WAIT); 15152196Smckusick } else if ((authsiz + 10*NFSX_UNSIGNED) < MHLEN) { 15252196Smckusick MH_ALIGN(mb, authsiz + 10*NFSX_UNSIGNED); 15352196Smckusick } else { 15452196Smckusick MH_ALIGN(mb, 8*NFSX_UNSIGNED); 15552196Smckusick } 15652196Smckusick mb->m_len = 0; 15752196Smckusick mreq = mb; 15852196Smckusick bpos = mtod(mb, caddr_t); 15952196Smckusick 16038420Smckusick /* 16152196Smckusick * First the RPC header. 16238420Smckusick */ 16352196Smckusick nfsm_build(tl, u_long *, 8*NFSX_UNSIGNED); 16452196Smckusick if (++nfs_xid == 0) 16552196Smckusick nfs_xid++; 16652196Smckusick *tl++ = *xidp = txdr_unsigned(nfs_xid); 16748049Smckusick *tl++ = rpc_call; 16848049Smckusick *tl++ = rpc_vers; 16952196Smckusick if (nqnfs) { 17052196Smckusick *tl++ = txdr_unsigned(NQNFS_PROG); 17152196Smckusick *tl++ = txdr_unsigned(NQNFS_VER1); 17252196Smckusick } else { 17352196Smckusick *tl++ = txdr_unsigned(NFS_PROG); 17452196Smckusick *tl++ = txdr_unsigned(NFS_VER2); 17552196Smckusick } 17652196Smckusick *tl++ = txdr_unsigned(procid); 17738420Smckusick 17852196Smckusick /* 17952196Smckusick * And then the authorization cred. 18052196Smckusick */ 18152196Smckusick *tl++ = txdr_unsigned(auth_type); 18252196Smckusick *tl = txdr_unsigned(authsiz); 18352196Smckusick switch (auth_type) { 18452196Smckusick case RPCAUTH_UNIX: 18552196Smckusick nfsm_build(tl, u_long *, auth_len); 18652196Smckusick *tl++ = 0; /* stamp ?? */ 18752196Smckusick *tl++ = 0; /* NULL hostname */ 18852196Smckusick *tl++ = txdr_unsigned(cr->cr_uid); 18952196Smckusick *tl++ = txdr_unsigned(cr->cr_groups[0]); 19052196Smckusick grpsiz = (auth_len >> 2) - 5; 19152196Smckusick *tl++ = txdr_unsigned(grpsiz); 19252196Smckusick for (i = 1; i <= grpsiz; i++) 19352196Smckusick *tl++ = txdr_unsigned(cr->cr_groups[i]); 19452196Smckusick break; 19552196Smckusick case RPCAUTH_NQNFS: 19652196Smckusick nfsm_build(tl, u_long *, 2*NFSX_UNSIGNED); 19752196Smckusick *tl++ = txdr_unsigned(cr->cr_uid); 19852196Smckusick *tl = txdr_unsigned(auth_len); 19952196Smckusick siz = auth_len; 20052196Smckusick while (siz > 0) { 20152196Smckusick if (M_TRAILINGSPACE(mb) == 0) { 20252196Smckusick MGET(mb2, M_WAIT, MT_DATA); 20352196Smckusick if (siz >= MINCLSIZE) 20452196Smckusick MCLGET(mb2, M_WAIT); 20552196Smckusick mb->m_next = mb2; 20652196Smckusick mb = mb2; 20752196Smckusick mb->m_len = 0; 20852196Smckusick bpos = mtod(mb, caddr_t); 20952196Smckusick } 21055057Spendry i = min(siz, M_TRAILINGSPACE(mb)); 21152196Smckusick bcopy(auth_str, bpos, i); 21252196Smckusick mb->m_len += i; 21352196Smckusick auth_str += i; 21452196Smckusick bpos += i; 21552196Smckusick siz -= i; 21638420Smckusick } 21752196Smckusick if ((siz = nfsm_rndup(auth_len) - auth_len) > 0) { 21852196Smckusick for (i = 0; i < siz; i++) 21952196Smckusick *bpos++ = '\0'; 22052196Smckusick mb->m_len += siz; 22152196Smckusick } 22252196Smckusick break; 22352196Smckusick }; 22452196Smckusick nfsm_build(tl, u_long *, 2*NFSX_UNSIGNED); 22552196Smckusick *tl++ = txdr_unsigned(RPCAUTH_NULL); 22652196Smckusick *tl = 0; 22752196Smckusick mb->m_next = mrest; 22852196Smckusick mreq->m_pkthdr.len = authsiz + 10*NFSX_UNSIGNED + mrest_len; 22952196Smckusick mreq->m_pkthdr.rcvif = (struct ifnet *)0; 23052196Smckusick *mbp = mb; 23138420Smckusick return (mreq); 23238420Smckusick } 23338420Smckusick 23438420Smckusick /* 23538420Smckusick * copies mbuf chain to the uio scatter/gather list 23638420Smckusick */ 23738420Smckusick nfsm_mbuftouio(mrep, uiop, siz, dpos) 23838420Smckusick struct mbuf **mrep; 23943353Smckusick register struct uio *uiop; 24038420Smckusick int siz; 24138420Smckusick caddr_t *dpos; 24238420Smckusick { 24343353Smckusick register char *mbufcp, *uiocp; 24438420Smckusick register int xfer, left, len; 24538420Smckusick register struct mbuf *mp; 24638420Smckusick long uiosiz, rem; 24741902Smckusick int error = 0; 24838420Smckusick 24938420Smckusick mp = *mrep; 25038420Smckusick mbufcp = *dpos; 25138420Smckusick len = mtod(mp, caddr_t)+mp->m_len-mbufcp; 25238420Smckusick rem = nfsm_rndup(siz)-siz; 25338420Smckusick while (siz > 0) { 25438420Smckusick if (uiop->uio_iovcnt <= 0 || uiop->uio_iov == NULL) 25541902Smckusick return (EFBIG); 25638420Smckusick left = uiop->uio_iov->iov_len; 25738420Smckusick uiocp = uiop->uio_iov->iov_base; 25838420Smckusick if (left > siz) 25938420Smckusick left = siz; 26038420Smckusick uiosiz = left; 26138420Smckusick while (left > 0) { 26238420Smckusick while (len == 0) { 26338420Smckusick mp = mp->m_next; 26438420Smckusick if (mp == NULL) 26538420Smckusick return (EBADRPC); 26638420Smckusick mbufcp = mtod(mp, caddr_t); 26738420Smckusick len = mp->m_len; 26838420Smckusick } 26938420Smckusick xfer = (left > len) ? len : left; 27038420Smckusick #ifdef notdef 27138420Smckusick /* Not Yet.. */ 27238420Smckusick if (uiop->uio_iov->iov_op != NULL) 27338420Smckusick (*(uiop->uio_iov->iov_op)) 27438420Smckusick (mbufcp, uiocp, xfer); 27538420Smckusick else 27638420Smckusick #endif 27738420Smckusick if (uiop->uio_segflg == UIO_SYSSPACE) 27838420Smckusick bcopy(mbufcp, uiocp, xfer); 27938420Smckusick else 28038420Smckusick copyout(mbufcp, uiocp, xfer); 28138420Smckusick left -= xfer; 28238420Smckusick len -= xfer; 28338420Smckusick mbufcp += xfer; 28438420Smckusick uiocp += xfer; 28539585Smckusick uiop->uio_offset += xfer; 28638420Smckusick uiop->uio_resid -= xfer; 28738420Smckusick } 28838420Smckusick if (uiop->uio_iov->iov_len <= siz) { 28938420Smckusick uiop->uio_iovcnt--; 29038420Smckusick uiop->uio_iov++; 29138420Smckusick } else { 29238420Smckusick uiop->uio_iov->iov_base += uiosiz; 29338420Smckusick uiop->uio_iov->iov_len -= uiosiz; 29438420Smckusick } 29538420Smckusick siz -= uiosiz; 29638420Smckusick } 29738420Smckusick *dpos = mbufcp; 29838420Smckusick *mrep = mp; 29941902Smckusick if (rem > 0) { 30041902Smckusick if (len < rem) 30141902Smckusick error = nfs_adv(mrep, dpos, rem, len); 30241902Smckusick else 30341902Smckusick *dpos += rem; 30441902Smckusick } 30541902Smckusick return (error); 30638420Smckusick } 30738420Smckusick 30838420Smckusick /* 30938420Smckusick * copies a uio scatter/gather list to an mbuf chain... 31038420Smckusick */ 31138420Smckusick nfsm_uiotombuf(uiop, mq, siz, bpos) 31238420Smckusick register struct uio *uiop; 31338420Smckusick struct mbuf **mq; 31438420Smckusick int siz; 31538420Smckusick caddr_t *bpos; 31638420Smckusick { 31743353Smckusick register char *uiocp; 31843353Smckusick register struct mbuf *mp, *mp2; 31952196Smckusick register int xfer, left, mlen; 32043353Smckusick int uiosiz, clflg, rem; 32143353Smckusick char *cp; 32238420Smckusick 32338420Smckusick if (siz > MLEN) /* or should it >= MCLBYTES ?? */ 32438420Smckusick clflg = 1; 32538420Smckusick else 32638420Smckusick clflg = 0; 32738420Smckusick rem = nfsm_rndup(siz)-siz; 32852196Smckusick mp = mp2 = *mq; 32938420Smckusick while (siz > 0) { 33038420Smckusick if (uiop->uio_iovcnt <= 0 || uiop->uio_iov == NULL) 33141902Smckusick return (EINVAL); 33238420Smckusick left = uiop->uio_iov->iov_len; 33338420Smckusick uiocp = uiop->uio_iov->iov_base; 33438420Smckusick if (left > siz) 33538420Smckusick left = siz; 33638420Smckusick uiosiz = left; 33738420Smckusick while (left > 0) { 33852196Smckusick mlen = M_TRAILINGSPACE(mp); 33952196Smckusick if (mlen == 0) { 34052196Smckusick MGET(mp, M_WAIT, MT_DATA); 34152196Smckusick if (clflg) 34252196Smckusick MCLGET(mp, M_WAIT); 34352196Smckusick mp->m_len = 0; 34452196Smckusick mp2->m_next = mp; 34552196Smckusick mp2 = mp; 34652196Smckusick mlen = M_TRAILINGSPACE(mp); 34752196Smckusick } 34852196Smckusick xfer = (left > mlen) ? mlen : left; 34938420Smckusick #ifdef notdef 35038420Smckusick /* Not Yet.. */ 35138420Smckusick if (uiop->uio_iov->iov_op != NULL) 35238420Smckusick (*(uiop->uio_iov->iov_op)) 35352196Smckusick (uiocp, mtod(mp, caddr_t)+mp->m_len, xfer); 35438420Smckusick else 35538420Smckusick #endif 35638420Smckusick if (uiop->uio_segflg == UIO_SYSSPACE) 35752196Smckusick bcopy(uiocp, mtod(mp, caddr_t)+mp->m_len, xfer); 35838420Smckusick else 35952196Smckusick copyin(uiocp, mtod(mp, caddr_t)+mp->m_len, xfer); 36052196Smckusick mp->m_len += xfer; 36138420Smckusick left -= xfer; 36238420Smckusick uiocp += xfer; 36339585Smckusick uiop->uio_offset += xfer; 36438420Smckusick uiop->uio_resid -= xfer; 36538420Smckusick } 36638420Smckusick if (uiop->uio_iov->iov_len <= siz) { 36738420Smckusick uiop->uio_iovcnt--; 36838420Smckusick uiop->uio_iov++; 36938420Smckusick } else { 37038420Smckusick uiop->uio_iov->iov_base += uiosiz; 37138420Smckusick uiop->uio_iov->iov_len -= uiosiz; 37238420Smckusick } 37338420Smckusick siz -= uiosiz; 37438420Smckusick } 37538420Smckusick if (rem > 0) { 37652196Smckusick if (rem > M_TRAILINGSPACE(mp)) { 37738420Smckusick MGET(mp, M_WAIT, MT_DATA); 37838420Smckusick mp->m_len = 0; 37938420Smckusick mp2->m_next = mp; 38038420Smckusick } 38138420Smckusick cp = mtod(mp, caddr_t)+mp->m_len; 38238420Smckusick for (left = 0; left < rem; left++) 38338420Smckusick *cp++ = '\0'; 38438420Smckusick mp->m_len += rem; 38538420Smckusick *bpos = cp; 38638420Smckusick } else 38738420Smckusick *bpos = mtod(mp, caddr_t)+mp->m_len; 38838420Smckusick *mq = mp; 38941902Smckusick return (0); 39038420Smckusick } 39138420Smckusick 39238420Smckusick /* 39338420Smckusick * Help break down an mbuf chain by setting the first siz bytes contiguous 39438420Smckusick * pointed to by returned val. 39538420Smckusick * If Updateflg == True we can overwrite the first part of the mbuf data 39652196Smckusick * (in this case it can never sleep, so it can be called from interrupt level) 39752196Smckusick * it may however block when Updateflg == False 39852196Smckusick * This is used by the macros nfsm_dissect and nfsm_dissecton for tough 39938420Smckusick * cases. (The macros use the vars. dpos and dpos2) 40038420Smckusick */ 40138420Smckusick nfsm_disct(mdp, dposp, siz, left, updateflg, cp2) 40238420Smckusick struct mbuf **mdp; 40338420Smckusick caddr_t *dposp; 40438420Smckusick int siz; 40538420Smckusick int left; 40638420Smckusick int updateflg; 40738420Smckusick caddr_t *cp2; 40838420Smckusick { 40938420Smckusick register struct mbuf *mp, *mp2; 41038420Smckusick register int siz2, xfer; 41152196Smckusick register caddr_t p; 41238420Smckusick 41338420Smckusick mp = *mdp; 41438420Smckusick while (left == 0) { 41538420Smckusick *mdp = mp = mp->m_next; 41638420Smckusick if (mp == NULL) 41741902Smckusick return (EBADRPC); 41838420Smckusick left = mp->m_len; 41938420Smckusick *dposp = mtod(mp, caddr_t); 42038420Smckusick } 42138420Smckusick if (left >= siz) { 42238420Smckusick *cp2 = *dposp; 42338420Smckusick *dposp += siz; 42438420Smckusick } else if (mp->m_next == NULL) { 42541902Smckusick return (EBADRPC); 42641902Smckusick } else if (siz > MHLEN) { 42738420Smckusick panic("nfs S too big"); 42838420Smckusick } else { 42938420Smckusick /* Iff update, you can overwrite, else must alloc new mbuf */ 43038420Smckusick if (updateflg) { 43138420Smckusick NFSMINOFF(mp); 43238420Smckusick } else { 43338420Smckusick MGET(mp2, M_WAIT, MT_DATA); 43438420Smckusick mp2->m_next = mp->m_next; 43538420Smckusick mp->m_next = mp2; 43638420Smckusick mp->m_len -= left; 43738420Smckusick mp = mp2; 43838420Smckusick } 43952196Smckusick *cp2 = p = mtod(mp, caddr_t); 44052196Smckusick bcopy(*dposp, p, left); /* Copy what was left */ 44138420Smckusick siz2 = siz-left; 44252196Smckusick p += left; 44338420Smckusick mp2 = mp->m_next; 44441902Smckusick /* Loop around copying up the siz2 bytes */ 44538420Smckusick while (siz2 > 0) { 44638420Smckusick if (mp2 == NULL) 44738420Smckusick return (EBADRPC); 44838420Smckusick xfer = (siz2 > mp2->m_len) ? mp2->m_len : siz2; 44941902Smckusick if (xfer > 0) { 45052196Smckusick bcopy(mtod(mp2, caddr_t), p, xfer); 45141902Smckusick NFSMADV(mp2, xfer); 45241902Smckusick mp2->m_len -= xfer; 45352196Smckusick p += xfer; 45441902Smckusick siz2 -= xfer; 45541902Smckusick } 45638420Smckusick if (siz2 > 0) 45738420Smckusick mp2 = mp2->m_next; 45838420Smckusick } 45938420Smckusick mp->m_len = siz; 46038420Smckusick *mdp = mp2; 46138420Smckusick *dposp = mtod(mp2, caddr_t); 46238420Smckusick } 46339494Smckusick return (0); 46438420Smckusick } 46538420Smckusick 46638420Smckusick /* 46741902Smckusick * Advance the position in the mbuf chain. 46838420Smckusick */ 46938420Smckusick nfs_adv(mdp, dposp, offs, left) 47038420Smckusick struct mbuf **mdp; 47138420Smckusick caddr_t *dposp; 47238420Smckusick int offs; 47338420Smckusick int left; 47438420Smckusick { 47538420Smckusick register struct mbuf *m; 47638420Smckusick register int s; 47738420Smckusick 47838420Smckusick m = *mdp; 47938420Smckusick s = left; 48038420Smckusick while (s < offs) { 48138420Smckusick offs -= s; 48238420Smckusick m = m->m_next; 48338420Smckusick if (m == NULL) 48441902Smckusick return (EBADRPC); 48538420Smckusick s = m->m_len; 48638420Smckusick } 48738420Smckusick *mdp = m; 48838420Smckusick *dposp = mtod(m, caddr_t)+offs; 48941902Smckusick return (0); 49038420Smckusick } 49138420Smckusick 49238420Smckusick /* 49338420Smckusick * Copy a string into mbufs for the hard cases... 49438420Smckusick */ 49538420Smckusick nfsm_strtmbuf(mb, bpos, cp, siz) 49638420Smckusick struct mbuf **mb; 49738420Smckusick char **bpos; 49838420Smckusick char *cp; 49938420Smckusick long siz; 50038420Smckusick { 50138420Smckusick register struct mbuf *m1, *m2; 50238420Smckusick long left, xfer, len, tlen; 50348049Smckusick u_long *tl; 50438420Smckusick int putsize; 50538420Smckusick 50638420Smckusick putsize = 1; 50738420Smckusick m2 = *mb; 50852196Smckusick left = M_TRAILINGSPACE(m2); 50938420Smckusick if (left > 0) { 51048049Smckusick tl = ((u_long *)(*bpos)); 51148049Smckusick *tl++ = txdr_unsigned(siz); 51238420Smckusick putsize = 0; 51338420Smckusick left -= NFSX_UNSIGNED; 51438420Smckusick m2->m_len += NFSX_UNSIGNED; 51538420Smckusick if (left > 0) { 51648049Smckusick bcopy(cp, (caddr_t) tl, left); 51738420Smckusick siz -= left; 51838420Smckusick cp += left; 51938420Smckusick m2->m_len += left; 52038420Smckusick left = 0; 52138420Smckusick } 52238420Smckusick } 52352196Smckusick /* Loop around adding mbufs */ 52438420Smckusick while (siz > 0) { 52538420Smckusick MGET(m1, M_WAIT, MT_DATA); 52638420Smckusick if (siz > MLEN) 52741902Smckusick MCLGET(m1, M_WAIT); 52838420Smckusick m1->m_len = NFSMSIZ(m1); 52938420Smckusick m2->m_next = m1; 53038420Smckusick m2 = m1; 53148049Smckusick tl = mtod(m1, u_long *); 53238420Smckusick tlen = 0; 53338420Smckusick if (putsize) { 53448049Smckusick *tl++ = txdr_unsigned(siz); 53538420Smckusick m1->m_len -= NFSX_UNSIGNED; 53638420Smckusick tlen = NFSX_UNSIGNED; 53738420Smckusick putsize = 0; 53838420Smckusick } 53938420Smckusick if (siz < m1->m_len) { 54038420Smckusick len = nfsm_rndup(siz); 54138420Smckusick xfer = siz; 54238420Smckusick if (xfer < len) 54348049Smckusick *(tl+(xfer>>2)) = 0; 54438420Smckusick } else { 54538420Smckusick xfer = len = m1->m_len; 54638420Smckusick } 54748049Smckusick bcopy(cp, (caddr_t) tl, xfer); 54838420Smckusick m1->m_len = len+tlen; 54938420Smckusick siz -= xfer; 55038420Smckusick cp += xfer; 55138420Smckusick } 55238420Smckusick *mb = m1; 55338420Smckusick *bpos = mtod(m1, caddr_t)+m1->m_len; 55441902Smckusick return (0); 55538420Smckusick } 55638420Smckusick 55738420Smckusick /* 55838420Smckusick * Called once to initialize data structures... 55938420Smckusick */ 56039444Smckusick nfs_init() 56138420Smckusick { 56238420Smckusick register int i; 56352196Smckusick union nqsrvthead *lhp; 56438420Smckusick 56552196Smckusick nfsrtt.pos = 0; 56638420Smckusick rpc_vers = txdr_unsigned(RPC_VER2); 56738420Smckusick rpc_call = txdr_unsigned(RPC_CALL); 56838420Smckusick rpc_reply = txdr_unsigned(RPC_REPLY); 56938420Smckusick rpc_msgdenied = txdr_unsigned(RPC_MSGDENIED); 57038420Smckusick rpc_msgaccepted = txdr_unsigned(RPC_MSGACCEPTED); 57138420Smckusick rpc_mismatch = txdr_unsigned(RPC_MISMATCH); 57252196Smckusick rpc_autherr = txdr_unsigned(RPC_AUTHERR); 57352196Smckusick rpc_rejectedcred = txdr_unsigned(AUTH_REJECTCRED); 57438420Smckusick rpc_auth_unix = txdr_unsigned(RPCAUTH_UNIX); 57552196Smckusick rpc_auth_kerb = txdr_unsigned(RPCAUTH_NQNFS); 57638420Smckusick nfs_vers = txdr_unsigned(NFS_VER2); 57738420Smckusick nfs_prog = txdr_unsigned(NFS_PROG); 57838420Smckusick nfs_true = txdr_unsigned(TRUE); 57938420Smckusick nfs_false = txdr_unsigned(FALSE); 58038420Smckusick /* Loop thru nfs procids */ 58138420Smckusick for (i = 0; i < NFS_NPROCS; i++) 58238420Smckusick nfs_procids[i] = txdr_unsigned(i); 58339345Smckusick /* Ensure async daemons disabled */ 58441902Smckusick for (i = 0; i < NFS_MAXASYNCDAEMON; i++) 58539345Smckusick nfs_iodwant[i] = (struct proc *)0; 58638420Smckusick nfs_xdrneg1 = txdr_unsigned(-1); 58738420Smckusick nfs_nhinit(); /* Init the nfsnode table */ 58852979Smckusick nfsrv_init(0); /* Init server data structures */ 58939755Smckusick nfsrv_initcache(); /* Init the server request cache */ 59041902Smckusick 59141902Smckusick /* 59252196Smckusick * Initialize the nqnfs server stuff. 59352196Smckusick */ 59452196Smckusick if (nqnfsstarttime == 0) { 59552196Smckusick nqnfsstarttime = boottime.tv_sec + nqsrv_maxlease 59652196Smckusick + nqsrv_clockskew + nqsrv_writeslack; 59752196Smckusick NQLOADNOVRAM(nqnfsstarttime); 59852196Smckusick nqnfs_prog = txdr_unsigned(NQNFS_PROG); 59952196Smckusick nqnfs_vers = txdr_unsigned(NQNFS_VER1); 60052196Smckusick nqthead.th_head[0] = &nqthead; 60152196Smckusick nqthead.th_head[1] = &nqthead; 602*55519Smckusick nqfhead = hashinit(NQLCHSZ, M_NQLEASE, &nqfheadhash); 60352196Smckusick } 60452196Smckusick 60552196Smckusick /* 60641902Smckusick * Initialize reply list and start timer 60741902Smckusick */ 60841902Smckusick nfsreqh.r_prev = nfsreqh.r_next = &nfsreqh; 60938420Smckusick nfs_timer(); 61038420Smckusick } 61138420Smckusick 61238420Smckusick /* 61338420Smckusick * Attribute cache routines. 61438420Smckusick * nfs_loadattrcache() - loads or updates the cache contents from attributes 61538420Smckusick * that are on the mbuf list 61638420Smckusick * nfs_getattrcache() - returns valid attributes if found in cache, returns 61738420Smckusick * error otherwise 61838420Smckusick */ 61938420Smckusick 62038420Smckusick /* 62139444Smckusick * Load the attribute cache (that lives in the nfsnode entry) with 62238420Smckusick * the values on the mbuf list and 62338420Smckusick * Iff vap not NULL 62438420Smckusick * copy the attributes to *vaper 62538420Smckusick */ 62639457Smckusick nfs_loadattrcache(vpp, mdp, dposp, vaper) 62739457Smckusick struct vnode **vpp; 62838420Smckusick struct mbuf **mdp; 62938420Smckusick caddr_t *dposp; 63038420Smckusick struct vattr *vaper; 63138420Smckusick { 63239457Smckusick register struct vnode *vp = *vpp; 63338420Smckusick register struct vattr *vap; 63438884Smacklem register struct nfsv2_fattr *fp; 63553553Sheideman extern int (**spec_nfsv2nodeop_p)(); 63653553Sheideman extern int (**spec_vnodeop_p)(); 637*55519Smckusick register struct nfsnode *np, *nq, **nhpp; 63839494Smckusick register long t1; 63939494Smckusick caddr_t dpos, cp2; 64039494Smckusick int error = 0; 64139494Smckusick struct mbuf *md; 64252196Smckusick enum vtype vtyp; 64352196Smckusick u_short vmode; 64442244Smckusick long rdev; 64539444Smckusick struct timeval mtime; 64639444Smckusick struct vnode *nvp; 64738420Smckusick 64838420Smckusick md = *mdp; 64938420Smckusick dpos = *dposp; 65052196Smckusick t1 = (mtod(md, caddr_t) + md->m_len) - dpos; 65138420Smckusick if (error = nfsm_disct(&md, &dpos, NFSX_FATTR, t1, TRUE, &cp2)) 65238420Smckusick return (error); 65338884Smacklem fp = (struct nfsv2_fattr *)cp2; 65452196Smckusick vtyp = nfstov_type(fp->fa_type); 65552196Smckusick vmode = fxdr_unsigned(u_short, fp->fa_mode); 65653384Smckusick if (vtyp == VNON || vtyp == VREG) 65752196Smckusick vtyp = IFTOVT(vmode); 65842244Smckusick rdev = fxdr_unsigned(long, fp->fa_rdev); 65939444Smckusick fxdr_time(&fp->fa_mtime, &mtime); 66039444Smckusick /* 66139444Smckusick * If v_type == VNON it is a new node, so fill in the v_type, 66239444Smckusick * n_mtime fields. Check to see if it represents a special 66339444Smckusick * device, and if so, check for a possible alias. Once the 66439444Smckusick * correct vnode has been obtained, fill in the rest of the 66539444Smckusick * information. 66639444Smckusick */ 66738420Smckusick np = VTONFS(vp); 66839444Smckusick if (vp->v_type == VNON) { 66952196Smckusick if (vtyp == VCHR && rdev == 0xffffffff) 67052196Smckusick vp->v_type = vtyp = VFIFO; 67142244Smckusick else 67252196Smckusick vp->v_type = vtyp; 67340295Smckusick if (vp->v_type == VFIFO) { 67440295Smckusick #ifdef FIFO 67553553Sheideman extern int (**fifo_nfsv2nodeop_p)(); 67653553Sheideman vp->v_op = fifo_nfsv2nodeop_p; 67740295Smckusick #else 67840295Smckusick return (EOPNOTSUPP); 67940295Smckusick #endif /* FIFO */ 68040295Smckusick } 68139444Smckusick if (vp->v_type == VCHR || vp->v_type == VBLK) { 68253553Sheideman vp->v_op = spec_nfsv2nodeop_p; 68342244Smckusick if (nvp = checkalias(vp, (dev_t)rdev, vp->v_mount)) { 68439444Smckusick /* 68551984Smckusick * Discard unneeded vnode, but save its nfsnode. 68651984Smckusick */ 687*55519Smckusick if (nq = np->n_forw) 688*55519Smckusick nq->n_back = np->n_back; 689*55519Smckusick *np->n_back = nq; 69051984Smckusick nvp->v_data = vp->v_data; 69151984Smckusick vp->v_data = NULL; 69253553Sheideman vp->v_op = spec_vnodeop_p; 69351984Smckusick vrele(vp); 69451984Smckusick vgone(vp); 69551984Smckusick /* 69639444Smckusick * Reinitialize aliased node. 69739444Smckusick */ 69839444Smckusick np->n_vnode = nvp; 699*55519Smckusick nhpp = (struct nfsnode **)nfs_hash(&np->n_fh); 700*55519Smckusick if (nq = *nhpp) 701*55519Smckusick nq->n_back = &np->n_forw; 702*55519Smckusick np->n_forw = nq; 703*55519Smckusick np->n_back = nhpp; 704*55519Smckusick *nhpp = np; 70551984Smckusick *vpp = vp = nvp; 70639444Smckusick } 70739444Smckusick } 70852196Smckusick if ((VFSTONFS(vp->v_mount)->nm_flag & NFSMNT_NQNFS) == 0) 70952196Smckusick np->n_mtime = mtime.tv_sec; 71039444Smckusick } 71138420Smckusick vap = &np->n_vattr; 71252196Smckusick vap->va_type = vtyp; 71352196Smckusick vap->va_mode = (vmode & 07777); 71438884Smacklem vap->va_nlink = fxdr_unsigned(u_short, fp->fa_nlink); 71538884Smacklem vap->va_uid = fxdr_unsigned(uid_t, fp->fa_uid); 71638884Smacklem vap->va_gid = fxdr_unsigned(gid_t, fp->fa_gid); 71738884Smacklem vap->va_size = fxdr_unsigned(u_long, fp->fa_size); 71845716Smckusick if ((np->n_flag & NMODIFIED) == 0 || vap->va_size > np->n_size) { 71938884Smacklem np->n_size = vap->va_size; 72053322Smckusick vnode_pager_setsize(vp, (u_long)np->n_size); 72145716Smckusick } 72238884Smacklem vap->va_blocksize = fxdr_unsigned(long, fp->fa_blocksize); 72342244Smckusick vap->va_rdev = (dev_t)rdev; 72442244Smckusick vap->va_bytes = fxdr_unsigned(long, fp->fa_blocks) * NFS_FABLKSIZE; 72542878Smckusick vap->va_fsid = vp->v_mount->mnt_stat.f_fsid.val[0]; 72638884Smacklem vap->va_fileid = fxdr_unsigned(long, fp->fa_fileid); 72754106Smckusick vap->va_atime.ts_sec = fxdr_unsigned(long, fp->fa_atime.tv_sec); 72854106Smckusick vap->va_atime.ts_nsec = 0; 72939755Smckusick vap->va_flags = fxdr_unsigned(u_long, fp->fa_atime.tv_usec); 73054106Smckusick vap->va_mtime.ts_sec = mtime.tv_sec; 73154106Smckusick vap->va_mtime.ts_nsec = mtime.tv_usec * 1000; 73254106Smckusick vap->va_ctime.ts_sec = fxdr_unsigned(long, fp->fa_ctime.tv_sec); 73354106Smckusick vap->va_ctime.ts_nsec = 0; 73439755Smckusick vap->va_gen = fxdr_unsigned(u_long, fp->fa_ctime.tv_usec); 73538420Smckusick np->n_attrstamp = time.tv_sec; 73638420Smckusick *dposp = dpos; 73738420Smckusick *mdp = md; 73838884Smacklem if (vaper != NULL) { 73938420Smckusick bcopy((caddr_t)vap, (caddr_t)vaper, sizeof(*vap)); 74038884Smacklem if ((np->n_flag & NMODIFIED) && (np->n_size > vap->va_size)) 74138884Smacklem vaper->va_size = np->n_size; 74253628Smckusick if (np->n_flag & NCHG) { 74354106Smckusick if (np->n_flag & NACC) { 74454106Smckusick vaper->va_atime.ts_sec = np->n_atim.tv_sec; 74554106Smckusick vaper->va_atime.ts_nsec = 74654106Smckusick np->n_atim.tv_usec * 1000; 74754106Smckusick } 74854106Smckusick if (np->n_flag & NUPD) { 74954106Smckusick vaper->va_mtime.ts_sec = np->n_mtim.tv_sec; 75054106Smckusick vaper->va_mtime.ts_nsec = 75154106Smckusick np->n_mtim.tv_usec * 1000; 75254106Smckusick } 75353628Smckusick } 75438884Smacklem } 75538420Smckusick return (0); 75638420Smckusick } 75738420Smckusick 75838420Smckusick /* 75938420Smckusick * Check the time stamp 76038420Smckusick * If the cache is valid, copy contents to *vap and return 0 76138420Smckusick * otherwise return an error 76238420Smckusick */ 76338420Smckusick nfs_getattrcache(vp, vap) 76438420Smckusick register struct vnode *vp; 76538420Smckusick struct vattr *vap; 76638420Smckusick { 76738420Smckusick register struct nfsnode *np; 76838420Smckusick 76938420Smckusick np = VTONFS(vp); 77052196Smckusick if (VFSTONFS(vp->v_mount)->nm_flag & NFSMNT_NQNFS) { 77152196Smckusick if (!NQNFS_CKCACHABLE(vp, NQL_READ) || np->n_attrstamp == 0) { 77252196Smckusick nfsstats.attrcache_misses++; 77352196Smckusick return (ENOENT); 77452196Smckusick } 77552196Smckusick } else if ((time.tv_sec - np->n_attrstamp) >= NFS_ATTRTIMEO) { 77638420Smckusick nfsstats.attrcache_misses++; 77738420Smckusick return (ENOENT); 77838420Smckusick } 77952196Smckusick nfsstats.attrcache_hits++; 78052196Smckusick bcopy((caddr_t)&np->n_vattr,(caddr_t)vap,sizeof(struct vattr)); 78152196Smckusick if ((np->n_flag & NMODIFIED) == 0) { 78252196Smckusick np->n_size = vap->va_size; 78353322Smckusick vnode_pager_setsize(vp, (u_long)np->n_size); 78452196Smckusick } else if (np->n_size > vap->va_size) 78552196Smckusick vap->va_size = np->n_size; 78653628Smckusick if (np->n_flag & NCHG) { 78754106Smckusick if (np->n_flag & NACC) { 78854106Smckusick vap->va_atime.ts_sec = np->n_atim.tv_sec; 78954106Smckusick vap->va_atime.ts_nsec = np->n_atim.tv_usec * 1000; 79054106Smckusick } 79154106Smckusick if (np->n_flag & NUPD) { 79254106Smckusick vap->va_mtime.ts_sec = np->n_mtim.tv_sec; 79354106Smckusick vap->va_mtime.ts_nsec = np->n_mtim.tv_usec * 1000; 79454106Smckusick } 79553628Smckusick } 79652196Smckusick return (0); 79738420Smckusick } 79838420Smckusick 79938420Smckusick /* 80052196Smckusick * Set up nameidata for a lookup() call and do it 80138420Smckusick */ 80252196Smckusick nfs_namei(ndp, fhp, len, slp, nam, mdp, dposp, p) 80338420Smckusick register struct nameidata *ndp; 80438420Smckusick fhandle_t *fhp; 80538420Smckusick int len; 80652196Smckusick struct nfssvc_sock *slp; 80752196Smckusick struct mbuf *nam; 80838420Smckusick struct mbuf **mdp; 80938420Smckusick caddr_t *dposp; 81049739Smckusick struct proc *p; 81138420Smckusick { 81238420Smckusick register int i, rem; 81338420Smckusick register struct mbuf *md; 81449739Smckusick register char *fromcp, *tocp; 81546514Smckusick struct vnode *dp; 81652315Sheideman int error, rdonly; 81752315Sheideman struct componentname *cnp = &ndp->ni_cnd; 81838420Smckusick 81952315Sheideman MALLOC(cnp->cn_pnbuf, char *, len + 1, M_NAMEI, M_WAITOK); 82049739Smckusick /* 82149739Smckusick * Copy the name from the mbuf list to ndp->ni_pnbuf 82249739Smckusick * and set the various ndp fields appropriately. 82349739Smckusick */ 82449739Smckusick fromcp = *dposp; 82552315Sheideman tocp = cnp->cn_pnbuf; 82649739Smckusick md = *mdp; 82749739Smckusick rem = mtod(md, caddr_t) + md->m_len - fromcp; 82852315Sheideman cnp->cn_hash = 0; 82949739Smckusick for (i = 0; i < len; i++) { 83049739Smckusick while (rem == 0) { 83149739Smckusick md = md->m_next; 83249739Smckusick if (md == NULL) { 83349739Smckusick error = EBADRPC; 83449739Smckusick goto out; 83542244Smckusick } 83649739Smckusick fromcp = mtod(md, caddr_t); 83749739Smckusick rem = md->m_len; 83838420Smckusick } 83949739Smckusick if (*fromcp == '\0' || *fromcp == '/') { 84049739Smckusick error = EINVAL; 84149739Smckusick goto out; 84242244Smckusick } 84349739Smckusick if (*fromcp & 0200) 84452315Sheideman if ((*fromcp&0377) == ('/'|0200) || cnp->cn_nameiop != DELETE) { 84549739Smckusick error = EINVAL; 84649739Smckusick goto out; 84749739Smckusick } 84852315Sheideman cnp->cn_hash += (unsigned char)*fromcp; 84949739Smckusick *tocp++ = *fromcp++; 85049739Smckusick rem--; 85149739Smckusick } 85249739Smckusick *tocp = '\0'; 85349739Smckusick *mdp = md; 85449739Smckusick *dposp = fromcp; 85549739Smckusick len = nfsm_rndup(len)-len; 85649739Smckusick if (len > 0) { 85749739Smckusick if (rem >= len) 85849739Smckusick *dposp += len; 85949739Smckusick else if (error = nfs_adv(mdp, dposp, len, rem)) 86049739Smckusick goto out; 86149739Smckusick } 86252315Sheideman ndp->ni_pathlen = tocp - cnp->cn_pnbuf; 86352315Sheideman cnp->cn_nameptr = cnp->cn_pnbuf; 86446514Smckusick /* 86546514Smckusick * Extract and set starting directory. 86646514Smckusick */ 86752315Sheideman if (error = nfsrv_fhtovp(fhp, FALSE, &dp, ndp->ni_cnd.cn_cred, slp, 86852315Sheideman nam, &rdonly)) 86949739Smckusick goto out; 87038425Smckusick if (dp->v_type != VDIR) { 87141902Smckusick vrele(dp); 87249739Smckusick error = ENOTDIR; 87349739Smckusick goto out; 87438425Smckusick } 87546514Smckusick ndp->ni_startdir = dp; 87652196Smckusick if (rdonly) 87752315Sheideman cnp->cn_flags |= (NOCROSSMOUNT | RDONLY); 87852196Smckusick else 87952315Sheideman cnp->cn_flags |= NOCROSSMOUNT; 88039345Smckusick /* 88149739Smckusick * And call lookup() to do the real work 88238420Smckusick */ 88352315Sheideman cnp->cn_proc = p; 88452315Sheideman if (error = lookup(ndp)) 88549739Smckusick goto out; 88649739Smckusick /* 88749739Smckusick * Check for encountering a symbolic link 88849739Smckusick */ 88952315Sheideman if (cnp->cn_flags & ISSYMLINK) { 89052315Sheideman if ((cnp->cn_flags & LOCKPARENT) && ndp->ni_pathlen == 1) 89149739Smckusick vput(ndp->ni_dvp); 89249739Smckusick else 89349739Smckusick vrele(ndp->ni_dvp); 89449739Smckusick vput(ndp->ni_vp); 89549739Smckusick ndp->ni_vp = NULL; 89649739Smckusick error = EINVAL; 89749739Smckusick goto out; 89849739Smckusick } 89949739Smckusick /* 90049739Smckusick * Check for saved name request 90149739Smckusick */ 90252315Sheideman if (cnp->cn_flags & (SAVENAME | SAVESTART)) { 90352315Sheideman cnp->cn_flags |= HASBUF; 90449739Smckusick return (0); 90549739Smckusick } 90649739Smckusick out: 90752315Sheideman FREE(cnp->cn_pnbuf, M_NAMEI); 90838420Smckusick return (error); 90938420Smckusick } 91038420Smckusick 91138420Smckusick /* 91238420Smckusick * A fiddled version of m_adj() that ensures null fill to a long 91338420Smckusick * boundary and only trims off the back end 91438420Smckusick */ 91552196Smckusick void 91638420Smckusick nfsm_adj(mp, len, nul) 91738420Smckusick struct mbuf *mp; 91838420Smckusick register int len; 91938420Smckusick int nul; 92038420Smckusick { 92138420Smckusick register struct mbuf *m; 92238420Smckusick register int count, i; 92338420Smckusick register char *cp; 92438420Smckusick 92538420Smckusick /* 92638420Smckusick * Trim from tail. Scan the mbuf chain, 92738420Smckusick * calculating its length and finding the last mbuf. 92838420Smckusick * If the adjustment only affects this mbuf, then just 92938420Smckusick * adjust and return. Otherwise, rescan and truncate 93038420Smckusick * after the remaining size. 93138420Smckusick */ 93238420Smckusick count = 0; 93338420Smckusick m = mp; 93438420Smckusick for (;;) { 93538420Smckusick count += m->m_len; 93638420Smckusick if (m->m_next == (struct mbuf *)0) 93738420Smckusick break; 93838420Smckusick m = m->m_next; 93938420Smckusick } 94038579Smckusick if (m->m_len > len) { 94138420Smckusick m->m_len -= len; 94238420Smckusick if (nul > 0) { 94338420Smckusick cp = mtod(m, caddr_t)+m->m_len-nul; 94438420Smckusick for (i = 0; i < nul; i++) 94538420Smckusick *cp++ = '\0'; 94638420Smckusick } 94738420Smckusick return; 94838420Smckusick } 94938420Smckusick count -= len; 95038420Smckusick if (count < 0) 95138420Smckusick count = 0; 95238420Smckusick /* 95338420Smckusick * Correct length for chain is "count". 95438420Smckusick * Find the mbuf with last data, adjust its length, 95538420Smckusick * and toss data from remaining mbufs on chain. 95638420Smckusick */ 95738420Smckusick for (m = mp; m; m = m->m_next) { 95838420Smckusick if (m->m_len >= count) { 95938420Smckusick m->m_len = count; 96038420Smckusick if (nul > 0) { 96138420Smckusick cp = mtod(m, caddr_t)+m->m_len-nul; 96238420Smckusick for (i = 0; i < nul; i++) 96338420Smckusick *cp++ = '\0'; 96438420Smckusick } 96538420Smckusick break; 96638420Smckusick } 96738420Smckusick count -= m->m_len; 96838420Smckusick } 96938420Smckusick while (m = m->m_next) 97038420Smckusick m->m_len = 0; 97138420Smckusick } 97238420Smckusick 97338420Smckusick /* 97438420Smckusick * nfsrv_fhtovp() - convert a fh to a vnode ptr (optionally locked) 97538420Smckusick * - look up fsid in mount list (if not found ret error) 97654739Smckusick * - get vp and export rights by calling VFS_FHTOVP() 97754739Smckusick * - if cred->cr_uid == 0 or MNT_EXPORTANON set it to credanon 97838420Smckusick * - if not lockflag unlock it with VOP_UNLOCK() 97938420Smckusick */ 98052196Smckusick nfsrv_fhtovp(fhp, lockflag, vpp, cred, slp, nam, rdonlyp) 98138420Smckusick fhandle_t *fhp; 98238420Smckusick int lockflag; 98338420Smckusick struct vnode **vpp; 98438420Smckusick struct ucred *cred; 98552196Smckusick struct nfssvc_sock *slp; 98652196Smckusick struct mbuf *nam; 98752196Smckusick int *rdonlyp; 98838420Smckusick { 98938420Smckusick register struct mount *mp; 99052196Smckusick register struct nfsuid *uidp; 99154739Smckusick struct ucred *credanon; 99254739Smckusick int error, exflags; 99338420Smckusick 99452196Smckusick *vpp = (struct vnode *)0; 99538420Smckusick if ((mp = getvfs(&fhp->fh_fsid)) == NULL) 99638420Smckusick return (ESTALE); 99754739Smckusick if (error = VFS_FHTOVP(mp, &fhp->fh_fid, nam, vpp, &exflags, &credanon)) 99854739Smckusick return (error); 99952196Smckusick /* 100052196Smckusick * Check/setup credentials. 100152196Smckusick */ 100254739Smckusick if (exflags & MNT_EXKERB) { 100352196Smckusick uidp = slp->ns_uidh[NUIDHASH(cred->cr_uid)]; 100452196Smckusick while (uidp) { 100552196Smckusick if (uidp->nu_uid == cred->cr_uid) 100652196Smckusick break; 100752196Smckusick uidp = uidp->nu_hnext; 100845282Smckusick } 100952196Smckusick if (uidp) { 101052196Smckusick if (cred->cr_ref != 1) 101152196Smckusick panic("nsrv fhtovp"); 101252196Smckusick *cred = uidp->nu_cr; 101352196Smckusick } else 101452196Smckusick return (NQNFS_AUTHERR); 101554739Smckusick } else if (cred->cr_uid == 0 || (exflags & MNT_EXPORTANON)) 101654739Smckusick *cred = *credanon; 101754739Smckusick if (exflags & MNT_EXRDONLY) 101852196Smckusick *rdonlyp = 1; 101945282Smckusick else 102052196Smckusick *rdonlyp = 0; 102152196Smckusick if (!lockflag) 102252196Smckusick VOP_UNLOCK(*vpp); 102352196Smckusick return (0); 102445282Smckusick } 102554988Smckusick 102654988Smckusick /* 102754988Smckusick * This function compares two net addresses by family and returns TRUE 102854988Smckusick * if they are the same host. 102954988Smckusick * If there is any doubt, return FALSE. 103054988Smckusick * The AF_INET family is handled as a special case so that address mbufs 103154988Smckusick * don't need to be saved to store "struct in_addr", which is only 4 bytes. 103254988Smckusick */ 103354988Smckusick netaddr_match(family, haddr, hmask, nam) 103454988Smckusick int family; 103554988Smckusick union nethostaddr *haddr; 103654988Smckusick union nethostaddr *hmask; 103754988Smckusick struct mbuf *nam; 103854988Smckusick { 103954988Smckusick register struct sockaddr_in *inetaddr; 104054988Smckusick #ifdef ISO 104154988Smckusick register struct sockaddr_iso *isoaddr1, *isoaddr2; 104254988Smckusick #endif 104354988Smckusick 104454988Smckusick 104554988Smckusick switch (family) { 104654988Smckusick case AF_INET: 104754988Smckusick inetaddr = mtod(nam, struct sockaddr_in *); 104854988Smckusick if (inetaddr->sin_family != AF_INET) 104954988Smckusick return (0); 105054988Smckusick if (hmask) { 105154988Smckusick if ((inetaddr->sin_addr.s_addr & hmask->had_inetaddr) == 105254988Smckusick (haddr->had_inetaddr & hmask->had_inetaddr)) 105354988Smckusick return (1); 105454988Smckusick } else if (inetaddr->sin_addr.s_addr == haddr->had_inetaddr) 105554988Smckusick return (1); 105654988Smckusick break; 105754988Smckusick #ifdef ISO 105854988Smckusick case AF_ISO: 105954988Smckusick isoaddr1 = mtod(nam, struct sockaddr_iso *); 106054988Smckusick if (isoaddr1->siso_family != AF_ISO) 106154988Smckusick return (0); 106254988Smckusick isoaddr2 = mtod(haddr->had_nam, struct sockaddr_iso *); 106354988Smckusick if (isoaddr1->siso_nlen > 0 && 106454988Smckusick isoaddr1->siso_nlen == isoaddr2->siso_nlen && 106554988Smckusick SAME_ISOADDR(isoaddr1, isoaddr2)) 106654988Smckusick return (1); 106754988Smckusick break; 106854988Smckusick #endif /* ISO */ 106954988Smckusick default: 107054988Smckusick break; 107154988Smckusick }; 107254988Smckusick return (0); 107354988Smckusick } 107454988Smckusick 107554988Smckusick /* 107654988Smckusick * Generate a hash code for an iso host address. Used by NETADDRHASH() for 107754988Smckusick * iso addresses. 107854988Smckusick */ 107954988Smckusick iso_addrhash(saddr) 108054988Smckusick struct sockaddr *saddr; 108154988Smckusick { 108254988Smckusick #ifdef ISO 108354988Smckusick register struct sockaddr_iso *siso; 108454988Smckusick register int i, sum; 108554988Smckusick 108654988Smckusick sum = 0; 108754988Smckusick for (i = 0; i < siso->siso_nlen; i++) 108854988Smckusick sum += siso->siso_data[i]; 108954988Smckusick return (sum & (NETHASHSZ - 1)); 109054988Smckusick #else 109154988Smckusick return (0); 109254988Smckusick #endif /* ISO */ 109354988Smckusick } 1094