Annotation of Net2/miscfs/fifofs/fifo_vnops.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1990 The Regents of the University of California.
        !             3:  * All rights reserved.
        !             4:  *
        !             5:  * Redistribution and use in source and binary forms, with or without
        !             6:  * modification, are permitted provided that the following conditions
        !             7:  * are met:
        !             8:  * 1. Redistributions of source code must retain the above copyright
        !             9:  *    notice, this list of conditions and the following disclaimer.
        !            10:  * 2. Redistributions in binary form must reproduce the above copyright
        !            11:  *    notice, this list of conditions and the following disclaimer in the
        !            12:  *    documentation and/or other materials provided with the distribution.
        !            13:  * 3. All advertising materials mentioning features or use of this software
        !            14:  *    must display the following acknowledgement:
        !            15:  *     This product includes software developed by the University of
        !            16:  *     California, Berkeley and its contributors.
        !            17:  * 4. Neither the name of the University nor the names of its contributors
        !            18:  *    may be used to endorse or promote products derived from this software
        !            19:  *    without specific prior written permission.
        !            20:  *
        !            21:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
        !            22:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
        !            23:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
        !            24:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
        !            25:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
        !            26:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
        !            27:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
        !            28:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
        !            29:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
        !            30:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
        !            31:  * SUCH DAMAGE.
        !            32:  *
        !            33:  *     @(#)fifo_vnops.c        7.7 (Berkeley) 4/15/91
        !            34:  */
        !            35: 
        !            36: #include "param.h"
        !            37: #include "time.h"
        !            38: #include "namei.h"
        !            39: #include "vnode.h"
        !            40: #include "socket.h"
        !            41: #include "socketvar.h"
        !            42: #include "stat.h"
        !            43: #include "systm.h"
        !            44: #include "ioctl.h"
        !            45: #include "file.h"
        !            46: #include "fifo.h"
        !            47: #include "errno.h"
        !            48: #include "malloc.h"
        !            49: 
        !            50: /*
        !            51:  * This structure is associated with the FIFO vnode and stores
        !            52:  * the state associated with the FIFO.
        !            53:  */
        !            54: struct fifoinfo {
        !            55:        struct socket   *fi_readsock;
        !            56:        struct socket   *fi_writesock;
        !            57:        long            fi_readers;
        !            58:        long            fi_writers;
        !            59: };
        !            60: 
        !            61: struct vnodeops fifo_vnodeops = {
        !            62:        fifo_lookup,            /* lookup */
        !            63:        fifo_create,            /* create */
        !            64:        fifo_mknod,             /* mknod */
        !            65:        fifo_open,              /* open */
        !            66:        fifo_close,             /* close */
        !            67:        fifo_access,            /* access */
        !            68:        fifo_getattr,           /* getattr */
        !            69:        fifo_setattr,           /* setattr */
        !            70:        fifo_read,              /* read */
        !            71:        fifo_write,             /* write */
        !            72:        fifo_ioctl,             /* ioctl */
        !            73:        fifo_select,            /* select */
        !            74:        fifo_mmap,              /* mmap */
        !            75:        fifo_fsync,             /* fsync */
        !            76:        fifo_seek,              /* seek */
        !            77:        fifo_remove,            /* remove */
        !            78:        fifo_link,              /* link */
        !            79:        fifo_rename,            /* rename */
        !            80:        fifo_mkdir,             /* mkdir */
        !            81:        fifo_rmdir,             /* rmdir */
        !            82:        fifo_symlink,           /* symlink */
        !            83:        fifo_readdir,           /* readdir */
        !            84:        fifo_readlink,          /* readlink */
        !            85:        fifo_abortop,           /* abortop */
        !            86:        fifo_inactive,          /* inactive */
        !            87:        fifo_reclaim,           /* reclaim */
        !            88:        fifo_lock,              /* lock */
        !            89:        fifo_unlock,            /* unlock */
        !            90:        fifo_bmap,              /* bmap */
        !            91:        fifo_strategy,          /* strategy */
        !            92:        fifo_print,             /* print */
        !            93:        fifo_islocked,          /* islocked */
        !            94:        fifo_advlock,           /* advlock */
        !            95: };
        !            96: 
        !            97: /*
        !            98:  * Trivial lookup routine that always fails.
        !            99:  */
        !           100: /* ARGSUSED */
        !           101: fifo_lookup(vp, ndp, p)
        !           102:        struct vnode *vp;
        !           103:        struct nameidata *ndp;
        !           104:        struct proc *p;
        !           105: {
        !           106: 
        !           107:        ndp->ni_dvp = vp;
        !           108:        ndp->ni_vp = NULL;
        !           109:        return (ENOTDIR);
        !           110: }
        !           111: 
        !           112: /*
        !           113:  * Open called to set up a new instance of a fifo or
        !           114:  * to find an active instance of a fifo.
        !           115:  */
        !           116: /* ARGSUSED */
        !           117: fifo_open(vp, mode, cred, p)
        !           118:        register struct vnode *vp;
        !           119:        int mode;
        !           120:        struct ucred *cred;
        !           121:        struct proc *p;
        !           122: {
        !           123:        register struct fifoinfo *fip;
        !           124:        struct socket *rso, *wso;
        !           125:        int error;
        !           126:        static char openstr[] = "fifo";
        !           127: 
        !           128:        if ((mode & (FREAD|FWRITE)) == (FREAD|FWRITE))
        !           129:                return (EINVAL);
        !           130:        if ((fip = vp->v_fifoinfo) == NULL) {
        !           131:                MALLOC(fip, struct fifoinfo *, sizeof(*fip), M_VNODE, M_WAITOK);
        !           132:                vp->v_fifoinfo = fip;
        !           133:                fip->fi_readers=0;
        !           134:                fip->fi_writers=0;
        !           135:                if (error = socreate(AF_UNIX, &rso, SOCK_STREAM, 0)) {
        !           136:                        free(fip, M_VNODE);
        !           137:                        vp->v_fifoinfo = NULL;
        !           138:                        return (error);
        !           139:                }
        !           140:                fip->fi_readsock = rso;
        !           141:                if (error = socreate(AF_UNIX, &wso, SOCK_STREAM, 0)) {
        !           142:                        (void)soclose(rso);
        !           143:                        free(fip, M_VNODE);
        !           144:                        vp->v_fifoinfo = NULL;
        !           145:                        return (error);
        !           146:                }
        !           147:                fip->fi_writesock = wso;
        !           148:                if (error = unp_connect2(wso, rso)) {
        !           149:                        (void)soclose(wso);
        !           150:                        (void)soclose(rso);
        !           151:                        free(fip, M_VNODE);
        !           152:                        vp->v_fifoinfo = NULL;
        !           153:                        return (error);
        !           154:                }
        !           155:                wso->so_state |= SS_CANTRCVMORE;
        !           156:                rso->so_state |= SS_CANTSENDMORE;
        !           157:        }
        !           158:        error = 0;
        !           159:        if (mode & FREAD) {
        !           160:                fip->fi_readers++;
        !           161:                if (fip->fi_readers == 1) {
        !           162:                        fip->fi_writesock->so_state &= ~SS_CANTSENDMORE;
        !           163:                        if (fip->fi_writers > 0)
        !           164:                                wakeup((caddr_t)&fip->fi_writers);
        !           165:                }
        !           166:                if (mode & O_NONBLOCK)
        !           167:                        return (0);
        !           168:                while (fip->fi_writers == 0) {
        !           169:                        VOP_UNLOCK(vp);
        !           170:                        error = tsleep((caddr_t)&fip->fi_readers, PSOCK,
        !           171:                            openstr, 0);
        !           172:                        VOP_LOCK(vp);
        !           173:                        if(error)
        !           174:                                break;
        !           175:                }
        !           176:        } else {
        !           177:                fip->fi_writers++;
        !           178:                if (fip->fi_readers == 0 && (mode & O_NONBLOCK)) {
        !           179:                        error = ENXIO;
        !           180:                } else {
        !           181:                        if (fip->fi_writers == 1) {
        !           182:                                fip->fi_readsock->so_state &= ~SS_CANTRCVMORE;
        !           183:                                if (fip->fi_readers > 0)
        !           184:                                        wakeup((caddr_t)&fip->fi_readers);
        !           185:                        }
        !           186:                        while (fip->fi_readers == 0) {
        !           187:                                VOP_UNLOCK(vp);
        !           188:                                error = tsleep((caddr_t)&fip->fi_writers,
        !           189:                                    PSOCK, openstr, 0);
        !           190:                                VOP_LOCK(vp);
        !           191:                                if(error)
        !           192:                                        break;
        !           193:                        }
        !           194:                }
        !           195:        }
        !           196:        if (error)
        !           197:                fifo_close(vp, mode, cred, p);
        !           198:        return (error);
        !           199: }
        !           200: 
        !           201: /*
        !           202:  * Vnode op for read
        !           203:  */
        !           204: /* ARGSUSED */
        !           205: fifo_read(vp, uio, ioflag, cred)
        !           206:        struct vnode *vp;
        !           207:        register struct uio *uio;
        !           208:        int ioflag;
        !           209:        struct ucred *cred;
        !           210: {
        !           211:        register struct socket *rso = vp->v_fifoinfo->fi_readsock;
        !           212:        int error, startresid;
        !           213: 
        !           214: #ifdef DIAGNOSTIC
        !           215:        if (uio->uio_rw != UIO_READ)
        !           216:                panic("fifo_read mode");
        !           217: #endif
        !           218:        if (uio->uio_resid == 0)
        !           219:                return (0);
        !           220:        if (ioflag & IO_NDELAY)
        !           221:                rso->so_state |= SS_NBIO;
        !           222:        startresid = uio->uio_resid;
        !           223:        VOP_UNLOCK(vp);
        !           224:        error = soreceive(rso, (struct mbuf **)0, uio, (int *)0,
        !           225:                (struct mbuf **)0, (struct mbuf **)0);
        !           226:        VOP_LOCK(vp);
        !           227:        /*
        !           228:         * Clear EOF indication after first such return.
        !           229:         */
        !           230:        if (uio->uio_resid == startresid)
        !           231:                rso->so_state &= ~SS_CANTRCVMORE;
        !           232:        if (ioflag & IO_NDELAY)
        !           233:                rso->so_state &= ~SS_NBIO;
        !           234:        return (error);
        !           235: }
        !           236: 
        !           237: /*
        !           238:  * Vnode op for write
        !           239:  */
        !           240: /* ARGSUSED */
        !           241: fifo_write(vp, uio, ioflag, cred)
        !           242:        struct vnode *vp;
        !           243:        register struct uio *uio;
        !           244:        int ioflag;
        !           245:        struct ucred *cred;
        !           246: {
        !           247:        struct socket *wso = vp->v_fifoinfo->fi_writesock;
        !           248:        int error;
        !           249: 
        !           250: #ifdef DIAGNOSTIC
        !           251:        if (uio->uio_rw != UIO_WRITE)
        !           252:                panic("fifo_write mode");
        !           253: #endif
        !           254:        if (ioflag & IO_NDELAY)
        !           255:                wso->so_state |= SS_NBIO;
        !           256:        VOP_UNLOCK(vp);
        !           257:        error = sosend(wso, (struct mbuf *)0, uio, 0, (struct mbuf *)0);
        !           258:        VOP_LOCK(vp);
        !           259:        if (ioflag & IO_NDELAY)
        !           260:                wso->so_state &= ~SS_NBIO;
        !           261:        return (error);
        !           262: }
        !           263: 
        !           264: /*
        !           265:  * Device ioctl operation.
        !           266:  */
        !           267: /* ARGSUSED */
        !           268: fifo_ioctl(vp, com, data, fflag, cred, p)
        !           269:        struct vnode *vp;
        !           270:        int com;
        !           271:        caddr_t data;
        !           272:        int fflag;
        !           273:        struct ucred *cred;
        !           274:        struct proc *p;
        !           275: {
        !           276:        struct file filetmp;
        !           277:        int error;
        !           278: 
        !           279:        if (com == FIONBIO)
        !           280:                return (0);
        !           281:        if (fflag & FREAD)
        !           282:                filetmp.f_data = (caddr_t)vp->v_fifoinfo->fi_readsock;
        !           283:        else
        !           284:                filetmp.f_data = (caddr_t)vp->v_fifoinfo->fi_writesock;
        !           285:        return (soo_ioctl(&filetmp, com, data, p));
        !           286: }
        !           287: 
        !           288: /* ARGSUSED */
        !           289: fifo_select(vp, which, fflag, cred, p)
        !           290:        struct vnode *vp;
        !           291:        int which, fflag;
        !           292:        struct ucred *cred;
        !           293:        struct proc *p;
        !           294: {
        !           295:        struct file filetmp;
        !           296:        int error;
        !           297: 
        !           298:        if (fflag & FREAD)
        !           299:                filetmp.f_data = (caddr_t)vp->v_fifoinfo->fi_readsock;
        !           300:        else
        !           301:                filetmp.f_data = (caddr_t)vp->v_fifoinfo->fi_writesock;
        !           302:        return (soo_select(&filetmp, which, p));
        !           303: }
        !           304: 
        !           305: /*
        !           306:  * This is a noop, simply returning what one has been given.
        !           307:  */
        !           308: fifo_bmap(vp, bn, vpp, bnp)
        !           309:        struct vnode *vp;
        !           310:        daddr_t bn;
        !           311:        struct vnode **vpp;
        !           312:        daddr_t *bnp;
        !           313: {
        !           314: 
        !           315:        if (vpp != NULL)
        !           316:                *vpp = vp;
        !           317:        if (bnp != NULL)
        !           318:                *bnp = bn;
        !           319:        return (0);
        !           320: }
        !           321: 
        !           322: /*
        !           323:  * At the moment we do not do any locking.
        !           324:  */
        !           325: /* ARGSUSED */
        !           326: fifo_lock(vp)
        !           327:        struct vnode *vp;
        !           328: {
        !           329: 
        !           330:        return (0);
        !           331: }
        !           332: 
        !           333: /* ARGSUSED */
        !           334: fifo_unlock(vp)
        !           335:        struct vnode *vp;
        !           336: {
        !           337: 
        !           338:        return (0);
        !           339: }
        !           340: 
        !           341: /*
        !           342:  * Device close routine
        !           343:  */
        !           344: /* ARGSUSED */
        !           345: fifo_close(vp, fflag, cred, p)
        !           346:        register struct vnode *vp;
        !           347:        int fflag;
        !           348:        struct ucred *cred;
        !           349:        struct proc *p;
        !           350: {
        !           351:        register struct fifoinfo *fip = vp->v_fifoinfo;
        !           352:        int error1, error2;
        !           353: 
        !           354:        if (fflag & FWRITE) {
        !           355:                fip->fi_writers--;
        !           356:                if (fip->fi_writers == 0)
        !           357:                        socantrcvmore(fip->fi_readsock);
        !           358:        } else {
        !           359:                fip->fi_readers--;
        !           360:                if (fip->fi_readers == 0)
        !           361:                        socantsendmore(fip->fi_writesock);
        !           362:        }
        !           363:        if (vp->v_usecount > 1)
        !           364:                return (0);
        !           365:        error1 = soclose(fip->fi_readsock);
        !           366:        error2 = soclose(fip->fi_writesock);
        !           367:        FREE(fip, M_VNODE);
        !           368:        vp->v_fifoinfo = NULL;
        !           369:        if (error1)
        !           370:                return (error1);
        !           371:        return (error2);
        !           372: }
        !           373: 
        !           374: /*
        !           375:  * Print out the contents of a fifo vnode.
        !           376:  */
        !           377: fifo_print(vp)
        !           378:        struct vnode *vp;
        !           379: {
        !           380: 
        !           381:        printf("tag VT_NON");
        !           382:        fifo_printinfo(vp);
        !           383:        printf("\n");
        !           384: }
        !           385: 
        !           386: /*
        !           387:  * Print out internal contents of a fifo vnode.
        !           388:  */
        !           389: fifo_printinfo(vp)
        !           390:        struct vnode *vp;
        !           391: {
        !           392:        register struct fifoinfo *fip = vp->v_fifoinfo;
        !           393: 
        !           394:        printf(", fifo with %d readers and %d writers",
        !           395:                fip->fi_readers, fip->fi_writers);
        !           396: }
        !           397: 
        !           398: /*
        !           399:  * Fifo failed operation
        !           400:  */
        !           401: fifo_ebadf()
        !           402: {
        !           403: 
        !           404:        return (EBADF);
        !           405: }
        !           406: 
        !           407: /*
        !           408:  * Fifo advisory byte-level locks.
        !           409:  */
        !           410: /* ARGSUSED */
        !           411: fifo_advlock(vp, id, op, fl, flags)
        !           412:        struct vnode *vp;
        !           413:        caddr_t id;
        !           414:        int op;
        !           415:        struct flock *fl;
        !           416:        int flags;
        !           417: {
        !           418: 
        !           419:        return (EOPNOTSUPP);
        !           420: }
        !           421: 
        !           422: /*
        !           423:  * Fifo bad operation
        !           424:  */
        !           425: fifo_badop()
        !           426: {
        !           427: 
        !           428:        panic("fifo_badop called");
        !           429:        /* NOTREACHED */
        !           430: }

unix.superglobalmegacorp.com

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