Annotation of lucent/sys/src/9/port/devmnt.c, revision 1.1.1.1

1.1       root        1: #include       "u.h"
                      2: #include       "../port/lib.h"
                      3: #include       "mem.h"
                      4: #include       "dat.h"
                      5: #include       "fns.h"
                      6: #include       "../port/error.h"
                      7: #include       "devtab.h"
                      8: 
                      9: typedef struct Mntrpc Mntrpc;
                     10: struct Mntrpc
                     11: {
                     12:        Mntrpc  *list;          /* Free/pending list */
                     13:        Fcall   request;        /* Outgoing file system protocol message */
                     14:        Fcall   reply;          /* Incoming reply */
                     15:        Mnt     *m;             /* Mount device during rpc */
                     16:        Rendez  r;              /* Place to hang out */
                     17:        char    *rpc;           /* I/O Data buffer */
                     18:        char    done;           /* Rpc completed */
                     19:        char    flushed;        /* Flush was sent */
                     20:        ushort  flushtag;       /* Tag flush sent on */
                     21:        char    flush[MAXMSG];  /* Somewhere to build flush */
                     22: };
                     23: 
                     24: struct Mnt
                     25: {
                     26:        int reads;              /* counters for debugging */
                     27:        int writes;
                     28:        int readerrs;
                     29:        int badlen;
                     30:        int goodconv;
                     31:        int noone;
                     32: 
                     33:        Ref;                    /* Count of attached channels */
                     34:        Chan    *c;             /* Channel to file service */
                     35:        Proc    *rip;           /* Reader in progress */
                     36:        Mntrpc  *queue;         /* Queue of pending requests on this channel */
                     37:        ulong   id;             /* Multiplexor id for channel check */
                     38:        Mnt     *list;          /* Free list */
                     39:        char    mux;            /* Set if the device does the multiplexing */
                     40:        int     blocksize;      /* read/write block size */
                     41:        ushort  flushtag;       /* Tag to send flush on */
                     42:        ushort  flushbase;      /* Base tag of flush window for this buffer */
                     43: };
                     44: 
                     45: struct Mntalloc
                     46: {
                     47:        Lock;
                     48:        Mnt     *list;          /* Mount devices in used */
                     49:        Mnt     *mntfree;       /* Free list */
                     50:        Mntrpc  *rpcfree;
                     51:        int     id;
                     52:        int     rpctag;
                     53: }mntalloc;
                     54: 
                     55: #define MAXRPC         (MAXFDATA+MAXMSG)
                     56: #define limit(n, max)  (n > max ? max : n)
                     57: 
                     58: Chan*  mattach(Mnt*, char*);
                     59: Mnt*   mntchk(Chan*);
                     60: void   mntdirfix(uchar*, Chan*);
                     61: void   mntdoclunk(Mnt *, Mntrpc *);
                     62: int    mntflush(Mnt*, Mntrpc*);
                     63: void   mntfree(Mntrpc*);
                     64: void   mntgate(Mnt*);
                     65: void   mntpntfree(Mnt*);
                     66: void   mntqrm(Mnt*, Mntrpc*);
                     67: Mntrpc*        mntralloc(void);
                     68: long   mntrdwr(int , Chan*, void*,long , ulong);
                     69: void   mntrpcread(Mnt*, Mntrpc*);
                     70: void   mountio(Mnt*, Mntrpc*);
                     71: void   mountmux(Mnt*, Mntrpc*);
                     72: void   mountrpc(Mnt*, Mntrpc*);
                     73: int    rpcattn(Mntrpc*);
                     74: void   mclose(Mnt*);
                     75: 
                     76: static int     defblocksize = MAXFDATA;
                     77: 
                     78: enum
                     79: {
                     80:        Tagspace = 1,
                     81:        Tagfls = 0x8000,
                     82:        Tagend = 0xfffe,
                     83: 
                     84:        ALIGN = 256,            /* Vme block mode alignment */
                     85: };
                     86: 
                     87: void
                     88: mntblocksize(int s)
                     89: {
                     90:        if(s < 256 || s > MAXFDATA)
                     91:                return;
                     92:        defblocksize = s;
                     93: }
                     94: 
                     95: void
                     96: mntreset(void)
                     97: {
                     98:        mntalloc.id = 1;
                     99:        mntalloc.rpctag = Tagspace;
                    100: }
                    101: 
                    102: void
                    103: mntinit(void)
                    104: {
                    105: }
                    106: 
                    107: Chan*
                    108: mntattach(char *muxattach)
                    109: {
                    110:        Mnt *m;
                    111:        Chan *c, *mc;
                    112:        struct bogus{
                    113:                Chan    *chan;
                    114:                char    *spec;
                    115:        }bogus;
                    116: 
                    117:        bogus = *((struct bogus *)muxattach);
                    118:        c = bogus.chan;
                    119: 
                    120:        lock(&mntalloc);
                    121:        for(m = mntalloc.list; m; m = m->list) {
                    122:                if(m->c == c && m->id) {
                    123:                        lock(m);
                    124:                        if(m->id && m->ref > 0 && m->c == c) {
                    125:                                unlock(&mntalloc);
                    126:                                m->ref++;
                    127:                                unlock(m);
                    128:                                return mattach(m, bogus.spec);
                    129:                        }
                    130:                        unlock(m);      
                    131:                }
                    132:        }
                    133: 
                    134:        m = mntalloc.mntfree;
                    135:        if(m != 0)
                    136:                mntalloc.mntfree = m->list;     
                    137:        else {
                    138:                m = malloc(sizeof(Mnt));
                    139:                if(m == 0) {
                    140:                        unlock(&mntalloc);
                    141:                        exhausted("mount devices");
                    142:                }
                    143:                m->flushbase = Tagfls;
                    144:                m->flushtag = Tagfls;
                    145:        }
                    146:        m->list = mntalloc.list;
                    147:        mntalloc.list = m;
                    148:        m->id = mntalloc.id++;
                    149:        lock(m);
                    150:        unlock(&mntalloc);
                    151: 
                    152:        m->ref = 1;
                    153:        m->queue = 0;
                    154:        m->rip = 0;
                    155:        m->c = c;
                    156:        m->c->flag |= CMSG;
                    157:        m->blocksize = defblocksize;
                    158: 
                    159:        switch(devchar[m->c->type]) {
                    160:        default:
                    161:                m->mux = 0;
                    162:                break;
                    163:        case 'C':                       /* Cyclone */
                    164:                m->mux = 1;
                    165:                break;
                    166:        }
                    167:        incref(m->c);
                    168:        unlock(m);
                    169: 
                    170:        if(waserror()) {
                    171:                mclose(m);
                    172:                nexterror();
                    173:        }
                    174: 
                    175:        c = mattach(m, bogus.spec);
                    176: 
                    177:        /*
                    178:         *  If exportfs mounts on behalf of a local devmnt, the mount
                    179:         *  point is folded onto the original channel to preserve a single
                    180:         *  fid/tag space.  CHDIR is cleared by exportfs to indicate it
                    181:         *  is supplying the mount.
                    182:         */
                    183:        mc = m->c;
                    184:        if(mc->type == devno('M', 0) && (c->qid.path&CHDIR) == 0) {
                    185:                c->qid.path |= CHDIR;
                    186:                c->mntptr = mc->mntptr;
                    187:                c->mchan = mc->mntptr->c;
                    188:                c->mqid = c->qid;
                    189:                incref(c->mntptr);
                    190:                mclose(m);
                    191:        }
                    192: 
                    193:        poperror();
                    194:        return c;
                    195: }
                    196: 
                    197: Chan *
                    198: mattach(Mnt *m, char *spec)
                    199: {
                    200:        Chan *c;
                    201:        Mntrpc *r;
                    202:        ulong id;
                    203: 
                    204:        r = mntralloc();
                    205:        c = devattach('M', spec);
                    206:        lock(&mntalloc);
                    207:        c->dev = mntalloc.id++;
                    208:        unlock(&mntalloc);
                    209:        c->mntptr = m;
                    210: 
                    211:        if(waserror()){
                    212:                mntfree(r);
                    213:                /* Close must not be called since
                    214:                 * it will call mnt recursively
                    215:                 */
                    216:                chanfree(c);
                    217:                nexterror();
                    218:        }
                    219: 
                    220:        r->request.type = Tattach;
                    221:        r->request.fid = c->fid;
                    222:        memmove(r->request.uname, u->p->user, NAMELEN);
                    223:        strncpy(r->request.aname, spec, NAMELEN);
                    224:        id = authrequest(m->c->session, &r->request);
                    225:        mountrpc(m, r);
                    226:        authreply(m->c->session, id, &r->reply);
                    227: 
                    228:        c->qid = r->reply.qid;
                    229:        c->mchan = m->c;
                    230:        c->mqid = c->qid;
                    231:        poperror();
                    232:        mntfree(r);
                    233:        return c;
                    234: }
                    235: 
                    236: Chan*
                    237: mntclone(Chan *c, Chan *nc)
                    238: {
                    239:        Mnt *m;
                    240:        Mntrpc *r;
                    241:        int alloc = 0;
                    242: 
                    243:        m = mntchk(c);
                    244:        r = mntralloc();
                    245:        if(nc == 0) {
                    246:                nc = newchan();
                    247:                alloc = 1;
                    248:        }
                    249:        if(waserror()) {
                    250:                mntfree(r);
                    251:                if(alloc)
                    252:                        close(nc);
                    253:                nexterror();
                    254:        }
                    255: 
                    256:        r->request.type = Tclone;
                    257:        r->request.fid = c->fid;
                    258:        r->request.newfid = nc->fid;
                    259:        mountrpc(m, r);
                    260: 
                    261:        devclone(c, nc);
                    262:        nc->mqid = c->qid;
                    263:        incref(m);
                    264: 
                    265:        USED(alloc);
                    266:        poperror();
                    267:        mntfree(r);
                    268:        return nc;
                    269: }
                    270: 
                    271: int     
                    272: mntwalk(Chan *c, char *name)
                    273: {
                    274:        Mnt *m;
                    275:        Mntrpc *r;
                    276: 
                    277:        m = mntchk(c);
                    278:        r = mntralloc();
                    279:        if(waserror()) {
                    280:                mntfree(r);
                    281:                return 0;
                    282:        }
                    283:        r->request.type = Twalk;
                    284:        r->request.fid = c->fid;
                    285:        strncpy(r->request.name, name, NAMELEN);
                    286:        mountrpc(m, r);
                    287: 
                    288:        c->qid = r->reply.qid;
                    289: 
                    290:        poperror();
                    291:        mntfree(r);
                    292:        return 1;
                    293: }
                    294: 
                    295: void    
                    296: mntstat(Chan *c, char *dp)
                    297: {
                    298:        Mnt *m;
                    299:        Mntrpc *r;
                    300: 
                    301:        m = mntchk(c);
                    302:        r = mntralloc();
                    303:        if(waserror()) {
                    304:                mntfree(r);
                    305:                nexterror();
                    306:        }
                    307:        r->request.type = Tstat;
                    308:        r->request.fid = c->fid;
                    309:        mountrpc(m, r);
                    310: 
                    311:        memmove(dp, r->reply.stat, DIRLEN);
                    312:        mntdirfix((uchar*)dp, c);
                    313:        poperror();
                    314:        mntfree(r);
                    315: }
                    316: 
                    317: Chan*
                    318: mntopen(Chan *c, int omode)
                    319: {
                    320:        Mnt *m;
                    321:        Mntrpc *r;
                    322: 
                    323:        m = mntchk(c);
                    324:        r = mntralloc();
                    325:        if(waserror()) {
                    326:                mntfree(r);
                    327:                nexterror();
                    328:        }
                    329:        r->request.type = Topen;
                    330:        r->request.fid = c->fid;
                    331:        r->request.mode = omode;
                    332:        mountrpc(m, r);
                    333: 
                    334:        c->qid = r->reply.qid;
                    335:        c->offset = 0;
                    336:        c->mode = openmode(omode);
                    337:        c->flag |= COPEN;
                    338:        poperror();
                    339:        mntfree(r);
                    340:        return c;
                    341: }
                    342: 
                    343: void    
                    344: mntcreate(Chan *c, char *name, int omode, ulong perm)
                    345: {
                    346:        Mnt *m;
                    347:        Mntrpc *r;
                    348: 
                    349:        m = mntchk(c);
                    350:        r = mntralloc();
                    351:        if(waserror()) {
                    352:                mntfree(r);
                    353:                nexterror();
                    354:        }
                    355:        r->request.type = Tcreate;
                    356:        r->request.fid = c->fid;
                    357:        r->request.mode = omode;
                    358:        r->request.perm = perm;
                    359:        strncpy(r->request.name, name, NAMELEN);
                    360:        mountrpc(m, r);
                    361: 
                    362:        c->qid = r->reply.qid;
                    363:        c->flag |= COPEN;
                    364:        c->mode = openmode(omode);
                    365:        poperror();
                    366:        mntfree(r);
                    367: }
                    368: 
                    369: void    
                    370: mntclunk(Chan *c, int t)
                    371: {
                    372:        Mnt *m;
                    373:        Mntrpc *r;
                    374:                
                    375:        m = mntchk(c);
                    376:        r = mntralloc();
                    377:        if(waserror()){
                    378:                mntdoclunk(m, r);
                    379:                nexterror();
                    380:        }
                    381: 
                    382:        r->request.type = t;
                    383:        r->request.fid = c->fid;
                    384:        mountrpc(m, r);
                    385:        mntdoclunk(m, r);
                    386:        poperror();
                    387: }
                    388: 
                    389: void
                    390: mclose(Mnt *m)
                    391: {
                    392:        Mntrpc *q, *r;
                    393: 
                    394:        if(decref(m) != 0)
                    395:                return;
                    396: 
                    397:        for(q = m->queue; q; q = r) {
                    398:                r = q->list;
                    399:                q->flushed = 0;
                    400:                mntfree(q);
                    401:        }
                    402:        m->id = 0;
                    403:        close(m->c);
                    404:        mntpntfree(m);
                    405: }
                    406: 
                    407: void
                    408: mntdoclunk(Mnt *m, Mntrpc *r)
                    409: {
                    410:        mntfree(r);
                    411:        mclose(m);
                    412: }
                    413: 
                    414: void
                    415: mntpntfree(Mnt *m)
                    416: {
                    417:        Mnt *f, **l;
                    418: 
                    419:        lock(&mntalloc);
                    420:        l = &mntalloc.list;
                    421:        for(f = *l; f; f = f->list) {
                    422:                if(f == m) {
                    423:                        *l = m->list;
                    424:                        break;
                    425:                }
                    426:                l = &f->list;
                    427:        }
                    428: 
                    429:        m->list = mntalloc.mntfree;
                    430:        mntalloc.mntfree = m;
                    431:        unlock(&mntalloc);
                    432: }
                    433: 
                    434: void
                    435: mntclose(Chan *c)
                    436: {
                    437:        mntclunk(c, Tclunk);
                    438: }
                    439: 
                    440: void    
                    441: mntremove(Chan *c)
                    442: {
                    443:        mntclunk(c, Tremove);
                    444: }
                    445: 
                    446: void
                    447: mntwstat(Chan *c, char *dp)
                    448: {
                    449:        Mnt *m;
                    450:        Mntrpc *r;
                    451: 
                    452:        m = mntchk(c);
                    453:        r = mntralloc();
                    454:        if(waserror()) {
                    455:                mntfree(r);
                    456:                nexterror();
                    457:        }
                    458:        r->request.type = Twstat;
                    459:        r->request.fid = c->fid;
                    460:        memmove(r->request.stat, dp, DIRLEN);
                    461:        mountrpc(m, r);
                    462:        poperror();
                    463:        mntfree(r);
                    464: }
                    465: 
                    466: long    
                    467: mntread(Chan *c, void *buf, long n, ulong offset)
                    468: {
                    469:        uchar *p, *e;
                    470: 
                    471:        n = mntrdwr(Tread, c, buf, n, offset);
                    472:        if(c->qid.path & CHDIR) 
                    473:                for(p = (uchar*)buf, e = &p[n]; p < e; p += DIRLEN)
                    474:                        mntdirfix(p, c);
                    475: 
                    476:        return n;
                    477: }
                    478: 
                    479: long    
                    480: mntwrite(Chan *c, void *buf, long n, ulong offset)
                    481: {
                    482:        return mntrdwr(Twrite, c, buf, n, offset);      
                    483: }
                    484: 
                    485: long
                    486: mntrdwr(int type, Chan *c, void *buf, long n, ulong offset)
                    487: {
                    488:        Mnt *m;
                    489:        Mntrpc *r;
                    490:        char *uba;
                    491:        ulong cnt, nr;
                    492: 
                    493:        m = mntchk(c);
                    494:        uba = buf;
                    495:        cnt = 0;
                    496:        for(;;) {
                    497:                r = mntralloc();
                    498:                if(waserror()) {
                    499:                        mntfree(r);
                    500:                        nexterror();
                    501:                }
                    502:                r->request.type = type;
                    503:                r->request.fid = c->fid;
                    504:                r->request.offset = offset;
                    505:                r->request.data = uba;
                    506:                r->request.count = limit(n, m->blocksize);
                    507:                mountrpc(m, r);
                    508:                nr = r->reply.count;
                    509:                if(nr > r->request.count)
                    510:                        nr = r->request.count;
                    511:                if(type == Tread)
                    512:                        memmove(uba, r->reply.data, nr);
                    513:                poperror();
                    514:                mntfree(r);
                    515:                offset += nr;
                    516:                uba += nr;
                    517:                cnt += nr;
                    518:                n -= nr;
                    519:                if(nr != r->request.count || n == 0 || u->nnote)
                    520:                        break;
                    521:        }
                    522:        return cnt;
                    523: }
                    524: 
                    525: void
                    526: mountrpc(Mnt *m, Mntrpc *r)
                    527: {
                    528:        r->reply.tag = 0;               /* poison the old values */
                    529:        r->reply.type = 4;
                    530: 
                    531:        mountio(m, r);
                    532:        if(r->reply.type == Rerror)
                    533:                error(r->reply.ename);
                    534: 
                    535:        if(r->reply.type == Rflush)
                    536:                error(Eintr);
                    537: 
                    538:        if(r->reply.type != r->request.type+1) {
                    539:                print("mnt: mismatched reply 0x%lux T%d R%d tags req %d fls %d rep %d\n",
                    540:                                r, r->request.type, r->reply.type, r->request.tag, 
                    541:                                r->flushtag, r->reply.tag);
                    542: 
                    543:                error(Emountrpc);
                    544:        }
                    545: }
                    546: 
                    547: void
                    548: mountio(Mnt *m, Mntrpc *r)
                    549: {
                    550:        int n;
                    551: 
                    552:        lock(m);
                    553:        r->flushed = 0;
                    554:        r->m = m;
                    555:        r->list = m->queue;
                    556:        m->queue = r;
                    557:        unlock(m);
                    558: 
                    559:        /* Transmit a file system rpc */
                    560:        n = convS2M(&r->request, r->rpc);
                    561:        if(waserror()) {
                    562:                if(mntflush(m, r) == 0)
                    563:                        nexterror();
                    564:        }
                    565:        else {
                    566:                if((*devtab[m->c->type].write)(m->c, r->rpc, n, 0) != n)
                    567:                        error(Emountrpc);
                    568:                m->writes++;
                    569:                poperror();
                    570:        }
                    571:        if(m->mux) {
                    572:                mntqrm(m, r);
                    573:                mntrpcread(m, r);
                    574:                return;
                    575:        }
                    576: 
                    577:        /* Gate readers onto the mount point one at a time */
                    578:        for(;;) {
                    579:                lock(m);
                    580:                if(m->rip == 0)
                    581:                        break;
                    582:                unlock(m);
                    583:                if(waserror()) {
                    584:                        if(mntflush(m, r) == 0)
                    585:                                nexterror();
                    586:                        continue;
                    587:                }
                    588:                sleep(&r->r, rpcattn, r);
                    589:                poperror();
                    590:                if(r->done)
                    591:                        return;
                    592:        }
                    593:        m->rip = u->p;
                    594:        unlock(m);
                    595:        while(r->done == 0) {
                    596:                mntrpcread(m, r);
                    597:                mountmux(m, r);
                    598:        }
                    599:        mntgate(m);
                    600: }
                    601: 
                    602: void
                    603: mntrpcread(Mnt *m, Mntrpc *r)
                    604: {
                    605:        int n;
                    606: 
                    607:        for(;;) {
                    608:                if(waserror()) {
                    609:                        m->readerrs++;
                    610:                        if(mntflush(m, r) == 0) {
                    611:                                if(m->mux == 0)
                    612:                                        mntgate(m);
                    613:                                nexterror();
                    614:                        }
                    615:                        continue;
                    616:                }
                    617:                r->reply.type = 0;
                    618:                r->reply.tag = 0;
                    619:                n = (*devtab[m->c->type].read)(m->c, r->rpc, MAXRPC, 0);
                    620:                poperror();
                    621:                m->reads++;
                    622:                if(n == 0){
                    623:                        m->badlen++;
                    624:                        continue;
                    625:                }
                    626: 
                    627:                if(convM2S(r->rpc, &r->reply, n) != 0){
                    628:                        m->goodconv++;
                    629:                        return;
                    630:                }
                    631:        }
                    632: }
                    633: 
                    634: void
                    635: mntgate(Mnt *m)
                    636: {
                    637:        Mntrpc *q;
                    638: 
                    639:        lock(m);
                    640:        m->rip = 0;
                    641:        for(q = m->queue; q; q = q->list)
                    642:                if(q->done == 0) {
                    643:                        lock(&q->r);
                    644:                        if(q->r.p) {
                    645:                                unlock(&q->r);
                    646:                                unlock(m);
                    647:                                wakeup(&q->r);
                    648:                                return;
                    649:                        }
                    650:                        unlock(&q->r);
                    651:                }
                    652:        unlock(m);
                    653: }
                    654: 
                    655: void
                    656: mountmux(Mnt *m, Mntrpc *r)
                    657: {
                    658:        char *dp;
                    659:        Mntrpc **l, *q;
                    660: 
                    661:        lock(m);
                    662:        l = &m->queue;
                    663:        for(q = *l; q; q = q->list) {
                    664:                if(q->request.tag == r->reply.tag
                    665:                || q->flushed && q->flushtag == r->reply.tag) {
                    666:                        *l = q->list;
                    667:                        unlock(m);
                    668:                        if(q != r) {            /* Completed someone else */
                    669:                                dp = q->rpc;
                    670:                                q->rpc = r->rpc;
                    671:                                r->rpc = dp;
                    672:                                memmove(&q->reply, &r->reply, sizeof(Fcall));
                    673:                                q->done = 1;
                    674:                                wakeup(&q->r);
                    675:                        }else
                    676:                                q->done = 1;
                    677:                        return;
                    678:                }
                    679:                l = &q->list;
                    680:        }
                    681:        m->noone++;
                    682:        unlock(m);
                    683: }
                    684: 
                    685: int
                    686: mntflush(Mnt *m, Mntrpc *r)
                    687: {
                    688:        int n;
                    689:        Fcall flush;
                    690: 
                    691:        lock(m);
                    692:        r->flushtag = m->flushtag++;
                    693:        if(m->flushtag == Tagend)
                    694:                m->flushtag = m->flushbase;
                    695:        r->flushed = 1;
                    696:        unlock(m);
                    697: 
                    698:        flush.type = Tflush;
                    699:        flush.tag = r->flushtag;
                    700:        flush.oldtag = r->request.tag;
                    701:        n = convS2M(&flush, r->flush);
                    702: 
                    703:        if(waserror()) {
                    704:                if(strcmp(u->error, Eintr) == 0)
                    705:                        return 1;
                    706:                mntqrm(m, r);
                    707:                return 0;
                    708:        }
                    709:        (*devtab[m->c->type].write)(m->c, r->flush, n, 0);
                    710:        poperror();
                    711:        return 1;
                    712: }
                    713: 
                    714: Mntrpc *
                    715: mntralloc(void)
                    716: {
                    717:        Mntrpc *new;
                    718: 
                    719:        lock(&mntalloc);
                    720:        new = mntalloc.rpcfree;
                    721:        if(new != 0)
                    722:                mntalloc.rpcfree = new->list;
                    723:        else {
                    724:                new = xalloc(sizeof(Mntrpc)+MAXRPC);
                    725:                if(new == 0) {
                    726:                        unlock(&mntalloc);
                    727:                        exhausted("mount rpc buffer");
                    728:                }
                    729:                new->rpc = (char*)new+sizeof(Mntrpc);
                    730:                new->request.tag = mntalloc.rpctag++;
                    731:        }
                    732:        unlock(&mntalloc);
                    733:        new->done = 0;
                    734:        new->flushed = 0;
                    735:        return new;
                    736: }
                    737: 
                    738: void
                    739: mntfree(Mntrpc *r)
                    740: {
                    741:        lock(&mntalloc);
                    742:        r->list = mntalloc.rpcfree;
                    743:        mntalloc.rpcfree = r;
                    744:        unlock(&mntalloc);
                    745: }
                    746: 
                    747: void
                    748: mntqrm(Mnt *m, Mntrpc *r)
                    749: {
                    750:        Mntrpc **l, *f;
                    751: 
                    752:        lock(m);
                    753:        r->done = 1;
                    754:        r->flushed = 0;
                    755: 
                    756:        l = &m->queue;
                    757:        for(f = *l; f; f = f->list) {
                    758:                if(f == r) {
                    759:                        *l = r->list;
                    760:                        break;
                    761:                }
                    762:                l = &f->list;
                    763:        }
                    764:        unlock(m);
                    765: }
                    766: 
                    767: Mnt *
                    768: mntchk(Chan *c)
                    769: {
                    770:        Mnt *m;
                    771: 
                    772:        m = c->mntptr;
                    773:        /* Was it closed and reused ? */
                    774:        if(m->id == 0 || m->id >= c->dev)       /* Sanity check */
                    775:                error(Eshutdown);
                    776:        return m;
                    777: }
                    778: 
                    779: void
                    780: mntdirfix(uchar *dirbuf, Chan *c)
                    781: {
                    782:        dirbuf[DIRLEN-4] = devchar[c->type]>>0;
                    783:        dirbuf[DIRLEN-3] = devchar[c->type]>>8;
                    784:        dirbuf[DIRLEN-2] = c->dev;
                    785:        dirbuf[DIRLEN-1] = c->dev>>8;
                    786: }
                    787: 
                    788: int
                    789: rpcattn(Mntrpc *r)
                    790: {
                    791:        return r->done || r->m->rip == 0;
                    792: }

unix.superglobalmegacorp.com

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