Annotation of XNU/bsd/miscfs/umapfs/umap_vfsops.c, revision 1.1.1.1

1.1       root        1: /*
                      2:  * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
                      3:  *
                      4:  * @APPLE_LICENSE_HEADER_START@
                      5:  * 
                      6:  * The contents of this file constitute Original Code as defined in and
                      7:  * are subject to the Apple Public Source License Version 1.1 (the
                      8:  * "License").  You may not use this file except in compliance with the
                      9:  * License.  Please obtain a copy of the License at
                     10:  * http://www.apple.com/publicsource and read it before using this file.
                     11:  * 
                     12:  * This Original Code and all software distributed under the License are
                     13:  * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
                     14:  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
                     15:  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
                     16:  * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the
                     17:  * License for the specific language governing rights and limitations
                     18:  * under the License.
                     19:  * 
                     20:  * @APPLE_LICENSE_HEADER_END@
                     21:  */
                     22: /*     $NetBSD: umap_vfsops.c,v 1.7 1995/03/09 12:05:59 mycroft Exp $  */
                     23: 
                     24: /*
                     25:  * Copyright (c) 1992, 1993
                     26:  *     The Regents of the University of California.  All rights reserved.
                     27:  *
                     28:  * This code is derived from software donated to Berkeley by
                     29:  * the UCLA Ficus project.
                     30:  *
                     31:  * Redistribution and use in source and binary forms, with or without
                     32:  * modification, are permitted provided that the following conditions
                     33:  * are met:
                     34:  * 1. Redistributions of source code must retain the above copyright
                     35:  *    notice, this list of conditions and the following disclaimer.
                     36:  * 2. Redistributions in binary form must reproduce the above copyright
                     37:  *    notice, this list of conditions and the following disclaimer in the
                     38:  *    documentation and/or other materials provided with the distribution.
                     39:  * 3. All advertising materials mentioning features or use of this software
                     40:  *    must display the following acknowledgement:
                     41:  *     This product includes software developed by the University of
                     42:  *     California, Berkeley and its contributors.
                     43:  * 4. Neither the name of the University nor the names of its contributors
                     44:  *    may be used to endorse or promote products derived from this software
                     45:  *    without specific prior written permission.
                     46:  *
                     47:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
                     48:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
                     49:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
                     50:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
                     51:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
                     52:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
                     53:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
                     54:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
                     55:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
                     56:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
                     57:  * SUCH DAMAGE.
                     58:  *
                     59:  *     from: @(#)null_vfsops.c       1.5 (Berkeley) 7/10/92
                     60:  *     @(#)umap_vfsops.c       8.3 (Berkeley) 1/21/94
                     61:  */
                     62: 
                     63: /*
                     64:  * Umap Layer
                     65:  * (See mount_umap(8) for a description of this layer.)
                     66:  */
                     67: 
                     68: #include <sys/param.h>
                     69: #include <sys/systm.h>
                     70: #include <sys/time.h>
                     71: #include <sys/types.h>
                     72: #include <sys/vnode.h>
                     73: #include <sys/mount.h>
                     74: #include <sys/namei.h>
                     75: #include <sys/malloc.h>
                     76: #include <miscfs/umapfs/umap.h>
                     77: 
                     78: /*
                     79:  * Mount umap layer
                     80:  */
                     81: int
                     82: umapfs_mount(mp, path, data, ndp, p)
                     83:        struct mount *mp;
                     84:        char *path;
                     85:        caddr_t data;
                     86:        struct nameidata *ndp;
                     87:        struct proc *p;
                     88: {
                     89:        struct umap_args args;
                     90:        struct vnode *lowerrootvp, *vp;
                     91:        struct vnode *umapm_rootvp;
                     92:        struct umap_mount *amp;
                     93:        size_t size;
                     94:        int error;
                     95: 
                     96: #ifdef UMAPFS_DIAGNOSTIC
                     97:        printf("umapfs_mount(mp = %x)\n", mp);
                     98: #endif
                     99: 
                    100:        /*
                    101:         * Update is a no-op
                    102:         */
                    103:        if (mp->mnt_flag & MNT_UPDATE) {
                    104:                return (EOPNOTSUPP);
                    105:                /* return (VFS_MOUNT(MOUNTTOUMAPMOUNT(mp)->umapm_vfs, path, data, ndp, p));*/
                    106:        }
                    107: 
                    108:        /*
                    109:         * Get argument
                    110:         */
                    111:        if (error = copyin(data, (caddr_t)&args, sizeof(struct umap_args)))
                    112:                return (error);
                    113: 
                    114:        /*
                    115:         * Find lower node
                    116:         */
                    117:        NDINIT(ndp, LOOKUP, FOLLOW|WANTPARENT|LOCKLEAF,
                    118:                UIO_USERSPACE, args.target, p);
                    119:        if (error = namei(ndp))
                    120:                return (error);
                    121: 
                    122:        /*
                    123:         * Sanity check on lower vnode
                    124:         */
                    125:        lowerrootvp = ndp->ni_vp;
                    126: #ifdef UMAPFS_DIAGNOSTIC
                    127:        printf("vp = %x, check for VDIR...\n", lowerrootvp);
                    128: #endif
                    129:        vrele(ndp->ni_dvp);
                    130:        ndp->ni_dvp = 0;
                    131: 
                    132:        if (lowerrootvp->v_type != VDIR) {
                    133:                vput(lowerrootvp);
                    134:                return (EINVAL);
                    135:        }
                    136: 
                    137: #ifdef UMAPFS_DIAGNOSTIC
                    138:        printf("mp = %x\n", mp);
                    139: #endif
                    140: 
                    141: //     amp = (struct umap_mount *) malloc(sizeof(struct umap_mount),
                    142: //                             M_UFSMNT, M_WAITOK);    /* XXX */
                    143:        MALLOC(amp, struct umap_mount *, sizeof(struct umap_mount),
                    144:                                M_UFSMNT, M_WAITOK);
                    145: 
                    146:        /*
                    147:         * Save reference to underlying FS
                    148:         */
                    149:        amp->umapm_vfs = lowerrootvp->v_mount;
                    150: 
                    151:        /* 
                    152:         * Now copy in the number of entries and maps for umap mapping.
                    153:         */
                    154:        amp->info_nentries = args.nentries;
                    155:        amp->info_gnentries = args.gnentries;
                    156:        error = copyin(args.mapdata, (caddr_t)amp->info_mapdata, 
                    157:            2*sizeof(u_long)*args.nentries);
                    158:        if (error)
                    159:                return (error);
                    160: 
                    161: #ifdef UMAP_DIAGNOSTIC
                    162:        printf("umap_mount:nentries %d\n",args.nentries);
                    163:        for (i = 0; i < args.nentries; i++)
                    164:                printf("   %d maps to %d\n", amp->info_mapdata[i][0],
                    165:                    amp->info_mapdata[i][1]);
                    166: #endif
                    167: 
                    168:        error = copyin(args.gmapdata, (caddr_t)amp->info_gmapdata, 
                    169:            2*sizeof(u_long)*args.nentries);
                    170:        if (error)
                    171:                return (error);
                    172: 
                    173: #ifdef UMAP_DIAGNOSTIC
                    174:        printf("umap_mount:gnentries %d\n",args.gnentries);
                    175:        for (i = 0; i < args.gnentries; i++)
                    176:                printf("        group %d maps to %d\n", 
                    177:                    amp->info_gmapdata[i][0],
                    178:                    amp->info_gmapdata[i][1]);
                    179: #endif
                    180: 
                    181: 
                    182:        /*
                    183:         * Save reference.  Each mount also holds
                    184:         * a reference on the root vnode.
                    185:         */
                    186:        error = umap_node_create(mp, lowerrootvp, &vp);
                    187:        /*
                    188:         * Unlock the node (either the lower or the alias)
                    189:         */
                    190:        VOP_UNLOCK(vp);
                    191:        /*
                    192:         * Make sure the node alias worked
                    193:         */
                    194:        if (error) {
                    195:                vrele(lowerrootvp);
                    196:                free(amp, M_UFSMNT);    /* XXX */
                    197:                return (error);
                    198:        }
                    199: 
                    200:        /*
                    201:         * Keep a held reference to the root vnode.
                    202:         * It is vrele'd in umapfs_unmount.
                    203:         */
                    204:        umapm_rootvp = vp;
                    205:        umapm_rootvp->v_flag |= VROOT;
                    206:        amp->umapm_rootvp = umapm_rootvp;
                    207:        if (UMAPVPTOLOWERVP(umapm_rootvp)->v_mount->mnt_flag & MNT_LOCAL)
                    208:                mp->mnt_flag |= MNT_LOCAL;
                    209:        mp->mnt_data = (qaddr_t) amp;
                    210:        getnewfsid(mp, makefstype(MOUNT_LOFS));
                    211: 
                    212:        (void) copyinstr(path, mp->mnt_stat.f_mntonname, MNAMELEN - 1, &size);
                    213:        bzero(mp->mnt_stat.f_mntonname + size, MNAMELEN - size);
                    214:        (void) copyinstr(args.target, mp->mnt_stat.f_mntfromname, MNAMELEN - 1, 
                    215:            &size);
                    216:        bzero(mp->mnt_stat.f_mntfromname + size, MNAMELEN - size);
                    217: #ifdef UMAPFS_DIAGNOSTIC
                    218:        printf("umapfs_mount: lower %s, alias at %s\n",
                    219:                mp->mnt_stat.f_mntfromname, mp->mnt_stat.f_mntonname);
                    220: #endif
                    221:        return (0);
                    222: }
                    223: 
                    224: /*
                    225:  * VFS start.  Nothing needed here - the start routine
                    226:  * on the underlying filesystem will have been called
                    227:  * when that filesystem was mounted.
                    228:  */
                    229: int
                    230: umapfs_start(mp, flags, p)
                    231:        struct mount *mp;
                    232:        int flags;
                    233:        struct proc *p;
                    234: {
                    235: 
                    236:        return (0);
                    237:        /* return (VFS_START(MOUNTTOUMAPMOUNT(mp)->umapm_vfs, flags, p)); */
                    238: }
                    239: 
                    240: /*
                    241:  * Free reference to umap layer
                    242:  */
                    243: int
                    244: umapfs_unmount(mp, mntflags, p)
                    245:        struct mount *mp;
                    246:        int mntflags;
                    247:        struct proc *p;
                    248: {
                    249:        struct vnode *umapm_rootvp = MOUNTTOUMAPMOUNT(mp)->umapm_rootvp;
                    250:        int error;
                    251:        int flags = 0;
                    252:        extern int doforce;
                    253: 
                    254: #ifdef UMAPFS_DIAGNOSTIC
                    255:        printf("umapfs_unmount(mp = %x)\n", mp);
                    256: #endif
                    257: 
                    258:        if (mntflags & MNT_FORCE) {
                    259:                /* lofs can never be rootfs so don't check for it */
                    260:                if (!doforce)
                    261:                        return (EINVAL);
                    262:                flags |= FORCECLOSE;
                    263:        }
                    264: 
                    265:        /*
                    266:         * Clear out buffer cache.  I don't think we
                    267:         * ever get anything cached at this level at the
                    268:         * moment, but who knows...
                    269:         */
                    270: #ifdef notyet
                    271:        mntflushbuf(mp, 0); 
                    272:        if (mntinvalbuf(mp, 1))
                    273:                return (EBUSY);
                    274: #endif
                    275:        if (umapm_rootvp->v_usecount > 1)
                    276:                return (EBUSY);
                    277:        if (error = vflush(mp, umapm_rootvp, flags))
                    278:                return (error);
                    279: 
                    280: #ifdef UMAPFS_DIAGNOSTIC
                    281:        vprint("alias root of lower", umapm_rootvp);
                    282: #endif  
                    283:        /*
                    284:         * Release reference on underlying root vnode
                    285:         */
                    286:        vrele(umapm_rootvp);
                    287:        /*
                    288:         * And blow it away for future re-use
                    289:         */
                    290:        vgone(umapm_rootvp);
                    291:        /*
                    292:         * Finally, throw away the umap_mount structure
                    293:         */
                    294:        free(mp->mnt_data, M_UFSMNT);   /* XXX */
                    295:        mp->mnt_data = 0;
                    296:        return (0);
                    297: }
                    298: 
                    299: int
                    300: umapfs_root(mp, vpp)
                    301:        struct mount *mp;
                    302:        struct vnode **vpp;
                    303: {
                    304:        struct vnode *vp;
                    305: 
                    306: #ifdef UMAPFS_DIAGNOSTIC
                    307:        printf("umapfs_root(mp = %x, vp = %x->%x)\n", mp,
                    308:                        MOUNTTOUMAPMOUNT(mp)->umapm_rootvp,
                    309:                        UMAPVPTOLOWERVP(MOUNTTOUMAPMOUNT(mp)->umapm_rootvp)
                    310:                        );
                    311: #endif
                    312: 
                    313:        /*
                    314:         * Return locked reference to root.
                    315:         */
                    316:        vp = MOUNTTOUMAPMOUNT(mp)->umapm_rootvp;
                    317:        VREF(vp);
                    318:        VOP_LOCK(vp);
                    319:        *vpp = vp;
                    320:        return (0);
                    321: }
                    322: 
                    323: int
                    324: umapfs_quotactl(mp, cmd, uid, arg, p)
                    325:        struct mount *mp;
                    326:        int cmd;
                    327:        uid_t uid;
                    328:        caddr_t arg;
                    329:        struct proc *p;
                    330: {
                    331: 
                    332:        return (VFS_QUOTACTL(MOUNTTOUMAPMOUNT(mp)->umapm_vfs, cmd, uid, arg, p));
                    333: }
                    334: 
                    335: int
                    336: umapfs_statfs(mp, sbp, p)
                    337:        struct mount *mp;
                    338:        struct statfs *sbp;
                    339:        struct proc *p;
                    340: {
                    341:        int error;
                    342:        struct statfs mstat;
                    343: 
                    344: #ifdef UMAPFS_DIAGNOSTIC
                    345:        printf("umapfs_statfs(mp = %x, vp = %x->%x)\n", mp,
                    346:                        MOUNTTOUMAPMOUNT(mp)->umapm_rootvp,
                    347:                        UMAPVPTOLOWERVP(MOUNTTOUMAPMOUNT(mp)->umapm_rootvp)
                    348:                        );
                    349: #endif
                    350: 
                    351:        bzero(&mstat, sizeof(mstat));
                    352: 
                    353:        error = VFS_STATFS(MOUNTTOUMAPMOUNT(mp)->umapm_vfs, &mstat, p);
                    354:        if (error)
                    355:                return (error);
                    356: 
                    357:        /* now copy across the "interesting" information and fake the rest */
                    358:        sbp->f_type = mstat.f_type;
                    359:        sbp->f_flags = mstat.f_flags;
                    360:        sbp->f_bsize = mstat.f_bsize;
                    361:        sbp->f_iosize = mstat.f_iosize;
                    362:        sbp->f_blocks = mstat.f_blocks;
                    363:        sbp->f_bfree = mstat.f_bfree;
                    364:        sbp->f_bavail = mstat.f_bavail;
                    365:        sbp->f_files = mstat.f_files;
                    366:        sbp->f_ffree = mstat.f_ffree;
                    367:        if (sbp != &mp->mnt_stat) {
                    368:                bcopy(&mp->mnt_stat.f_fsid, &sbp->f_fsid, sizeof(sbp->f_fsid));
                    369:                bcopy(mp->mnt_stat.f_mntonname, sbp->f_mntonname, MNAMELEN);
                    370:                bcopy(mp->mnt_stat.f_mntfromname, sbp->f_mntfromname, MNAMELEN);
                    371:        }
                    372:        strncpy(sbp->f_fstypename, mp->mnt_op->vfs_name, MFSNAMELEN);
                    373:        sbp->f_fstypename[MFSNAMELEN] = '\0';
                    374:        return (0);
                    375: }
                    376: 
                    377: int
                    378: umapfs_sync(mp, waitfor, cred, p)
                    379:        struct mount *mp;
                    380:        int waitfor;
                    381:        struct ucred *cred;
                    382:        struct proc *p;
                    383: {
                    384: 
                    385:        /*
                    386:         * XXX - Assumes no data cached at umap layer.
                    387:         */
                    388:        return (0);
                    389: }
                    390: 
                    391: int
                    392: umapfs_vget(mp, ino, vpp)
                    393:        struct mount *mp;
                    394:        ino_t ino;
                    395:        struct vnode **vpp;
                    396: {
                    397:        
                    398:        return (VFS_VGET(MOUNTTOUMAPMOUNT(mp)->umapm_vfs, ino, vpp));
                    399: }
                    400: 
                    401: int
                    402: umapfs_fhtovp(mp, fidp, nam, vpp, exflagsp, credanonp)
                    403:        struct mount *mp;
                    404:        struct fid *fidp;
                    405:        struct mbuf *nam;
                    406:        struct vnode **vpp;
                    407:        int *exflagsp;
                    408:        struct ucred**credanonp;
                    409: {
                    410: 
                    411:        return (EOPNOTSUPP);
                    412: }
                    413: 
                    414: int
                    415: umapfs_vptofh(vp, fhp)
                    416:        struct vnode *vp;
                    417:        struct fid *fhp;
                    418: {
                    419: 
                    420:        return (EOPNOTSUPP);
                    421: }
                    422: 
                    423: int umapfs_init __P((void));
                    424: 
                    425: struct vfsops umap_vfsops = {
                    426:        MOUNT_UMAP,
                    427:        umapfs_mount,
                    428:        umapfs_start,
                    429:        umapfs_unmount,
                    430:        umapfs_root,
                    431:        umapfs_quotactl,
                    432:        umapfs_statfs,
                    433:        umapfs_sync,
                    434:        umapfs_vget,
                    435:        umapfs_fhtovp,
                    436:        umapfs_vptofh,
                    437:        umapfs_init,
                    438: };

unix.superglobalmegacorp.com

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