Annotation of Net2/nfs/nfs_syscalls.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1989 The Regents of the University of California.
        !             3:  * All rights reserved.
        !             4:  *
        !             5:  * This code is derived from software contributed to Berkeley by
        !             6:  * Rick Macklem at The University of Guelph.
        !             7:  *
        !             8:  * Redistribution and use in source and binary forms, with or without
        !             9:  * modification, are permitted provided that the following conditions
        !            10:  * are met:
        !            11:  * 1. Redistributions of source code must retain the above copyright
        !            12:  *    notice, this list of conditions and the following disclaimer.
        !            13:  * 2. Redistributions in binary form must reproduce the above copyright
        !            14:  *    notice, this list of conditions and the following disclaimer in the
        !            15:  *    documentation and/or other materials provided with the distribution.
        !            16:  * 3. All advertising materials mentioning features or use of this software
        !            17:  *    must display the following acknowledgement:
        !            18:  *     This product includes software developed by the University of
        !            19:  *     California, Berkeley and its contributors.
        !            20:  * 4. Neither the name of the University nor the names of its contributors
        !            21:  *    may be used to endorse or promote products derived from this software
        !            22:  *    without specific prior written permission.
        !            23:  *
        !            24:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
        !            25:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
        !            26:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
        !            27:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
        !            28:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
        !            29:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
        !            30:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
        !            31:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
        !            32:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
        !            33:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
        !            34:  * SUCH DAMAGE.
        !            35:  *
        !            36:  *     @(#)nfs_syscalls.c      7.26 (Berkeley) 4/16/91
        !            37:  */
        !            38: 
        !            39: #include "param.h"
        !            40: #include "systm.h"
        !            41: #include "kernel.h"
        !            42: #include "file.h"
        !            43: #include "stat.h"
        !            44: #include "namei.h"
        !            45: #include "vnode.h"
        !            46: #include "mount.h"
        !            47: #include "proc.h"
        !            48: #include "malloc.h"
        !            49: #include "buf.h"
        !            50: #include "mbuf.h"
        !            51: #include "socket.h"
        !            52: #include "socketvar.h"
        !            53: #include "domain.h"
        !            54: #include "protosw.h"
        !            55: 
        !            56: #include "../netinet/in.h"
        !            57: #include "../netinet/tcp.h"
        !            58: 
        !            59: #include "nfsv2.h"
        !            60: #include "nfs.h"
        !            61: #include "nfsrvcache.h"
        !            62: 
        !            63: /* Global defs. */
        !            64: extern u_long nfs_prog, nfs_vers;
        !            65: extern int (*nfsrv_procs[NFS_NPROCS])();
        !            66: extern struct buf nfs_bqueue;
        !            67: extern int nfs_numasync;
        !            68: extern struct proc *nfs_iodwant[NFS_MAXASYNCDAEMON];
        !            69: extern int nfs_tcpnodelay;
        !            70: struct mbuf *nfs_compress();
        !            71: 
        !            72: #define        TRUE    1
        !            73: #define        FALSE   0
        !            74: 
        !            75: static int nfs_asyncdaemon[NFS_MAXASYNCDAEMON];
        !            76: static int compressreply[NFS_NPROCS] = {
        !            77:        FALSE,
        !            78:        TRUE,
        !            79:        TRUE,
        !            80:        FALSE,
        !            81:        TRUE,
        !            82:        TRUE,
        !            83:        FALSE,
        !            84:        FALSE,
        !            85:        TRUE,
        !            86:        TRUE,
        !            87:        TRUE,
        !            88:        TRUE,
        !            89:        TRUE,
        !            90:        TRUE,
        !            91:        TRUE,
        !            92:        TRUE,
        !            93:        TRUE,
        !            94:        TRUE,
        !            95: };
        !            96: /*
        !            97:  * NFS server system calls
        !            98:  * getfh() lives here too, but maybe should move to kern/vfs_syscalls.c
        !            99:  */
        !           100: 
        !           101: /*
        !           102:  * Get file handle system call
        !           103:  */
        !           104: /* ARGSUSED */
        !           105: getfh(p, uap, retval)
        !           106:        struct proc *p;
        !           107:        register struct args {
        !           108:                char    *fname;
        !           109:                fhandle_t *fhp;
        !           110:        } *uap;
        !           111:        int *retval;
        !           112: {
        !           113:        register struct nameidata *ndp;
        !           114:        register struct vnode *vp;
        !           115:        fhandle_t fh;
        !           116:        int error;
        !           117:        struct nameidata nd;
        !           118: 
        !           119:        /*
        !           120:         * Must be super user
        !           121:         */
        !           122:        if (error = suser(p->p_ucred, &p->p_acflag))
        !           123:                return (error);
        !           124:        ndp = &nd;
        !           125:        ndp->ni_nameiop = LOOKUP | LOCKLEAF | FOLLOW;
        !           126:        ndp->ni_segflg = UIO_USERSPACE;
        !           127:        ndp->ni_dirp = uap->fname;
        !           128:        if (error = namei(ndp, p))
        !           129:                return (error);
        !           130:        vp = ndp->ni_vp;
        !           131:        bzero((caddr_t)&fh, sizeof(fh));
        !           132:        fh.fh_fsid = vp->v_mount->mnt_stat.f_fsid;
        !           133:        error = VFS_VPTOFH(vp, &fh.fh_fid);
        !           134:        vput(vp);
        !           135:        if (error)
        !           136:                return (error);
        !           137:        error = copyout((caddr_t)&fh, (caddr_t)uap->fhp, sizeof (fh));
        !           138:        return (error);
        !           139: }
        !           140: 
        !           141: /*
        !           142:  * Nfs server psuedo system call for the nfsd's
        !           143:  * Never returns unless it fails or gets killed
        !           144:  */
        !           145: /* ARGSUSED */
        !           146: nfssvc(p, uap, retval)
        !           147:        struct proc *p;
        !           148:        register struct args {
        !           149:                int s;
        !           150:                caddr_t mskval;
        !           151:                int msklen;
        !           152:                caddr_t mtchval;
        !           153:                int mtchlen;
        !           154:        } *uap;
        !           155:        int *retval;
        !           156: {
        !           157:        register struct mbuf *m;
        !           158:        register int siz;
        !           159:        register struct ucred *cr;
        !           160:        struct file *fp;
        !           161:        struct mbuf *mreq, *mrep, *nam, *md;
        !           162:        struct mbuf msk, mtch;
        !           163:        struct socket *so;
        !           164:        caddr_t dpos;
        !           165:        int procid, repstat, error, cacherep, wascomp;
        !           166:        u_long retxid;
        !           167: 
        !           168:        /*
        !           169:         * Must be super user
        !           170:         */
        !           171:        if (error = suser(p->p_ucred, &p->p_acflag))
        !           172:                return (error);
        !           173:        if (error = getsock(p->p_fd, uap->s, &fp))
        !           174:                return (error);
        !           175:        so = (struct socket *)fp->f_data;
        !           176:        if (sosendallatonce(so))
        !           177:                siz = NFS_MAXPACKET;
        !           178:        else
        !           179:                siz = NFS_MAXPACKET + sizeof(u_long);
        !           180:        if (error = soreserve(so, siz, siz))
        !           181:                goto bad;
        !           182:        if (error = sockargs(&nam, uap->mskval, uap->msklen, MT_SONAME))
        !           183:                goto bad;
        !           184:        bcopy((caddr_t)nam, (caddr_t)&msk, sizeof (struct mbuf));
        !           185:        msk.m_data = msk.m_dat;
        !           186:        m_freem(nam);
        !           187:        if (error = sockargs(&nam, uap->mtchval, uap->mtchlen, MT_SONAME))
        !           188:                goto bad;
        !           189:        bcopy((caddr_t)nam, (caddr_t)&mtch, sizeof (struct mbuf));
        !           190:        mtch.m_data = mtch.m_dat;
        !           191:        m_freem(nam);
        !           192: 
        !           193:        /* Copy the cred so others don't see changes */
        !           194:        cr = p->p_ucred = crcopy(p->p_ucred);
        !           195: 
        !           196:        /*
        !           197:         * Set protocol specific options { for now TCP only } and
        !           198:         * reserve some space. For datagram sockets, this can get called
        !           199:         * repeatedly for the same socket, but that isn't harmful.
        !           200:         */
        !           201:        if (so->so_proto->pr_flags & PR_CONNREQUIRED) {
        !           202:                MGET(m, M_WAIT, MT_SOOPTS);
        !           203:                *mtod(m, int *) = 1;
        !           204:                m->m_len = sizeof(int);
        !           205:                sosetopt(so, SOL_SOCKET, SO_KEEPALIVE, m);
        !           206:        }
        !           207:        if (so->so_proto->pr_domain->dom_family == AF_INET &&
        !           208:            so->so_proto->pr_protocol == IPPROTO_TCP &&
        !           209:            nfs_tcpnodelay) {
        !           210:                MGET(m, M_WAIT, MT_SOOPTS);
        !           211:                *mtod(m, int *) = 1;
        !           212:                m->m_len = sizeof(int);
        !           213:                sosetopt(so, IPPROTO_TCP, TCP_NODELAY, m);
        !           214:        }
        !           215:        so->so_rcv.sb_flags &= ~SB_NOINTR;
        !           216:        so->so_rcv.sb_timeo = 0;
        !           217:        so->so_snd.sb_flags &= ~SB_NOINTR;
        !           218:        so->so_snd.sb_timeo = 0;
        !           219: 
        !           220:        /*
        !           221:         * Just loop around doin our stuff until SIGKILL
        !           222:         */
        !           223:        for (;;) {
        !           224:                if (error = nfs_getreq(so, nfs_prog, nfs_vers, NFS_NPROCS-1,
        !           225:                   &nam, &mrep, &md, &dpos, &retxid, &procid, cr,
        !           226:                   &msk, &mtch, &wascomp)) {
        !           227:                        if (nam)
        !           228:                                m_freem(nam);
        !           229:                        if (error == EPIPE || error == EINTR ||
        !           230:                            error == ERESTART) {
        !           231:                                error = 0;
        !           232:                                goto bad;
        !           233:                        }
        !           234:                        so->so_error = 0;
        !           235:                        continue;
        !           236:                }
        !           237: 
        !           238:                if (nam)
        !           239:                        cacherep = nfsrv_getcache(nam, retxid, procid, &mreq);
        !           240:                else
        !           241:                        cacherep = RC_DOIT;
        !           242:                switch (cacherep) {
        !           243:                case RC_DOIT:
        !           244:                        if (error = (*(nfsrv_procs[procid]))(mrep, md, dpos,
        !           245:                                cr, retxid, &mreq, &repstat, p)) {
        !           246:                                nfsstats.srv_errs++;
        !           247:                                if (nam) {
        !           248:                                        nfsrv_updatecache(nam, retxid, procid,
        !           249:                                                FALSE, repstat, mreq);
        !           250:                                        m_freem(nam);
        !           251:                                }
        !           252:                                break;
        !           253:                        }
        !           254:                        nfsstats.srvrpccnt[procid]++;
        !           255:                        if (nam)
        !           256:                                nfsrv_updatecache(nam, retxid, procid, TRUE,
        !           257:                                        repstat, mreq);
        !           258:                        mrep = (struct mbuf *)0;
        !           259:                case RC_REPLY:
        !           260:                        m = mreq;
        !           261:                        siz = 0;
        !           262:                        while (m) {
        !           263:                                siz += m->m_len;
        !           264:                                m = m->m_next;
        !           265:                        }
        !           266:                        if (siz <= 0 || siz > NFS_MAXPACKET) {
        !           267:                                printf("mbuf siz=%d\n",siz);
        !           268:                                panic("Bad nfs svc reply");
        !           269:                        }
        !           270:                        mreq->m_pkthdr.len = siz;
        !           271:                        mreq->m_pkthdr.rcvif = (struct ifnet *)0;
        !           272:                        if (wascomp && compressreply[procid]) {
        !           273:                                mreq = nfs_compress(mreq);
        !           274:                                siz = mreq->m_pkthdr.len;
        !           275:                        }
        !           276:                        /*
        !           277:                         * For non-atomic protocols, prepend a Sun RPC
        !           278:                         * Record Mark.
        !           279:                         */
        !           280:                        if (!sosendallatonce(so)) {
        !           281:                                M_PREPEND(mreq, sizeof(u_long), M_WAIT);
        !           282:                                *mtod(mreq, u_long *) = htonl(0x80000000 | siz);
        !           283:                        }
        !           284:                        error = nfs_send(so, nam, mreq, (struct nfsreq *)0);
        !           285:                        if (nam)
        !           286:                                m_freem(nam);
        !           287:                        if (mrep)
        !           288:                                m_freem(mrep);
        !           289:                        if (error) {
        !           290:                                if (error == EPIPE || error == EINTR ||
        !           291:                                    error == ERESTART)
        !           292:                                        goto bad;
        !           293:                                so->so_error = 0;
        !           294:                        }
        !           295:                        break;
        !           296:                case RC_DROPIT:
        !           297:                        m_freem(mrep);
        !           298:                        m_freem(nam);
        !           299:                        break;
        !           300:                };
        !           301:        }
        !           302: bad:
        !           303:        return (error);
        !           304: }
        !           305: 
        !           306: /*
        !           307:  * Nfs pseudo system call for asynchronous i/o daemons.
        !           308:  * These babies just pretend to be disk interrupt service routines
        !           309:  * for client nfs. They are mainly here for read ahead/write behind.
        !           310:  * Never returns unless it fails or gets killed
        !           311:  */
        !           312: /* ARGSUSED */
        !           313: async_daemon(p, uap, retval)
        !           314:        struct proc *p;
        !           315:        struct args *uap;
        !           316:        int *retval;
        !           317: {
        !           318:        register struct buf *bp, *dp;
        !           319:        register int i, myiod;
        !           320:        int error;
        !           321: 
        !           322:        /*
        !           323:         * Must be super user
        !           324:         */
        !           325:        if (error = suser(p->p_ucred, &p->p_acflag))
        !           326:                return (error);
        !           327:        /*
        !           328:         * Assign my position or return error if too many already running
        !           329:         */
        !           330:        myiod = -1;
        !           331:        for (i = 0; i < NFS_MAXASYNCDAEMON; i++)
        !           332:                if (nfs_asyncdaemon[i] == 0) {
        !           333:                        nfs_asyncdaemon[i]++;
        !           334:                        myiod = i;
        !           335:                        break;
        !           336:                }
        !           337:        if (myiod == -1)
        !           338:                return (EBUSY);
        !           339:        nfs_numasync++;
        !           340:        dp = &nfs_bqueue;
        !           341:        /*
        !           342:         * Just loop around doin our stuff until SIGKILL
        !           343:         */
        !           344:        for (;;) {
        !           345:                while (dp->b_actf == NULL && error == 0) {
        !           346:                        nfs_iodwant[myiod] = p;
        !           347:                        error = tsleep((caddr_t)&nfs_iodwant[myiod],
        !           348:                                PWAIT | PCATCH, "nfsidl", 0);
        !           349:                        nfs_iodwant[myiod] = (struct proc *)0;
        !           350:                }
        !           351:                while (dp->b_actf != NULL) {
        !           352:                        /* Take one off the end of the list */
        !           353:                        bp = dp->b_actl;
        !           354:                        if (bp->b_actl == dp) {
        !           355:                                dp->b_actf = dp->b_actl = (struct buf *)0;
        !           356:                        } else {
        !           357:                                dp->b_actl = bp->b_actl;
        !           358:                                bp->b_actl->b_actf = dp;
        !           359:                        }
        !           360:                        (void) nfs_doio(bp);
        !           361:                }
        !           362:                if (error) {
        !           363:                        nfs_asyncdaemon[myiod] = 0;
        !           364:                        nfs_numasync--;
        !           365:                        return (error);
        !           366:                }
        !           367:        }
        !           368: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.