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

Annotation of sys/ufs/ext2fs/ext2fs_vfsops.c, Revision 1.1

1.1     ! nbrk        1: /*     $OpenBSD: ext2fs_vfsops.c,v 1.48 2007/06/17 20:15:25 jasper Exp $       */
        !             2: /*     $NetBSD: ext2fs_vfsops.c,v 1.1 1997/06/11 09:34:07 bouyer Exp $ */
        !             3:
        !             4: /*
        !             5:  * Copyright (c) 1997 Manuel Bouyer.
        !             6:  * Copyright (c) 1989, 1991, 1993, 1994
        !             7:  *     The Regents of the University of California.  All rights reserved.
        !             8:  *
        !             9:  * Redistribution and use in source and binary forms, with or without
        !            10:  * modification, are permitted provided that the following conditions
        !            11:  * are met:
        !            12:  * 1. Redistributions of source code must retain the above copyright
        !            13:  *     notice, this list of conditions and the following disclaimer.
        !            14:  * 2. Redistributions in binary form must reproduce the above copyright
        !            15:  *     notice, this list of conditions and the following disclaimer in the
        !            16:  *     documentation and/or other materials provided with the distribution.
        !            17:  * 3. 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:  *     @(#)ffs_vfsops.c        8.14 (Berkeley) 11/28/94
        !            34:  * Modified for ext2fs by Manuel Bouyer.
        !            35:  */
        !            36:
        !            37: #include <sys/param.h>
        !            38: #include <sys/systm.h>
        !            39: #include <sys/namei.h>
        !            40: #include <sys/proc.h>
        !            41: #include <sys/kernel.h>
        !            42: #include <sys/vnode.h>
        !            43: #include <sys/socket.h>
        !            44: #include <sys/mount.h>
        !            45: #include <sys/buf.h>
        !            46: #include <sys/device.h>
        !            47: #include <sys/mbuf.h>
        !            48: #include <sys/file.h>
        !            49: #include <sys/disklabel.h>
        !            50: #include <sys/ioctl.h>
        !            51: #include <sys/errno.h>
        !            52: #include <sys/malloc.h>
        !            53: #include <sys/pool.h>
        !            54: #include <sys/lock.h>
        !            55:
        !            56: #include <miscfs/specfs/specdev.h>
        !            57:
        !            58: #include <ufs/ufs/quota.h>
        !            59: #include <ufs/ufs/ufsmount.h>
        !            60: #include <ufs/ufs/inode.h>
        !            61: #include <ufs/ufs/dir.h>
        !            62: #include <ufs/ufs/ufs_extern.h>
        !            63:
        !            64: #include <ufs/ext2fs/ext2fs.h>
        !            65: #include <ufs/ext2fs/ext2fs_extern.h>
        !            66:
        !            67: extern struct lock ufs_hashlock;
        !            68:
        !            69: int ext2fs_sbupdate(struct ufsmount *, int);
        !            70: static int ext2fs_checksb(struct ext2fs *, int);
        !            71:
        !            72: extern struct vnodeopv_desc ext2fs_vnodeop_opv_desc;
        !            73: extern struct vnodeopv_desc ext2fs_specop_opv_desc;
        !            74: #ifdef FIFO
        !            75: extern struct vnodeopv_desc ext2fs_fifoop_opv_desc;
        !            76: #endif
        !            77:
        !            78: struct vnodeopv_desc *ext2fs_vnodeopv_descs[] = {
        !            79:        &ext2fs_vnodeop_opv_desc,
        !            80:        &ext2fs_specop_opv_desc,
        !            81: #ifdef FIFO
        !            82:        &ext2fs_fifoop_opv_desc,
        !            83: #endif
        !            84:        NULL,
        !            85: };
        !            86:
        !            87: const struct vfsops ext2fs_vfsops = {
        !            88:        ext2fs_mount,
        !            89:        ufs_start,
        !            90:        ext2fs_unmount,
        !            91:        ufs_root,
        !            92:        ufs_quotactl,
        !            93:        ext2fs_statfs,
        !            94:        ext2fs_sync,
        !            95:        ext2fs_vget,
        !            96:        ext2fs_fhtovp,
        !            97:        ext2fs_vptofh,
        !            98:        ext2fs_init,
        !            99:        ext2fs_sysctl,
        !           100:        ufs_check_export
        !           101: };
        !           102:
        !           103: struct pool ext2fs_inode_pool;
        !           104: struct pool ext2fs_dinode_pool;
        !           105:
        !           106: extern u_long ext2gennumber;
        !           107:
        !           108: int
        !           109: ext2fs_init(struct vfsconf *vfsp)
        !           110: {
        !           111:        pool_init(&ext2fs_inode_pool, sizeof(struct inode), 0, 0, 0,
        !           112:            "ext2inopl", &pool_allocator_nointr);
        !           113:        pool_init(&ext2fs_dinode_pool, sizeof(struct ext2fs_dinode), 0, 0, 0,
        !           114:            "ext2dinopl", &pool_allocator_nointr);
        !           115:
        !           116:        return (ufs_init(vfsp));
        !           117: }
        !           118:
        !           119: /*
        !           120:  * Called by main() when ext2fs is going to be mounted as root.
        !           121:  *
        !           122:  * Name is updated by mount(8) after booting.
        !           123:  */
        !           124: #define ROOTNAME       "root_device"
        !           125:
        !           126: int
        !           127: ext2fs_mountroot(void)
        !           128: {
        !           129:        struct m_ext2fs *fs;
        !           130:         struct mount *mp;
        !           131:        struct proc *p = curproc;       /* XXX */
        !           132:        struct ufsmount *ump;
        !           133:        int error;
        !           134:
        !           135:        /*
        !           136:         * Get vnodes for swapdev and rootdev.
        !           137:         */
        !           138:        if (bdevvp(swapdev, &swapdev_vp) || bdevvp(rootdev, &rootvp))
        !           139:                panic("ext2fs_mountroot: can't setup bdevvp's");
        !           140:
        !           141:        if ((error = vfs_rootmountalloc("ext2fs", "root_device", &mp)) != 0) {
        !           142:                vrele(rootvp);
        !           143:                return (error);
        !           144:        }
        !           145:
        !           146:        if ((error = ext2fs_mountfs(rootvp, mp, p)) != 0) {
        !           147:                mp->mnt_vfc->vfc_refcount--;
        !           148:                vfs_unbusy(mp);
        !           149:                free(mp, M_MOUNT);
        !           150:                vrele(rootvp);
        !           151:                return (error);
        !           152:        }
        !           153:
        !           154:        CIRCLEQ_INSERT_TAIL(&mountlist, mp, mnt_list);
        !           155:        ump = VFSTOUFS(mp);
        !           156:        fs = ump->um_e2fs;
        !           157:        bzero(fs->e2fs_fsmnt, sizeof(fs->e2fs_fsmnt));
        !           158:        (void) copystr(mp->mnt_stat.f_mntonname, fs->e2fs_fsmnt,
        !           159:            sizeof(fs->e2fs_fsmnt) - 1, 0);
        !           160:        if (fs->e2fs.e2fs_rev > E2FS_REV0) {
        !           161:                bzero(fs->e2fs.e2fs_fsmnt, sizeof(fs->e2fs.e2fs_fsmnt));
        !           162:                (void) copystr(mp->mnt_stat.f_mntonname, fs->e2fs.e2fs_fsmnt,
        !           163:                    sizeof(fs->e2fs.e2fs_fsmnt) - 1, 0);
        !           164:        }
        !           165:        (void)ext2fs_statfs(mp, &mp->mnt_stat, p);
        !           166:        vfs_unbusy(mp);
        !           167:        inittodr(fs->e2fs.e2fs_wtime);
        !           168:        return (0);
        !           169: }
        !           170:
        !           171: /*
        !           172:  * VFS Operations.
        !           173:  *
        !           174:  * mount system call
        !           175:  */
        !           176: int
        !           177: ext2fs_mount(struct mount *mp, const char *path, void *data,
        !           178:     struct nameidata *ndp, struct proc *p)
        !           179: {
        !           180:        struct vnode *devvp;
        !           181:        struct ufs_args args;
        !           182:        struct ufsmount *ump = NULL;
        !           183:        struct m_ext2fs *fs;
        !           184:        size_t size;
        !           185:        int error, flags;
        !           186:        mode_t accessmode;
        !           187:
        !           188:        error = copyin(data, (caddr_t)&args, sizeof (struct ufs_args));
        !           189:        if (error)
        !           190:                return (error);
        !           191:        /*
        !           192:         * If updating, check whether changing from read-only to
        !           193:         * read/write; if there is no device name, that's all we do.
        !           194:         */
        !           195:        if (mp->mnt_flag & MNT_UPDATE) {
        !           196:                ump = VFSTOUFS(mp);
        !           197:                fs = ump->um_e2fs;
        !           198:                if (fs->e2fs_ronly == 0 && (mp->mnt_flag & MNT_RDONLY)) {
        !           199:                        flags = WRITECLOSE;
        !           200:                        if (mp->mnt_flag & MNT_FORCE)
        !           201:                                flags |= FORCECLOSE;
        !           202:                        error = ext2fs_flushfiles(mp, flags, p);
        !           203:                        if (error == 0 &&
        !           204:                                ext2fs_cgupdate(ump, MNT_WAIT) == 0 &&
        !           205:                                (fs->e2fs.e2fs_state & E2FS_ERRORS) == 0) {
        !           206:                                fs->e2fs.e2fs_state = E2FS_ISCLEAN;
        !           207:                                (void) ext2fs_sbupdate(ump, MNT_WAIT);
        !           208:                        }
        !           209:                        if (error)
        !           210:                                return (error);
        !           211:                        fs->e2fs_ronly = 1;
        !           212:                }
        !           213:                if (mp->mnt_flag & MNT_RELOAD) {
        !           214:                        error = ext2fs_reload(mp, ndp->ni_cnd.cn_cred, p);
        !           215:                        if (error)
        !           216:                                return (error);
        !           217:                }
        !           218:                if (fs->e2fs_ronly && (mp->mnt_flag & MNT_WANTRDWR)) {
        !           219:                        /*
        !           220:                         * If upgrade to read-write by non-root, then verify
        !           221:                         * that user has necessary permissions on the device.
        !           222:                         */
        !           223:                        if (p->p_ucred->cr_uid != 0) {
        !           224:                                devvp = ump->um_devvp;
        !           225:                                vn_lock(devvp, LK_EXCLUSIVE | LK_RETRY, p);
        !           226:                                error = VOP_ACCESS(devvp, VREAD | VWRITE,
        !           227:                                                   p->p_ucred, p);
        !           228:                                if (error) {
        !           229:                                        VOP_UNLOCK(devvp, 0, p);
        !           230:                                        return (error);
        !           231:                                }
        !           232:                                VOP_UNLOCK(devvp, 0, p);
        !           233:                        }
        !           234:                        fs->e2fs_ronly = 0;
        !           235:                        if (fs->e2fs.e2fs_state == E2FS_ISCLEAN)
        !           236:                                fs->e2fs.e2fs_state = 0;
        !           237:                        else
        !           238:                                fs->e2fs.e2fs_state = E2FS_ERRORS;
        !           239:                        fs->e2fs_fmod = 1;
        !           240:                }
        !           241:                if (args.fspec == 0) {
        !           242:                        /*
        !           243:                         * Process export requests.
        !           244:                         */
        !           245:                        return (vfs_export(mp, &ump->um_export,
        !           246:                            &args.export_info));
        !           247:                }
        !           248:        }
        !           249:        /*
        !           250:         * Not an update, or updating the name: look up the name
        !           251:         * and verify that it refers to a sensible block device.
        !           252:         */
        !           253:        NDINIT(ndp, LOOKUP, FOLLOW, UIO_USERSPACE, args.fspec, p);
        !           254:        if ((error = namei(ndp)) != 0)
        !           255:                return (error);
        !           256:        devvp = ndp->ni_vp;
        !           257:
        !           258:        if (devvp->v_type != VBLK) {
        !           259:                vrele(devvp);
        !           260:                return (ENOTBLK);
        !           261:        }
        !           262:        if (major(devvp->v_rdev) >= nblkdev) {
        !           263:                vrele(devvp);
        !           264:                return (ENXIO);
        !           265:        }
        !           266:        /*
        !           267:         * If mount by non-root, then verify that user has necessary
        !           268:         * permissions on the device.
        !           269:         */
        !           270:        if (p->p_ucred->cr_uid != 0) {
        !           271:                accessmode = VREAD;
        !           272:                if ((mp->mnt_flag & MNT_RDONLY) == 0)
        !           273:                        accessmode |= VWRITE;
        !           274:                vn_lock(devvp, LK_EXCLUSIVE | LK_RETRY, p);
        !           275:                error = VOP_ACCESS(devvp, accessmode, p->p_ucred, p);
        !           276:                if (error) {
        !           277:                        vput(devvp);
        !           278:                        return (error);
        !           279:                }
        !           280:                VOP_UNLOCK(devvp, 0, p);
        !           281:        }
        !           282:        if ((mp->mnt_flag & MNT_UPDATE) == 0)
        !           283:                error = ext2fs_mountfs(devvp, mp, p);
        !           284:        else {
        !           285:                if (devvp != ump->um_devvp)
        !           286:                        error = EINVAL; /* XXX needs translation */
        !           287:                else
        !           288:                        vrele(devvp);
        !           289:        }
        !           290:        if (error) {
        !           291:                vrele(devvp);
        !           292:                return (error);
        !           293:        }
        !           294:        ump = VFSTOUFS(mp);
        !           295:        fs = ump->um_e2fs;
        !           296:        (void) copyinstr(path, fs->e2fs_fsmnt, sizeof(fs->e2fs_fsmnt) - 1,
        !           297:            &size);
        !           298:        bzero(fs->e2fs_fsmnt + size, sizeof(fs->e2fs_fsmnt) - size);
        !           299:        if (fs->e2fs.e2fs_rev > E2FS_REV0) {
        !           300:                (void) copystr(mp->mnt_stat.f_mntonname, fs->e2fs.e2fs_fsmnt,
        !           301:                    sizeof(fs->e2fs.e2fs_fsmnt) - 1, &size);
        !           302:                bzero(fs->e2fs.e2fs_fsmnt, sizeof(fs->e2fs.e2fs_fsmnt) - size);
        !           303:        }
        !           304:        bcopy(fs->e2fs_fsmnt, mp->mnt_stat.f_mntonname, MNAMELEN);
        !           305:        (void) copyinstr(args.fspec, mp->mnt_stat.f_mntfromname, MNAMELEN - 1,
        !           306:                &size);
        !           307:        bzero(mp->mnt_stat.f_mntfromname + size, MNAMELEN - size);
        !           308:        if (fs->e2fs_fmod != 0) {       /* XXX */
        !           309:                fs->e2fs_fmod = 0;
        !           310:                if (fs->e2fs.e2fs_state == 0)
        !           311:                        fs->e2fs.e2fs_wtime = time_second;
        !           312:                else
        !           313:                        printf("%s: file system not clean; please fsck(8)\n",
        !           314:                                mp->mnt_stat.f_mntfromname);
        !           315:                (void) ext2fs_cgupdate(ump, MNT_WAIT);
        !           316:        }
        !           317:        return (0);
        !           318: }
        !           319:
        !           320: int ext2fs_reload_vnode(struct vnode *, void *args);
        !           321:
        !           322: struct ext2fs_reload_args {
        !           323:        struct m_ext2fs *fs;
        !           324:        struct proc *p;
        !           325:        struct ucred *cred;
        !           326:        struct vnode *devvp;
        !           327: };
        !           328:
        !           329: int
        !           330: ext2fs_reload_vnode(struct vnode *vp, void *args)
        !           331: {
        !           332:        struct ext2fs_reload_args *era = args;
        !           333:        struct buf *bp;
        !           334:        struct inode *ip;
        !           335:        int error;
        !           336:        caddr_t cp;
        !           337:
        !           338:        /*
        !           339:         * Step 4: invalidate all inactive vnodes.
        !           340:         */
        !           341:        if (vp->v_usecount == 0) {
        !           342:                vgonel(vp, era->p);
        !           343:                return (0);
        !           344:        }
        !           345:
        !           346:        /*
        !           347:         * Step 5: invalidate all cached file data.
        !           348:         */
        !           349:        if (vget(vp, LK_EXCLUSIVE, era->p))
        !           350:                return (0);
        !           351:
        !           352:        if (vinvalbuf(vp, 0, era->cred, era->p, 0, 0))
        !           353:                panic("ext2fs_reload: dirty2");
        !           354:        /*
        !           355:         * Step 6: re-read inode data for all active vnodes.
        !           356:         */
        !           357:        ip = VTOI(vp);
        !           358:        error = bread(era->devvp,
        !           359:            fsbtodb(era->fs, ino_to_fsba(era->fs, ip->i_number)),
        !           360:            (int)era->fs->e2fs_bsize, NOCRED, &bp);
        !           361:        if (error) {
        !           362:                vput(vp);
        !           363:                return (error);
        !           364:        }
        !           365:        cp = (caddr_t)bp->b_data +
        !           366:            (ino_to_fsbo(era->fs, ip->i_number) * EXT2_DINODE_SIZE);
        !           367:        e2fs_iload((struct ext2fs_dinode *)cp, ip->i_e2din);
        !           368:        brelse(bp);
        !           369:        vput(vp);
        !           370:        return (0);
        !           371: }
        !           372:
        !           373: /*
        !           374:  * Reload all incore data for a filesystem (used after running fsck on
        !           375:  * the root filesystem and finding things to fix). The filesystem must
        !           376:  * be mounted read-only.
        !           377:  *
        !           378:  * Things to do to update the mount:
        !           379:  *     1) invalidate all cached meta-data.
        !           380:  *     2) re-read superblock from disk.
        !           381:  *     3) re-read summary information from disk.
        !           382:  *     4) invalidate all inactive vnodes.
        !           383:  *     5) invalidate all cached file data.
        !           384:  *     6) re-read inode data for all active vnodes.
        !           385:  */
        !           386: int
        !           387: ext2fs_reload(struct mount *mountp, struct ucred *cred, struct proc *p)
        !           388: {
        !           389:        struct vnode *devvp;
        !           390:        struct buf *bp;
        !           391:        struct m_ext2fs *fs;
        !           392:        struct ext2fs *newfs;
        !           393:        struct partinfo dpart;
        !           394:        int i, size, error;
        !           395:        struct ext2fs_reload_args era;
        !           396:
        !           397:        if ((mountp->mnt_flag & MNT_RDONLY) == 0)
        !           398:                return (EINVAL);
        !           399:        /*
        !           400:         * Step 1: invalidate all cached meta-data.
        !           401:         */
        !           402:        devvp = VFSTOUFS(mountp)->um_devvp;
        !           403:        if (vinvalbuf(devvp, 0, cred, p, 0, 0))
        !           404:                panic("ext2fs_reload: dirty1");
        !           405:
        !           406:        /*
        !           407:         * Step 2: re-read superblock from disk.
        !           408:         */
        !           409:        if (VOP_IOCTL(devvp, DIOCGPART, (caddr_t)&dpart, FREAD, NOCRED, p) != 0)
        !           410:                size = DEV_BSIZE;
        !           411:        else
        !           412:                size = dpart.disklab->d_secsize;
        !           413:        error = bread(devvp, (int32_t)(SBOFF / size), SBSIZE, NOCRED, &bp);
        !           414:        if (error) {
        !           415:                brelse(bp);
        !           416:                return (error);
        !           417:        }
        !           418:        newfs = (struct ext2fs *)bp->b_data;
        !           419:        error = ext2fs_checksb(newfs, (mountp->mnt_flag & MNT_RDONLY) != 0);
        !           420:        if (error) {
        !           421:                brelse(bp);
        !           422:                return (error);
        !           423:        }
        !           424:
        !           425:        fs = VFSTOUFS(mountp)->um_e2fs;
        !           426:        /*
        !           427:         * copy in new superblock, and compute in-memory values
        !           428:         */
        !           429:        e2fs_sbload(newfs, &fs->e2fs);
        !           430:        fs->e2fs_ncg =
        !           431:            howmany(fs->e2fs.e2fs_bcount - fs->e2fs.e2fs_first_dblock,
        !           432:            fs->e2fs.e2fs_bpg);
        !           433:        /* XXX assume hw bsize = 512 */
        !           434:        fs->e2fs_fsbtodb = fs->e2fs.e2fs_log_bsize + 1;
        !           435:        fs->e2fs_bsize = 1024 << fs->e2fs.e2fs_log_bsize;
        !           436:        fs->e2fs_bshift = LOG_MINBSIZE + fs->e2fs.e2fs_log_bsize;
        !           437:        fs->e2fs_qbmask = fs->e2fs_bsize - 1;
        !           438:        fs->e2fs_bmask = ~fs->e2fs_qbmask;
        !           439:        fs->e2fs_ngdb = howmany(fs->e2fs_ncg,
        !           440:                        fs->e2fs_bsize / sizeof(struct ext2_gd));
        !           441:        fs->e2fs_ipb = fs->e2fs_bsize / EXT2_DINODE_SIZE;
        !           442:        fs->e2fs_itpg = fs->e2fs.e2fs_ipg/fs->e2fs_ipb;
        !           443:
        !           444:        /*
        !           445:         * Step 3: re-read summary information from disk.
        !           446:         */
        !           447:
        !           448:        for (i=0; i < fs->e2fs_ngdb; i++) {
        !           449:                error = bread(devvp ,
        !           450:                    fsbtodb(fs, ((fs->e2fs_bsize>1024)? 0 : 1) + i + 1),
        !           451:                    fs->e2fs_bsize, NOCRED, &bp);
        !           452:                if (error) {
        !           453:                        brelse(bp);
        !           454:                        return (error);
        !           455:                }
        !           456:                e2fs_cgload((struct ext2_gd*)bp->b_data,
        !           457:                    &fs->e2fs_gd[i* fs->e2fs_bsize / sizeof(struct ext2_gd)],
        !           458:                    fs->e2fs_bsize);
        !           459:                brelse(bp);
        !           460:        }
        !           461:
        !           462:        era.p = p;
        !           463:        era.cred = cred;
        !           464:        era.fs = fs;
        !           465:        era.devvp = devvp;
        !           466:
        !           467:        error = vfs_mount_foreach_vnode(mountp, ext2fs_reload_vnode, &era);
        !           468:
        !           469:        return (error);
        !           470: }
        !           471:
        !           472: /*
        !           473:  * Common code for mount and mountroot
        !           474:  */
        !           475: int
        !           476: ext2fs_mountfs(struct vnode *devvp, struct mount *mp, struct proc *p)
        !           477: {
        !           478:        struct ufsmount *ump;
        !           479:        struct buf *bp;
        !           480:        struct ext2fs *fs;
        !           481:        struct m_ext2fs *m_fs;
        !           482:        dev_t dev;
        !           483:        struct partinfo dpart;
        !           484:        int error, i, size, ronly;
        !           485:        struct ucred *cred;
        !           486:
        !           487:        dev = devvp->v_rdev;
        !           488:        cred = p ? p->p_ucred : NOCRED;
        !           489:        /*
        !           490:         * Disallow multiple mounts of the same device.
        !           491:         * Disallow mounting of a device that is currently in use
        !           492:         * (except for root, which might share swap device for miniroot).
        !           493:         * Flush out any old buffers remaining from a previous use.
        !           494:         */
        !           495:        if ((error = vfs_mountedon(devvp)) != 0)
        !           496:                return (error);
        !           497:        if (vcount(devvp) > 1 && devvp != rootvp)
        !           498:                return (EBUSY);
        !           499:        if ((error = vinvalbuf(devvp, V_SAVE, cred, p, 0, 0)) != 0)
        !           500:                return (error);
        !           501:
        !           502:        ronly = (mp->mnt_flag & MNT_RDONLY) != 0;
        !           503:        error = VOP_OPEN(devvp, ronly ? FREAD : FREAD|FWRITE, FSCRED, p);
        !           504:        if (error)
        !           505:                return (error);
        !           506:        if (VOP_IOCTL(devvp, DIOCGPART, (caddr_t)&dpart, FREAD, cred, p) != 0)
        !           507:                size = DEV_BSIZE;
        !           508:        else
        !           509:                size = dpart.disklab->d_secsize;
        !           510:
        !           511:        bp = NULL;
        !           512:        ump = NULL;
        !           513:
        !           514: #ifdef DEBUG_EXT2
        !           515:        printf("sb size: %d ino size %d\n", sizeof(struct ext2fs),
        !           516:            EXT2_DINODE_SIZE);
        !           517: #endif
        !           518:        error = bread(devvp, (SBOFF / DEV_BSIZE), SBSIZE, cred, &bp);
        !           519:        if (error)
        !           520:                goto out;
        !           521:        fs = (struct ext2fs *)bp->b_data;
        !           522:        error = ext2fs_checksb(fs, ronly);
        !           523:        if (error)
        !           524:                goto out;
        !           525:        ump = malloc(sizeof *ump, M_UFSMNT, M_WAITOK);
        !           526:        memset((caddr_t)ump, 0, sizeof *ump);
        !           527:        ump->um_e2fs = malloc(sizeof(struct m_ext2fs), M_UFSMNT, M_WAITOK);
        !           528:        memset((caddr_t)ump->um_e2fs, 0, sizeof(struct m_ext2fs));
        !           529:        e2fs_sbload((struct ext2fs*)bp->b_data, &ump->um_e2fs->e2fs);
        !           530:        brelse(bp);
        !           531:        bp = NULL;
        !           532:        m_fs = ump->um_e2fs;
        !           533:        m_fs->e2fs_ronly = ronly;
        !           534:        ump->um_fstype = UM_EXT2FS;
        !           535:        if (ronly == 0) {
        !           536:                if (m_fs->e2fs.e2fs_state == E2FS_ISCLEAN)
        !           537:                        m_fs->e2fs.e2fs_state = 0;
        !           538:                else
        !           539:                        m_fs->e2fs.e2fs_state = E2FS_ERRORS;
        !           540:                m_fs->e2fs_fmod = 1;
        !           541:        }
        !           542:
        !           543:        /* compute dynamic sb infos */
        !           544:        m_fs->e2fs_ncg =
        !           545:                howmany(m_fs->e2fs.e2fs_bcount - m_fs->e2fs.e2fs_first_dblock,
        !           546:                m_fs->e2fs.e2fs_bpg);
        !           547:        /* XXX assume hw bsize = 512 */
        !           548:        m_fs->e2fs_fsbtodb = m_fs->e2fs.e2fs_log_bsize + 1;
        !           549:        m_fs->e2fs_bsize = 1024 << m_fs->e2fs.e2fs_log_bsize;
        !           550:        m_fs->e2fs_bshift = LOG_MINBSIZE + m_fs->e2fs.e2fs_log_bsize;
        !           551:        m_fs->e2fs_qbmask = m_fs->e2fs_bsize - 1;
        !           552:        m_fs->e2fs_bmask = ~m_fs->e2fs_qbmask;
        !           553:        m_fs->e2fs_ngdb = howmany(m_fs->e2fs_ncg,
        !           554:                m_fs->e2fs_bsize / sizeof(struct ext2_gd));
        !           555:        m_fs->e2fs_ipb = m_fs->e2fs_bsize / EXT2_DINODE_SIZE;
        !           556:        m_fs->e2fs_itpg = m_fs->e2fs.e2fs_ipg/m_fs->e2fs_ipb;
        !           557:
        !           558:        m_fs->e2fs_gd = malloc(m_fs->e2fs_ngdb * m_fs->e2fs_bsize,
        !           559:                M_UFSMNT, M_WAITOK);
        !           560:        for (i=0; i < m_fs->e2fs_ngdb; i++) {
        !           561:                error = bread(devvp ,
        !           562:                    fsbtodb(m_fs, ((m_fs->e2fs_bsize>1024)? 0 : 1) + i + 1),
        !           563:                    m_fs->e2fs_bsize, NOCRED, &bp);
        !           564:                if (error) {
        !           565:                        free(m_fs->e2fs_gd, M_UFSMNT);
        !           566:                        goto out;
        !           567:                }
        !           568:                e2fs_cgload((struct ext2_gd*)bp->b_data,
        !           569:                    &m_fs->e2fs_gd[
        !           570:                        i * m_fs->e2fs_bsize / sizeof(struct ext2_gd)],
        !           571:                    m_fs->e2fs_bsize);
        !           572:                brelse(bp);
        !           573:                bp = NULL;
        !           574:        }
        !           575:
        !           576:        mp->mnt_data = (qaddr_t)ump;
        !           577:        mp->mnt_stat.f_fsid.val[0] = (long)dev;
        !           578:        mp->mnt_stat.f_fsid.val[1] = mp->mnt_vfc->vfc_typenum;
        !           579:        mp->mnt_maxsymlinklen = EXT2_MAXSYMLINKLEN;
        !           580:        mp->mnt_flag |= MNT_LOCAL;
        !           581:        ump->um_mountp = mp;
        !           582:        ump->um_dev = dev;
        !           583:        ump->um_devvp = devvp;
        !           584:        ump->um_nindir = NINDIR(m_fs);
        !           585:        ump->um_bptrtodb = m_fs->e2fs_fsbtodb;
        !           586:        ump->um_seqinc = 1; /* no frags */
        !           587:        devvp->v_specmountpoint = mp;
        !           588:        return (0);
        !           589: out:
        !           590:        if (bp)
        !           591:                brelse(bp);
        !           592:        vn_lock(devvp, LK_EXCLUSIVE | LK_RETRY, p);
        !           593:        (void)VOP_CLOSE(devvp, ronly ? FREAD : FREAD|FWRITE, cred, p);
        !           594:        VOP_UNLOCK(devvp, 0, p);
        !           595:        if (ump) {
        !           596:                free(ump->um_e2fs, M_UFSMNT);
        !           597:                free(ump, M_UFSMNT);
        !           598:                mp->mnt_data = (qaddr_t)0;
        !           599:        }
        !           600:        return (error);
        !           601: }
        !           602:
        !           603: /*
        !           604:  * unmount system call
        !           605:  */
        !           606: int
        !           607: ext2fs_unmount(struct mount *mp, int mntflags, struct proc *p)
        !           608: {
        !           609:        struct ufsmount *ump;
        !           610:        struct m_ext2fs *fs;
        !           611:        int error, flags;
        !           612:
        !           613:        flags = 0;
        !           614:        if (mntflags & MNT_FORCE)
        !           615:                flags |= FORCECLOSE;
        !           616:        if ((error = ext2fs_flushfiles(mp, flags, p)) != 0)
        !           617:                return (error);
        !           618:        ump = VFSTOUFS(mp);
        !           619:        fs = ump->um_e2fs;
        !           620:        if (fs->e2fs_ronly == 0 &&
        !           621:                ext2fs_cgupdate(ump, MNT_WAIT) == 0 &&
        !           622:                (fs->e2fs.e2fs_state & E2FS_ERRORS) == 0) {
        !           623:                fs->e2fs.e2fs_state = E2FS_ISCLEAN;
        !           624:                (void) ext2fs_sbupdate(ump, MNT_WAIT);
        !           625:        }
        !           626:
        !           627:        if (ump->um_devvp->v_type != VBAD)
        !           628:                ump->um_devvp->v_specmountpoint = NULL;
        !           629:        vn_lock(ump->um_devvp, LK_EXCLUSIVE | LK_RETRY, p);
        !           630:        error = VOP_CLOSE(ump->um_devvp, fs->e2fs_ronly ? FREAD : FREAD|FWRITE,
        !           631:                NOCRED, p);
        !           632:        vput(ump->um_devvp);
        !           633:        free(fs->e2fs_gd, M_UFSMNT);
        !           634:        free(fs, M_UFSMNT);
        !           635:        free(ump, M_UFSMNT);
        !           636:        mp->mnt_data = (qaddr_t)0;
        !           637:        mp->mnt_flag &= ~MNT_LOCAL;
        !           638:        return (error);
        !           639: }
        !           640:
        !           641: /*
        !           642:  * Flush out all the files in a filesystem.
        !           643:  */
        !           644: int
        !           645: ext2fs_flushfiles(struct mount *mp, int flags, struct proc *p)
        !           646: {
        !           647:        struct ufsmount *ump;
        !           648:        int error;
        !           649:
        !           650:        ump = VFSTOUFS(mp);
        !           651:        /*
        !           652:         * Flush all the files.
        !           653:         */
        !           654:        if ((error = vflush(mp, NULL, flags)) != 0)
        !           655:                return (error);
        !           656:        /*
        !           657:         * Flush filesystem metadata.
        !           658:         */
        !           659:        vn_lock(ump->um_devvp, LK_EXCLUSIVE | LK_RETRY, p);
        !           660:        error = VOP_FSYNC(ump->um_devvp, p->p_ucred, MNT_WAIT, p);
        !           661:        VOP_UNLOCK(ump->um_devvp, 0, p);
        !           662:        return (error);
        !           663: }
        !           664:
        !           665: /*
        !           666:  * Get file system statistics.
        !           667:  */
        !           668: int
        !           669: ext2fs_statfs(struct mount *mp, struct statfs *sbp, struct proc *p)
        !           670: {
        !           671:        struct ufsmount *ump;
        !           672:        struct m_ext2fs *fs;
        !           673:        u_int32_t overhead, overhead_per_group;
        !           674:        int i, ngroups;
        !           675:
        !           676:        ump = VFSTOUFS(mp);
        !           677:        fs = ump->um_e2fs;
        !           678:        if (fs->e2fs.e2fs_magic != E2FS_MAGIC)
        !           679:                panic("ext2fs_statfs");
        !           680:
        !           681:        /*
        !           682:         * Compute the overhead (FS structures)
        !           683:         */
        !           684:        overhead_per_group = 1 /* block bitmap */ +
        !           685:                                 1 /* inode bitmap */ +
        !           686:                                 fs->e2fs_itpg;
        !           687:        overhead = fs->e2fs.e2fs_first_dblock +
        !           688:                   fs->e2fs_ncg * overhead_per_group;
        !           689:        if (fs->e2fs.e2fs_rev > E2FS_REV0 &&
        !           690:            fs->e2fs.e2fs_features_rocompat & EXT2F_ROCOMPAT_SPARSESUPER) {
        !           691:                for (i = 0, ngroups = 0; i < fs->e2fs_ncg; i++) {
        !           692:                        if (cg_has_sb(i))
        !           693:                                ngroups++;
        !           694:                }
        !           695:        } else {
        !           696:                ngroups = fs->e2fs_ncg;
        !           697:        }
        !           698:        overhead += ngroups * (1 + fs->e2fs_ngdb);
        !           699:
        !           700:        sbp->f_bsize = fs->e2fs_bsize;
        !           701:        sbp->f_iosize = fs->e2fs_bsize;
        !           702:        sbp->f_blocks = fs->e2fs.e2fs_bcount - overhead;
        !           703:        sbp->f_bfree = fs->e2fs.e2fs_fbcount;
        !           704:        sbp->f_bavail = sbp->f_bfree - fs->e2fs.e2fs_rbcount;
        !           705:        sbp->f_files =  fs->e2fs.e2fs_icount;
        !           706:        sbp->f_ffree = fs->e2fs.e2fs_ficount;
        !           707:        if (sbp != &mp->mnt_stat) {
        !           708:                bcopy(mp->mnt_stat.f_mntonname, sbp->f_mntonname, MNAMELEN);
        !           709:                bcopy(mp->mnt_stat.f_mntfromname, sbp->f_mntfromname, MNAMELEN);
        !           710:        }
        !           711:        strncpy(sbp->f_fstypename, mp->mnt_vfc->vfc_name, MFSNAMELEN);
        !           712:        return (0);
        !           713: }
        !           714:
        !           715: int ext2fs_sync_vnode(struct vnode *vp, void *);
        !           716:
        !           717: struct ext2fs_sync_args {
        !           718:        int allerror;
        !           719:        int waitfor;
        !           720:        struct proc *p;
        !           721:        struct ucred *cred;
        !           722: };
        !           723:
        !           724: int
        !           725: ext2fs_sync_vnode(struct vnode *vp, void *args)
        !           726: {
        !           727:        struct ext2fs_sync_args *esa = args;
        !           728:        struct inode *ip;
        !           729:        int error;
        !           730:
        !           731:        ip = VTOI(vp);
        !           732:        if (vp->v_type == VNON ||
        !           733:            ((ip->i_flag & (IN_ACCESS | IN_CHANGE | IN_MODIFIED | IN_UPDATE)) == 0 &&
        !           734:                LIST_EMPTY(&vp->v_dirtyblkhd)) ||
        !           735:            esa->waitfor == MNT_LAZY) {
        !           736:                return (0);
        !           737:        }
        !           738:
        !           739:        if (vget(vp, LK_EXCLUSIVE | LK_NOWAIT, esa->p))
        !           740:                return (0);
        !           741:
        !           742:        if ((error = VOP_FSYNC(vp, esa->cred, esa->waitfor, esa->p)) != 0)
        !           743:                esa->allerror = error;
        !           744:        vput(vp);
        !           745:        return (0);
        !           746: }
        !           747: /*
        !           748:  * Go through the disk queues to initiate sandbagged IO;
        !           749:  * go through the inodes to write those that have been modified;
        !           750:  * initiate the writing of the super block if it has been modified.
        !           751:  *
        !           752:  * Should always be called with the mount point locked.
        !           753:  */
        !           754: int
        !           755: ext2fs_sync(struct mount *mp, int waitfor, struct ucred *cred, struct proc *p)
        !           756: {
        !           757:        struct ufsmount *ump = VFSTOUFS(mp);
        !           758:        struct m_ext2fs *fs;
        !           759:        int error, allerror = 0;
        !           760:        struct ext2fs_sync_args esa;
        !           761:
        !           762:        fs = ump->um_e2fs;
        !           763:        if (fs->e2fs_ronly != 0) {              /* XXX */
        !           764:                printf("fs = %s\n", fs->e2fs_fsmnt);
        !           765:                panic("update: rofs mod");
        !           766:        }
        !           767:
        !           768:        /*
        !           769:         * Write back each (modified) inode.
        !           770:         */
        !           771:        esa.p = p;
        !           772:        esa.cred = cred;
        !           773:        esa.allerror = 0;
        !           774:        esa.waitfor = waitfor;
        !           775:
        !           776:        vfs_mount_foreach_vnode(mp, ext2fs_sync_vnode, &esa);
        !           777:        if (esa.allerror != 0)
        !           778:                allerror = esa.allerror;
        !           779:
        !           780:        /*
        !           781:         * Force stale file system control information to be flushed.
        !           782:         */
        !           783:        if (waitfor != MNT_LAZY) {
        !           784:                vn_lock(ump->um_devvp, LK_EXCLUSIVE | LK_RETRY, p);
        !           785:                if ((error = VOP_FSYNC(ump->um_devvp, cred, waitfor, p)) != 0)
        !           786:                        allerror = error;
        !           787:                VOP_UNLOCK(ump->um_devvp, 0, p);
        !           788:        }
        !           789:        /*
        !           790:         * Write back modified superblock.
        !           791:         */
        !           792:        if (fs->e2fs_fmod != 0) {
        !           793:                fs->e2fs_fmod = 0;
        !           794:                fs->e2fs.e2fs_wtime = time_second;
        !           795:                if ((error = ext2fs_cgupdate(ump, waitfor)))
        !           796:                        allerror = error;
        !           797:        }
        !           798:        return (allerror);
        !           799: }
        !           800:
        !           801: /*
        !           802:  * Look up a EXT2FS dinode number to find its incore vnode, otherwise read it
        !           803:  * in from disk.  If it is in core, wait for the lock bit to clear, then
        !           804:  * return the inode locked.  Detection and handling of mount points must be
        !           805:  * done by the calling routine.
        !           806:  */
        !           807: int
        !           808: ext2fs_vget(struct mount *mp, ino_t ino, struct vnode **vpp)
        !           809: {
        !           810:        struct m_ext2fs *fs;
        !           811:        struct inode *ip;
        !           812:        struct ext2fs_dinode *dp;
        !           813:        struct ufsmount *ump;
        !           814:        struct buf *bp;
        !           815:        struct vnode *vp;
        !           816:        dev_t dev;
        !           817:        int error;
        !           818:
        !           819:        ump = VFSTOUFS(mp);
        !           820:        dev = ump->um_dev;
        !           821:
        !           822:  retry:
        !           823:        if ((*vpp = ufs_ihashget(dev, ino)) != NULL)
        !           824:                return (0);
        !           825:
        !           826:        /* Allocate a new vnode/inode. */
        !           827:        if ((error = getnewvnode(VT_EXT2FS, mp, ext2fs_vnodeop_p, &vp)) != 0) {
        !           828:                *vpp = NULL;
        !           829:                return (error);
        !           830:        }
        !           831:
        !           832:        ip = pool_get(&ext2fs_inode_pool, PR_WAITOK);
        !           833:        memset(ip, 0, sizeof(struct inode));
        !           834:        lockinit(&ip->i_lock, PINOD, "inode", 0, 0);
        !           835:        vp->v_data = ip;
        !           836:        ip->i_vnode = vp;
        !           837:        ip->i_ump = ump;
        !           838:        ip->i_e2fs = fs = ump->um_e2fs;
        !           839:        ip->i_dev = dev;
        !           840:        ip->i_number = ino;
        !           841:        ip->i_e2fs_last_lblk = 0;
        !           842:        ip->i_e2fs_last_blk = 0;
        !           843:
        !           844:        /*
        !           845:         * Put it onto its hash chain and lock it so that other requests for
        !           846:         * this inode will block if they arrive while we are sleeping waiting
        !           847:         * for old data structures to be purged or for the contents of the
        !           848:         * disk portion of this inode to be read.
        !           849:         */
        !           850:        error = ufs_ihashins(ip);
        !           851:
        !           852:        if (error) {
        !           853:                vrele(vp);
        !           854:
        !           855:                if (error == EEXIST)
        !           856:                        goto retry;
        !           857:
        !           858:                return (error);
        !           859:        }
        !           860:
        !           861:        /* Read in the disk contents for the inode, copy into the inode. */
        !           862:        error = bread(ump->um_devvp, fsbtodb(fs, ino_to_fsba(fs, ino)),
        !           863:                          (int)fs->e2fs_bsize, NOCRED, &bp);
        !           864:        if (error) {
        !           865:                /*
        !           866:                 * The inode does not contain anything useful, so it would
        !           867:                 * be misleading to leave it on its hash chain. With mode
        !           868:                 * still zero, it will be unlinked and returned to the free
        !           869:                 * list by vput().
        !           870:                 */
        !           871:                vput(vp);
        !           872:                brelse(bp);
        !           873:                *vpp = NULL;
        !           874:                return (error);
        !           875:        }
        !           876:
        !           877:        dp = (struct ext2fs_dinode *) bp->b_data + ino_to_fsbo(fs, ino);
        !           878:        ip->i_e2din = pool_get(&ext2fs_dinode_pool, PR_WAITOK);
        !           879:        e2fs_iload(dp, ip->i_e2din);
        !           880:        brelse(bp);
        !           881:
        !           882:        ip->i_effnlink = ip->i_e2fs_nlink;
        !           883:
        !           884:        /*
        !           885:         * The fields for storing the UID and GID of an ext2fs inode are
        !           886:         * limited to 16 bits. To overcome this limitation, Linux decided to
        !           887:         * scatter the highest bits of these values into a previously reserved
        !           888:         * area on the disk inode. We deal with this situation by having two
        !           889:         * 32-bit fields *out* of the disk inode to hold the complete values.
        !           890:         * Now that we are reading in the inode, compute these fields.
        !           891:         */
        !           892:        ip->i_e2fs_uid = ip->i_e2fs_uid_low | (ip->i_e2fs_uid_high << 16);
        !           893:        ip->i_e2fs_gid = ip->i_e2fs_gid_low | (ip->i_e2fs_gid_high << 16);
        !           894:
        !           895:        /* If the inode was deleted, reset all fields */
        !           896:        if (ip->i_e2fs_dtime != 0) {
        !           897:                ip->i_e2fs_mode = ip->i_e2fs_nblock = 0;
        !           898:                (void)ext2fs_setsize(ip, 0);
        !           899:        }
        !           900:
        !           901:        /*
        !           902:         * Initialize the vnode from the inode, check for aliases.
        !           903:         * Note that the underlying vnode may have changed.
        !           904:         */
        !           905:        error = ext2fs_vinit(mp, ext2fs_specop_p, EXT2FS_FIFOOPS, &vp);
        !           906:        if (error) {
        !           907:                vput(vp);
        !           908:                *vpp = NULL;
        !           909:                return (error);
        !           910:        }
        !           911:
        !           912:        /*
        !           913:         * Finish inode initialization now that aliasing has been resolved.
        !           914:         */
        !           915:        VREF(ip->i_devvp);
        !           916:        /*
        !           917:         * Set up a generation number for this inode if it does not
        !           918:         * already have one. This should only happen on old filesystems.
        !           919:         */
        !           920:        if (ip->i_e2fs_gen == 0) {
        !           921:                if (++ext2gennumber < (u_long)time_second)
        !           922:                        ext2gennumber = time_second;
        !           923:                ip->i_e2fs_gen = ext2gennumber;
        !           924:                if ((vp->v_mount->mnt_flag & MNT_RDONLY) == 0)
        !           925:                        ip->i_flag |= IN_MODIFIED;
        !           926:        }
        !           927:
        !           928:        *vpp = vp;
        !           929:        return (0);
        !           930: }
        !           931:
        !           932: /*
        !           933:  * File handle to vnode
        !           934:  *
        !           935:  * Have to be really careful about stale file handles:
        !           936:  * - check that the inode number is valid
        !           937:  * - call ext2fs_vget() to get the locked inode
        !           938:  * - check for an unallocated inode (i_mode == 0)
        !           939:  * - check that the given client host has export rights and return
        !           940:  *   those rights via. exflagsp and credanonp
        !           941:  */
        !           942: int
        !           943: ext2fs_fhtovp(struct mount *mp, struct fid *fhp, struct vnode **vpp)
        !           944: {
        !           945:        struct inode *ip;
        !           946:        struct vnode *nvp;
        !           947:        int error;
        !           948:        struct ufid *ufhp;
        !           949:        struct m_ext2fs *fs;
        !           950:
        !           951:        ufhp = (struct ufid *)fhp;
        !           952:        fs = VFSTOUFS(mp)->um_e2fs;
        !           953:        if ((ufhp->ufid_ino < EXT2_FIRSTINO && ufhp->ufid_ino != EXT2_ROOTINO) ||
        !           954:                ufhp->ufid_ino > fs->e2fs_ncg * fs->e2fs.e2fs_ipg)
        !           955:                return (ESTALE);
        !           956:
        !           957:        if ((error = VFS_VGET(mp, ufhp->ufid_ino, &nvp)) != 0) {
        !           958:                *vpp = NULLVP;
        !           959:                return (error);
        !           960:        }
        !           961:        ip = VTOI(nvp);
        !           962:        if (ip->i_e2fs_mode == 0 || ip->i_e2fs_dtime != 0 ||
        !           963:                ip->i_e2fs_gen != ufhp->ufid_gen) {
        !           964:                vput(nvp);
        !           965:                *vpp = NULLVP;
        !           966:                return (ESTALE);
        !           967:        }
        !           968:        *vpp = nvp;
        !           969:        return (0);
        !           970: }
        !           971:
        !           972: /*
        !           973:  * Vnode pointer to File handle
        !           974:  */
        !           975: /* ARGSUSED */
        !           976: int
        !           977: ext2fs_vptofh(struct vnode *vp, struct fid *fhp)
        !           978: {
        !           979:        struct inode *ip;
        !           980:        struct ufid *ufhp;
        !           981:
        !           982:        ip = VTOI(vp);
        !           983:        ufhp = (struct ufid *)fhp;
        !           984:        ufhp->ufid_len = sizeof(struct ufid);
        !           985:        ufhp->ufid_ino = ip->i_number;
        !           986:        ufhp->ufid_gen = ip->i_e2fs_gen;
        !           987:        return (0);
        !           988: }
        !           989:
        !           990: /*
        !           991:  * no sysctl for ext2fs
        !           992:  */
        !           993:
        !           994: int
        !           995: ext2fs_sysctl(int *name, u_int namelen, void *oldp, size_t *oldlenp,
        !           996:     void *newp, size_t newlen, struct proc *p)
        !           997: {
        !           998:        return (EOPNOTSUPP);
        !           999: }
        !          1000:
        !          1001: /*
        !          1002:  * Write a superblock and associated information back to disk.
        !          1003:  */
        !          1004: int
        !          1005: ext2fs_sbupdate(struct ufsmount *mp, int waitfor)
        !          1006: {
        !          1007:        struct m_ext2fs *fs = mp->um_e2fs;
        !          1008:        struct buf *bp;
        !          1009:        int error = 0;
        !          1010:
        !          1011:        bp = getblk(mp->um_devvp, SBLOCK, SBSIZE, 0, 0);
        !          1012:        e2fs_sbsave(&fs->e2fs, (struct ext2fs *) bp->b_data);
        !          1013:        if (waitfor == MNT_WAIT)
        !          1014:                error = bwrite(bp);
        !          1015:        else
        !          1016:                bawrite(bp);
        !          1017:        return (error);
        !          1018: }
        !          1019:
        !          1020: int
        !          1021: ext2fs_cgupdate(struct ufsmount *mp, int waitfor)
        !          1022: {
        !          1023:        struct m_ext2fs *fs = mp->um_e2fs;
        !          1024:        struct buf *bp;
        !          1025:        int i, error = 0, allerror = 0;
        !          1026:
        !          1027:        allerror = ext2fs_sbupdate(mp, waitfor);
        !          1028:        for (i = 0; i < fs->e2fs_ngdb; i++) {
        !          1029:                bp = getblk(mp->um_devvp, fsbtodb(fs, ((fs->e2fs_bsize>1024)?0:1)+i+1),
        !          1030:                    fs->e2fs_bsize, 0, 0);
        !          1031:                e2fs_cgsave(&fs->e2fs_gd[i* fs->e2fs_bsize / sizeof(struct ext2_gd)], (struct ext2_gd*)bp->b_data, fs->e2fs_bsize);
        !          1032:                if (waitfor == MNT_WAIT)
        !          1033:                        error = bwrite(bp);
        !          1034:                else
        !          1035:                        bawrite(bp);
        !          1036:        }
        !          1037:
        !          1038:        if (!allerror && error)
        !          1039:                allerror = error;
        !          1040:        return (allerror);
        !          1041: }
        !          1042:
        !          1043: static int
        !          1044: ext2fs_checksb(struct ext2fs *fs, int ronly)
        !          1045: {
        !          1046:        if (fs2h16(fs->e2fs_magic) != E2FS_MAGIC) {
        !          1047:                return (EIO);           /* XXX needs translation */
        !          1048:        }
        !          1049:        if (fs2h32(fs->e2fs_rev) > E2FS_REV1) {
        !          1050: #ifdef DIAGNOSTIC
        !          1051:                printf("Ext2 fs: unsupported revision number: %x\n",
        !          1052:                    fs2h32(fs->e2fs_rev));
        !          1053: #endif
        !          1054:                return (EIO);           /* XXX needs translation */
        !          1055:        }
        !          1056:        if (fs2h32(fs->e2fs_log_bsize) > 2) { /* block size = 1024|2048|4096 */
        !          1057: #ifdef DIAGNOSTIC
        !          1058:                printf("Ext2 fs: bad block size: %d (expected <=2 for ext2 fs)\n",
        !          1059:                    fs2h32(fs->e2fs_log_bsize));
        !          1060: #endif
        !          1061:                return (EIO);      /* XXX needs translation */
        !          1062:        }
        !          1063:        if (fs2h32(fs->e2fs_rev) > E2FS_REV0) {
        !          1064:                if (fs2h32(fs->e2fs_first_ino) != EXT2_FIRSTINO ||
        !          1065:                    fs2h16(fs->e2fs_inode_size) != EXT2_DINODE_SIZE) {
        !          1066:                        printf("Ext2 fs: unsupported inode size\n");
        !          1067:                        return (EINVAL);      /* XXX needs translation */
        !          1068:                }
        !          1069:                if (fs2h32(fs->e2fs_features_incompat) &
        !          1070:                    ~EXT2F_INCOMPAT_SUPP) {
        !          1071:                        printf("Ext2 fs: unsupported optional feature\n");
        !          1072:                        return (EINVAL);      /* XXX needs translation */
        !          1073:                }
        !          1074:                if (!ronly && fs2h32(fs->e2fs_features_rocompat) &
        !          1075:                    ~EXT2F_ROCOMPAT_SUPP) {
        !          1076:                        return (EROFS);      /* XXX needs translation */
        !          1077:                }
        !          1078:        }
        !          1079:        return (0);
        !          1080: }

CVSweb