Annotation of XNU/bsd/ufs/lfs/lfs_alloc.c, revision 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: lfs_alloc.c,v 1.2 1994/06/29 06:46:47 cgd Exp $        */
        !            23: 
        !            24: /*
        !            25:  * Copyright (c) 1991, 1993
        !            26:  *     The Regents of the University of California.  All rights reserved.
        !            27:  *
        !            28:  * Redistribution and use in source and binary forms, with or without
        !            29:  * modification, are permitted provided that the following conditions
        !            30:  * are met:
        !            31:  * 1. Redistributions of source code must retain the above copyright
        !            32:  *    notice, this list of conditions and the following disclaimer.
        !            33:  * 2. Redistributions in binary form must reproduce the above copyright
        !            34:  *    notice, this list of conditions and the following disclaimer in the
        !            35:  *    documentation and/or other materials provided with the distribution.
        !            36:  * 3. All advertising materials mentioning features or use of this software
        !            37:  *    must display the following acknowledgement:
        !            38:  *     This product includes software developed by the University of
        !            39:  *     California, Berkeley and its contributors.
        !            40:  * 4. Neither the name of the University nor the names of its contributors
        !            41:  *    may be used to endorse or promote products derived from this software
        !            42:  *    without specific prior written permission.
        !            43:  *
        !            44:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
        !            45:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
        !            46:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
        !            47:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
        !            48:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
        !            49:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
        !            50:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
        !            51:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
        !            52:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
        !            53:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
        !            54:  * SUCH DAMAGE.
        !            55:  *
        !            56:  *     @(#)lfs_alloc.c 8.4 (Berkeley) 1/4/94
        !            57:  */
        !            58: 
        !            59: #include <sys/param.h>
        !            60: #include <sys/kernel.h>
        !            61: #include <sys/buf.h>
        !            62: #include <sys/vnode.h>
        !            63: #include <sys/syslog.h>
        !            64: #include <sys/mount.h>
        !            65: #include <sys/malloc.h>
        !            66: 
        !            67: #include <vm/vm.h>
        !            68: 
        !            69: #include <ufs/ufs/quota.h>
        !            70: #include <ufs/ufs/inode.h>
        !            71: #include <ufs/ufs/ufsmount.h>
        !            72: 
        !            73: #include <ufs/lfs/lfs.h>
        !            74: #include <ufs/lfs/lfs_extern.h>
        !            75: 
        !            76: extern u_long nextgennumber;
        !            77: 
        !            78: /* Allocate a new inode. */
        !            79: /* ARGSUSED */
        !            80: int
        !            81: lfs_valloc(ap)
        !            82:        struct vop_valloc_args /* {
        !            83:                struct vnode *a_pvp;
        !            84:                int a_mode;
        !            85:                struct ucred *a_cred;
        !            86:                struct vnode **a_vpp;
        !            87:        } */ *ap;
        !            88: {
        !            89:        struct lfs *fs;
        !            90:        struct buf *bp;
        !            91:        struct ifile *ifp;
        !            92:        struct inode *ip;
        !            93:        struct vnode *vp;
        !            94:        daddr_t blkno;
        !            95:        ino_t new_ino;
        !            96:        u_long i, max;
        !            97:        int error;
        !            98: 
        !            99:        /* Get the head of the freelist. */
        !           100:        fs = VTOI(ap->a_pvp)->i_lfs;
        !           101:        new_ino = fs->lfs_free;
        !           102: #ifdef ALLOCPRINT
        !           103:        printf("lfs_ialloc: allocate inode %d\n", new_ino);
        !           104: #endif
        !           105: 
        !           106:        /*
        !           107:         * Remove the inode from the free list and write the new start
        !           108:         * of the free list into the superblock.
        !           109:         */
        !           110:        LFS_IENTRY(ifp, fs, new_ino, bp);
        !           111:        if (ifp->if_daddr != LFS_UNUSED_DADDR)
        !           112:                panic("lfs_ialloc: inuse inode on the free list");
        !           113:        fs->lfs_free = ifp->if_nextfree;
        !           114:        brelse(bp);
        !           115: 
        !           116:        /* Extend IFILE so that the next lfs_valloc will succeed. */
        !           117:        if (fs->lfs_free == LFS_UNUSED_INUM) {
        !           118:                vp = fs->lfs_ivnode;
        !           119:                ip = VTOI(vp);
        !           120:                blkno = lblkno(fs, ip->i_size);
        !           121:                lfs_balloc(vp, fs->lfs_bsize, blkno, &bp);
        !           122:                ip->i_size += fs->lfs_bsize;
        !           123:                vnode_pager_setsize(vp, (u_long)ip->i_size);
        !           124:                vnode_pager_uncache(vp);
        !           125: 
        !           126:                i = (blkno - fs->lfs_segtabsz - fs->lfs_cleansz) *
        !           127:                    fs->lfs_ifpb;
        !           128:                fs->lfs_free = i;
        !           129:                max = i + fs->lfs_ifpb;
        !           130:                for (ifp = (struct ifile *)bp->b_data; i < max; ++ifp) {
        !           131:                        ifp->if_version = 1;
        !           132:                        ifp->if_daddr = LFS_UNUSED_DADDR;
        !           133:                        ifp->if_nextfree = ++i;
        !           134:                }
        !           135:                ifp--;
        !           136:                ifp->if_nextfree = LFS_UNUSED_INUM;
        !           137:                if (error = VOP_BWRITE(bp))
        !           138:                        return (error);
        !           139:        }
        !           140: 
        !           141:        /* Create a vnode to associate with the inode. */
        !           142:        if (error = lfs_vcreate(ap->a_pvp->v_mount, new_ino, &vp))
        !           143:                return (error);
        !           144: 
        !           145: 
        !           146:        ip = VTOI(vp);
        !           147:        /* Zero out the direct and indirect block addresses. */
        !           148:        bzero(&ip->i_din, sizeof(struct dinode));
        !           149:        ip->i_din.di_inumber = new_ino;
        !           150: 
        !           151:        /* Set a new generation number for this inode. */
        !           152:        if (++nextgennumber < (u_long)time.tv_sec)
        !           153:                nextgennumber = time.tv_sec;
        !           154:        ip->i_gen = nextgennumber;
        !           155: 
        !           156:        /* Insert into the inode hash table. */
        !           157:        ufs_ihashins(ip);
        !           158: 
        !           159:        if (error = ufs_vinit(vp->v_mount, lfs_specop_p, LFS_FIFOOPS, &vp)) {
        !           160:                vput(vp);
        !           161:                *ap->a_vpp = NULL;
        !           162:                return (error);
        !           163:        }
        !           164: 
        !           165:        *ap->a_vpp = vp;
        !           166:        vp->v_flag |= VDIROP;
        !           167:        VREF(ip->i_devvp);
        !           168: 
        !           169:        /* Set superblock modified bit and increment file count. */
        !           170:        fs->lfs_fmod = 1;
        !           171:        ++fs->lfs_nfiles;
        !           172:        return (0);
        !           173: }
        !           174: 
        !           175: /* Create a new vnode/inode pair and initialize what fields we can. */
        !           176: int
        !           177: lfs_vcreate(mp, ino, vpp)
        !           178:        struct mount *mp;
        !           179:        ino_t ino;
        !           180:        struct vnode **vpp;
        !           181: {
        !           182:        extern int (**lfs_vnodeop_p)();
        !           183:        struct inode *ip;
        !           184:        struct ufsmount *ump;
        !           185:        int error, i;
        !           186: 
        !           187:        MALLOC(ip, struct inode *, sizeof(struct inode), M_LFSNODE, M_WAITOK);
        !           188:        /* Create the vnode. */
        !           189:        if (error = getnewvnode(VT_LFS, mp, lfs_vnodeop_p, vpp)) {
        !           190:                FREE(ip, M_LFSNODE);
        !           191:                *vpp = NULL;
        !           192:                return (error);
        !           193:        }
        !           194: 
        !           195:        /* Get a pointer to the private mount structure. */
        !           196:        ump = VFSTOUFS(mp);
        !           197: 
        !           198:        /* Initialize the inode. */
        !           199:        (*vpp)->v_data = ip;
        !           200:        ip->i_vnode = *vpp;
        !           201:        ip->i_devvp = ump->um_devvp;
        !           202:        ip->i_flag = IN_MODIFIED;
        !           203:        ip->i_dev = ump->um_dev;
        !           204:        ip->i_number = ip->i_din.di_inumber = ino;
        !           205: ip->i_din.di_spare[0] = 0xdeadbeef;
        !           206: ip->i_din.di_spare[1] = 0xdeadbeef;
        !           207:        ip->i_lfs = ump->um_lfs;
        !           208: #ifdef QUOTA
        !           209:        for (i = 0; i < MAXQUOTAS; i++)
        !           210:                ip->i_dquot[i] = NODQUOT;
        !           211: #endif
        !           212:        ip->i_lockf = 0;
        !           213:        ip->i_diroff = 0;
        !           214:        ip->i_mode = 0;
        !           215:        ip->i_size = 0;
        !           216:        ip->i_blocks = 0;
        !           217:        ++ump->um_lfs->lfs_uinodes;
        !           218:        return (0);
        !           219: }
        !           220: 
        !           221: /* Free an inode. */
        !           222: /* ARGUSED */
        !           223: int
        !           224: lfs_vfree(ap)
        !           225:        struct vop_vfree_args /* {
        !           226:                struct vnode *a_pvp;
        !           227:                ino_t a_ino;
        !           228:                int a_mode;
        !           229:        } */ *ap;
        !           230: {
        !           231:        SEGUSE *sup;
        !           232:        struct buf *bp;
        !           233:        struct ifile *ifp;
        !           234:        struct inode *ip;
        !           235:        struct lfs *fs;
        !           236:        daddr_t old_iaddr;
        !           237:        ino_t ino;
        !           238: 
        !           239:        /* Get the inode number and file system. */
        !           240:        ip = VTOI(ap->a_pvp);
        !           241:        fs = ip->i_lfs;
        !           242:        ino = ip->i_number;
        !           243:        if (ip->i_flag & IN_MODIFIED) {
        !           244:                --fs->lfs_uinodes;
        !           245:                ip->i_flag &=
        !           246:                    ~(IN_ACCESS | IN_CHANGE | IN_MODIFIED | IN_UPDATE);
        !           247:        }
        !           248:        /*
        !           249:         * Set the ifile's inode entry to unused, increment its version number
        !           250:         * and link it into the free chain.
        !           251:         */
        !           252:        LFS_IENTRY(ifp, fs, ino, bp);
        !           253:        old_iaddr = ifp->if_daddr;
        !           254:        ifp->if_daddr = LFS_UNUSED_DADDR;
        !           255:        ++ifp->if_version;
        !           256:        ifp->if_nextfree = fs->lfs_free;
        !           257:        fs->lfs_free = ino;
        !           258:        (void) VOP_BWRITE(bp);
        !           259: 
        !           260:        if (old_iaddr != LFS_UNUSED_DADDR) {
        !           261:                LFS_SEGENTRY(sup, fs, datosn(fs, old_iaddr), bp);
        !           262: #if DIAGNOSTIC
        !           263:                if (sup->su_nbytes < sizeof(struct dinode))
        !           264:                        panic("lfs_vfree: negative byte count (segment %d)\n",
        !           265:                            datosn(fs, old_iaddr));
        !           266: #endif
        !           267:                sup->su_nbytes -= sizeof(struct dinode);
        !           268:                (void) VOP_BWRITE(bp);
        !           269:        }
        !           270: 
        !           271:        /* Set superblock modified bit and decrement file count. */
        !           272:        fs->lfs_fmod = 1;
        !           273:        --fs->lfs_nfiles;
        !           274:        return (0);
        !           275: }

unix.superglobalmegacorp.com

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