[BACK]Return to mfs_vfsops.c CVS log [TXT][DIR] Up to [local] / sys / ufs / mfs

Annotation of sys/ufs/mfs/mfs_vfsops.c, Revision 1.1.1.1

1.1       nbrk        1: /*     $OpenBSD: mfs_vfsops.c,v 1.34 2006/06/25 15:01:54 sturm Exp $   */
                      2: /*     $NetBSD: mfs_vfsops.c,v 1.10 1996/02/09 22:31:28 christos Exp $ */
                      3:
                      4: /*
                      5:  * Copyright (c) 1989, 1990, 1993, 1994
                      6:  *     The Regents of the University of California.  All rights reserved.
                      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. Neither the name of the University nor the names of its contributors
                     17:  *    may be used to endorse or promote products derived from this software
                     18:  *    without specific prior written permission.
                     19:  *
                     20:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
                     21:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
                     22:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
                     23:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
                     24:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
                     25:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
                     26:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
                     27:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
                     28:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
                     29:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
                     30:  * SUCH DAMAGE.
                     31:  *
                     32:  *     @(#)mfs_vfsops.c        8.4 (Berkeley) 4/16/94
                     33:  */
                     34:
                     35: #include <sys/param.h>
                     36: #include <sys/systm.h>
                     37: #include <sys/time.h>
                     38: #include <sys/kernel.h>
                     39: #include <sys/proc.h>
                     40: #include <sys/buf.h>
                     41: #include <sys/mount.h>
                     42: #include <sys/signalvar.h>
                     43: #include <sys/vnode.h>
                     44: #include <sys/malloc.h>
                     45: #include <sys/kthread.h>
                     46:
                     47: #include <ufs/ufs/quota.h>
                     48: #include <ufs/ufs/inode.h>
                     49: #include <ufs/ufs/ufsmount.h>
                     50: #include <ufs/ufs/ufs_extern.h>
                     51:
                     52: #include <ufs/ffs/fs.h>
                     53: #include <ufs/ffs/ffs_extern.h>
                     54:
                     55: #include <ufs/mfs/mfsnode.h>
                     56: #include <ufs/mfs/mfs_extern.h>
                     57:
                     58: caddr_t        mfs_rootbase;   /* address of mini-root in kernel virtual memory */
                     59: u_long mfs_rootsize;   /* size of mini-root in bytes */
                     60:
                     61: static int mfs_minor;  /* used for building internal dev_t */
                     62:
                     63: extern int (**mfs_vnodeop_p)(void *);
                     64:
                     65: /*
                     66:  * mfs vfs operations.
                     67:  */
                     68: const struct vfsops mfs_vfsops = {
                     69:        mfs_mount,
                     70:        mfs_start,
                     71:        ffs_unmount,
                     72:        ufs_root,
                     73:        ufs_quotactl,
                     74:        mfs_statfs,
                     75:        ffs_sync,
                     76:        ffs_vget,
                     77:        ffs_fhtovp,
                     78:        ffs_vptofh,
                     79:        mfs_init,
                     80:        ffs_sysctl,
                     81:        mfs_checkexp
                     82: };
                     83:
                     84: /*
                     85:  * Called by main() when mfs is going to be mounted as root.
                     86:  */
                     87:
                     88: int
                     89: mfs_mountroot(void)
                     90: {
                     91:        struct fs *fs;
                     92:        struct mount *mp;
                     93:        struct proc *p = curproc;
                     94:        struct ufsmount *ump;
                     95:        struct mfsnode *mfsp;
                     96:        int error;
                     97:
                     98:        if ((error = bdevvp(swapdev, &swapdev_vp)) ||
                     99:            (error = bdevvp(rootdev, &rootvp))) {
                    100:                printf("mfs_mountroot: can't setup bdevvp's");
                    101:                return (error);
                    102:        }
                    103:        if ((error = vfs_rootmountalloc("mfs", "mfs_root", &mp)) != 0)
                    104:                return (error);
                    105:        mfsp = malloc(sizeof *mfsp, M_MFSNODE, M_WAITOK);
                    106:        rootvp->v_data = mfsp;
                    107:        rootvp->v_op = mfs_vnodeop_p;
                    108:        rootvp->v_tag = VT_MFS;
                    109:        mfsp->mfs_baseoff = mfs_rootbase;
                    110:        mfsp->mfs_size = mfs_rootsize;
                    111:        mfsp->mfs_vnode = rootvp;
                    112:        mfsp->mfs_pid = p->p_pid;
                    113:        mfsp->mfs_buflist = (struct buf *)0;
                    114:        if ((error = ffs_mountfs(rootvp, mp, p)) != 0) {
                    115:                mp->mnt_vfc->vfc_refcount--;
                    116:                vfs_unbusy(mp);
                    117:                free(mp, M_MOUNT);
                    118:                free(mfsp, M_MFSNODE);
                    119:                return (error);
                    120:        }
                    121:
                    122:        CIRCLEQ_INSERT_TAIL(&mountlist, mp, mnt_list);
                    123:        ump = VFSTOUFS(mp);
                    124:        fs = ump->um_fs;
                    125:        (void) copystr(mp->mnt_stat.f_mntonname, fs->fs_fsmnt, MNAMELEN - 1, 0);
                    126:        (void)ffs_statfs(mp, &mp->mnt_stat, p);
                    127:        vfs_unbusy(mp);
                    128:        inittodr((time_t)0);
                    129:
                    130:        return (0);
                    131: }
                    132:
                    133: /*
                    134:  * This is called early in boot to set the base address and size
                    135:  * of the mini-root.
                    136:  */
                    137: int
                    138: mfs_initminiroot(caddr_t base)
                    139: {
                    140:        struct fs *fs = (struct fs *)(base + SBOFF);
                    141:        extern int (*mountroot)(void);
                    142:
                    143:        /* check for valid super block */
                    144:        if (fs->fs_magic != FS_MAGIC || fs->fs_bsize > MAXBSIZE ||
                    145:            fs->fs_bsize < sizeof(struct fs))
                    146:                return (0);
                    147:        mountroot = mfs_mountroot;
                    148:        mfs_rootbase = base;
                    149:        mfs_rootsize = fs->fs_fsize * fs->fs_size;
                    150:        rootdev = makedev(255, mfs_minor);
                    151:        mfs_minor++;
                    152:        return (mfs_rootsize);
                    153: }
                    154:
                    155: /*
                    156:  * VFS Operations.
                    157:  *
                    158:  * mount system call
                    159:  */
                    160: /* ARGSUSED */
                    161: int
                    162: mfs_mount(struct mount *mp, const char *path, void *data,
                    163:     struct nameidata *ndp, struct proc *p)
                    164: {
                    165:        struct vnode *devvp;
                    166:        struct mfs_args args;
                    167:        struct ufsmount *ump;
                    168:        struct fs *fs;
                    169:        struct mfsnode *mfsp;
                    170:        size_t size;
                    171:        int flags, error;
                    172:
                    173:        error = copyin(data, (caddr_t)&args, sizeof (struct mfs_args));
                    174:        if (error)
                    175:                return (error);
                    176:
                    177:        /*
                    178:         * If updating, check whether changing from read-only to
                    179:         * read/write; if there is no device name, that's all we do.
                    180:         */
                    181:        if (mp->mnt_flag & MNT_UPDATE) {
                    182:                ump = VFSTOUFS(mp);
                    183:                fs = ump->um_fs;
                    184:                if (fs->fs_ronly == 0 && (mp->mnt_flag & MNT_RDONLY)) {
                    185:                        flags = WRITECLOSE;
                    186:                        if (mp->mnt_flag & MNT_FORCE)
                    187:                                flags |= FORCECLOSE;
                    188:                        error = ffs_flushfiles(mp, flags, p);
                    189:                        if (error)
                    190:                                return (error);
                    191:                }
                    192:                if (fs->fs_ronly && (mp->mnt_flag & MNT_WANTRDWR))
                    193:                        fs->fs_ronly = 0;
                    194: #ifdef EXPORTMFS
                    195:                if (args.fspec == 0)
                    196:                        return (vfs_export(mp, &ump->um_export,
                    197:                            &args.export_info));
                    198: #endif
                    199:                return (0);
                    200:        }
                    201:        error = getnewvnode(VT_MFS, (struct mount *)0, mfs_vnodeop_p, &devvp);
                    202:        if (error)
                    203:                return (error);
                    204:        devvp->v_type = VBLK;
                    205:        if (checkalias(devvp, makedev(255, mfs_minor), (struct mount *)0))
                    206:                panic("mfs_mount: dup dev");
                    207:        mfs_minor++;
                    208:        mfsp = malloc(sizeof *mfsp, M_MFSNODE, M_WAITOK);
                    209:        devvp->v_data = mfsp;
                    210:        mfsp->mfs_baseoff = args.base;
                    211:        mfsp->mfs_size = args.size;
                    212:        mfsp->mfs_vnode = devvp;
                    213:        mfsp->mfs_pid = p->p_pid;
                    214:        mfsp->mfs_buflist = (struct buf *)0;
                    215:        if ((error = ffs_mountfs(devvp, mp, p)) != 0) {
                    216:                mfsp->mfs_buflist = (struct buf *)-1;
                    217:                vrele(devvp);
                    218:                return (error);
                    219:        }
                    220:        ump = VFSTOUFS(mp);
                    221:        fs = ump->um_fs;
                    222:        (void) copyinstr(path, fs->fs_fsmnt, sizeof(fs->fs_fsmnt) - 1, &size);
                    223:        bzero(fs->fs_fsmnt + size, sizeof(fs->fs_fsmnt) - size);
                    224:        bcopy(fs->fs_fsmnt, mp->mnt_stat.f_mntonname, MNAMELEN);
                    225:        (void) copyinstr(args.fspec, mp->mnt_stat.f_mntfromname, MNAMELEN - 1,
                    226:            &size);
                    227:        bzero(mp->mnt_stat.f_mntfromname + size, MNAMELEN - size);
                    228:        bcopy(&args, &mp->mnt_stat.mount_info.mfs_args, sizeof(args));
                    229:        return (0);
                    230: }
                    231:
                    232: int    mfs_pri = PWAIT | PCATCH;               /* XXX prob. temp */
                    233:
                    234: /*
                    235:  * Used to grab the process and keep it in the kernel to service
                    236:  * memory filesystem I/O requests.
                    237:  *
                    238:  * Loop servicing I/O requests.
                    239:  * Copy the requested data into or out of the memory filesystem
                    240:  * address space.
                    241:  */
                    242: /* ARGSUSED */
                    243: int
                    244: mfs_start(struct mount *mp, int flags, struct proc *p)
                    245: {
                    246:        struct vnode *vp = VFSTOUFS(mp)->um_devvp;
                    247:        struct mfsnode *mfsp = VTOMFS(vp);
                    248:        struct buf *bp;
                    249:        caddr_t base;
                    250:        int sleepreturn = 0;
                    251:
                    252:        base = mfsp->mfs_baseoff;
                    253:        while (mfsp->mfs_buflist != (struct buf *)-1) {
                    254:                while ((bp = mfsp->mfs_buflist) != NULL) {
                    255:                        mfsp->mfs_buflist = bp->b_actf;
                    256:                        mfs_doio(bp, base);
                    257:                        wakeup((caddr_t)bp);
                    258:                }
                    259:                /*
                    260:                 * If a non-ignored signal is received, try to unmount.
                    261:                 * If that fails, clear the signal (it has been "processed"),
                    262:                 * otherwise we will loop here, as tsleep will always return
                    263:                 * EINTR/ERESTART.
                    264:                 */
                    265:                if (sleepreturn != 0) {
                    266:                        if (vfs_busy(mp, VB_WRITE|VB_NOWAIT) ||
                    267:                            dounmount(mp, 0, p, NULL))
                    268:                                CLRSIG(p, CURSIG(p));
                    269:                        sleepreturn = 0;
                    270:                        continue;
                    271:                }
                    272:                sleepreturn = tsleep((caddr_t)vp, mfs_pri, "mfsidl", 0);
                    273:        }
                    274:        return (0);
                    275: }
                    276:
                    277: /*
                    278:  * Get file system statistics.
                    279:  */
                    280: int
                    281: mfs_statfs(struct mount *mp, struct statfs *sbp, struct proc *p)
                    282: {
                    283:        int error;
                    284:
                    285:        error = ffs_statfs(mp, sbp, p);
                    286:        strncpy(&sbp->f_fstypename[0], mp->mnt_vfc->vfc_name, MFSNAMELEN);
                    287:        if (sbp != &mp->mnt_stat)
                    288:                bcopy(&mp->mnt_stat.mount_info.mfs_args,
                    289:                    &sbp->mount_info.mfs_args, sizeof(struct mfs_args));
                    290:        return (error);
                    291: }
                    292:
                    293: /*
                    294:  * check export permission, not supported
                    295:  */
                    296: /* ARGUSED */
                    297: int
                    298: mfs_checkexp(struct mount *mp, struct mbuf *nam, int *exflagsp,
                    299:     struct ucred **credanonp)
                    300: {
                    301:        return (EOPNOTSUPP);
                    302: }
                    303:
                    304: /*
                    305:  * Memory based filesystem initialization.
                    306:  */
                    307: int
                    308: mfs_init(struct vfsconf *vfsp)
                    309: {
                    310:        return (ffs_init(vfsp));
                    311: }

CVSweb