Annotation of XNU/bsd/ufs/lfs/lfs_syscalls.c, revision 1.1.1.1

1.1       root        1: /*
                      2:  * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
                      3:  *
                      4:  * @APPLE_LICENSE_HEADER_START@
                      5:  * 
                      6:  * The contents of this file constitute Original Code as defined in and
                      7:  * are subject to the Apple Public Source License Version 1.1 (the
                      8:  * "License").  You may not use this file except in compliance with the
                      9:  * License.  Please obtain a copy of the License at
                     10:  * http://www.apple.com/publicsource and read it before using this file.
                     11:  * 
                     12:  * This Original Code and all software distributed under the License are
                     13:  * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
                     14:  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
                     15:  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
                     16:  * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the
                     17:  * License for the specific language governing rights and limitations
                     18:  * under the License.
                     19:  * 
                     20:  * @APPLE_LICENSE_HEADER_END@
                     21:  */
                     22: /*     $NetBSD: lfs_syscalls.c,v 1.8 1995/03/21 13:34:08 mycroft Exp $ */
                     23: 
                     24: /*-
                     25:  * Copyright (c) 1991, 1993, 1994
                     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_syscalls.c      8.6 (Berkeley) 6/16/94
                     57:  */
                     58: 
                     59: #include <sys/param.h>
                     60: #include <sys/systm.h>
                     61: #include <sys/proc.h>
                     62: #include <sys/buf.h>
                     63: #include <sys/mount.h>
                     64: #include <sys/vnode.h>
                     65: #include <sys/malloc.h>
                     66: #include <sys/kernel.h>
                     67: 
                     68: #include <sys/syscallargs.h>
                     69: 
                     70: #include <ufs/ufs/quota.h>
                     71: #include <ufs/ufs/inode.h>
                     72: #include <ufs/ufs/ufsmount.h>
                     73: #include <ufs/ufs/ufs_extern.h>
                     74: 
                     75: #include <ufs/lfs/lfs.h>
                     76: #include <ufs/lfs/lfs_extern.h>
                     77: #define BUMP_FIP(SP) \
                     78:        (SP)->fip = (FINFO *) (&(SP)->fip->fi_blocks[(SP)->fip->fi_nblocks])
                     79: 
                     80: #define INC_FINFO(SP) ++((SEGSUM *)((SP)->segsum))->ss_nfinfo
                     81: #define DEC_FINFO(SP) --((SEGSUM *)((SP)->segsum))->ss_nfinfo
                     82: 
                     83: /*
                     84:  * Before committing to add something to a segment summary, make sure there
                     85:  * is enough room.  S is the bytes added to the summary.
                     86:  */
                     87: #define        CHECK_SEG(s)                    \
                     88: if (sp->sum_bytes_left < (s)) {                \
                     89:        (void) lfs_writeseg(fs, sp);    \
                     90: }
                     91: struct buf *lfs_fakebuf __P((struct vnode *, int, size_t, caddr_t));
                     92: 
                     93: /*
                     94:  * lfs_markv:
                     95:  *
                     96:  * This will mark inodes and blocks dirty, so they are written into the log.
                     97:  * It will block until all the blocks have been written.  The segment create
                     98:  * time passed in the block_info and inode_info structures is used to decide
                     99:  * if the data is valid for each block (in case some process dirtied a block
                    100:  * or inode that is being cleaned between the determination that a block is
                    101:  * live and the lfs_markv call).
                    102:  *
                    103:  *  0 on success
                    104:  * -1/errno is return on error.
                    105:  */
                    106: int
                    107: lfs_markv(p, uap, retval)
                    108:        struct proc *p;
                    109:        struct lfs_markv_args /* {
                    110:                syscallarg(fsid_t *) fsidp;
                    111:                syscallarg(struct block_info *) blkiov;
                    112:                syscallarg(int) blkcnt;
                    113:        } */ *uap;
                    114:        register_t *retval;
                    115: {
                    116:        struct segment *sp;
                    117:        BLOCK_INFO *blkp;
                    118:        IFILE *ifp;
                    119:        struct buf *bp, **bpp;
                    120:        struct inode *ip;
                    121:        struct lfs *fs;
                    122:        struct mount *mntp;
                    123:        struct vnode *vp;
                    124:        fsid_t fsid;
                    125:        void *start;
                    126:        ino_t lastino;
                    127:        daddr_t b_daddr, v_daddr;
                    128:        u_long bsize;
                    129:        int cnt, error;
                    130: 
                    131:        if (error = suser(p->p_ucred, &p->p_acflag))
                    132:                return (error);
                    133: 
                    134:        if (error = copyin(SCARG(uap, fsidp), &fsid, sizeof(fsid_t)))
                    135:                return (error);
                    136:        if ((mntp = getvfs(&fsid)) == NULL)
                    137:                return (EINVAL);
                    138: 
                    139:        cnt = SCARG(uap, blkcnt);
                    140: //     start = malloc(cnt * sizeof(BLOCK_INFO), M_SEGMENT, M_WAITOK);
                    141:        MALLOC(start, void *, cnt * sizeof(BLOCK_INFO), M_SEGMENT, M_WAITOK);
                    142:        if (error = copyin(SCARG(uap, blkiov), start, cnt * sizeof(BLOCK_INFO)))
                    143:                goto err1;
                    144: 
                    145:        /* Mark blocks/inodes dirty.  */
                    146:        fs = VFSTOUFS(mntp)->um_lfs;
                    147:        bsize = fs->lfs_bsize;
                    148:        error = 0;
                    149: 
                    150:        lfs_seglock(fs, SEGM_SYNC | SEGM_CLEAN);
                    151:        sp = fs->lfs_sp;
                    152:        for (v_daddr = LFS_UNUSED_DADDR, lastino = LFS_UNUSED_INUM,
                    153:            blkp = start; cnt--; ++blkp) {
                    154:                /*
                    155:                 * Get the IFILE entry (only once) and see if the file still
                    156:                 * exists.
                    157:                 */
                    158:                if (lastino != blkp->bi_inode) {
                    159:                        if (lastino != LFS_UNUSED_INUM) {
                    160:                                /* Finish up last file */
                    161:                                if (sp->fip->fi_nblocks == 0) {
                    162:                                        DEC_FINFO(sp);
                    163:                                        sp->sum_bytes_left +=
                    164:                                            sizeof(FINFO) - sizeof(daddr_t);
                    165:                                } else {
                    166:                                        lfs_updatemeta(sp);
                    167:                                        BUMP_FIP(sp);
                    168:                                }
                    169: 
                    170:                                lfs_writeinode(fs, sp, ip);
                    171:                                lfs_vunref(vp);
                    172:                        }
                    173: 
                    174:                        /* Start a new file */
                    175:                        CHECK_SEG(sizeof(FINFO));
                    176:                        sp->sum_bytes_left -= sizeof(FINFO) - sizeof(daddr_t);
                    177:                        INC_FINFO(sp);
                    178:                        sp->start_lbp = &sp->fip->fi_blocks[0];
                    179:                        sp->vp = NULL;
                    180:                        sp->fip->fi_version = blkp->bi_version;
                    181:                        sp->fip->fi_nblocks = 0;
                    182:                        sp->fip->fi_ino = blkp->bi_inode;
                    183:                        lastino = blkp->bi_inode;
                    184:                        if (blkp->bi_inode == LFS_IFILE_INUM)
                    185:                                v_daddr = fs->lfs_idaddr;
                    186:                        else {
                    187:                                LFS_IENTRY(ifp, fs, blkp->bi_inode, bp);
                    188:                                v_daddr = ifp->if_daddr;
                    189:                                brelse(bp);
                    190:                        }
                    191:                        if (v_daddr == LFS_UNUSED_DADDR)
                    192:                                continue;
                    193: 
                    194:                        /* Get the vnode/inode. */
                    195:                        if (lfs_fastvget(mntp, blkp->bi_inode, v_daddr, &vp,
                    196:                            blkp->bi_lbn == LFS_UNUSED_LBN ? 
                    197:                            blkp->bi_bp : NULL)) {
                    198: #if DIAGNOSTIC
                    199:                                printf("lfs_markv: VFS_VGET failed (%d)\n",
                    200:                                    blkp->bi_inode);
                    201: #endif
                    202:                                lastino = LFS_UNUSED_INUM;
                    203:                                v_daddr = LFS_UNUSED_DADDR;
                    204:                                continue;
                    205:                        }
                    206:                        sp->vp = vp;
                    207:                        ip = VTOI(vp);
                    208:                } else if (v_daddr == LFS_UNUSED_DADDR)
                    209:                        continue;
                    210: 
                    211:                /* If this BLOCK_INFO didn't contain a block, keep going. */
                    212:                if (blkp->bi_lbn == LFS_UNUSED_LBN)
                    213:                        continue;
                    214:                if (VOP_BMAP(vp, blkp->bi_lbn, NULL, &b_daddr, NULL) ||
                    215:                    b_daddr != blkp->bi_daddr)
                    216:                        continue;
                    217:                /*
                    218:                 * If we got to here, then we are keeping the block.  If it
                    219:                 * is an indirect block, we want to actually put it in the
                    220:                 * buffer cache so that it can be updated in the finish_meta
                    221:                 * section.  If it's not, we need to allocate a fake buffer
                    222:                 * so that writeseg can perform the copyin and write the buffer.
                    223:                 */
                    224:                if (blkp->bi_lbn >= 0)  /* Data Block */
                    225:                        bp = lfs_fakebuf(vp, blkp->bi_lbn, bsize,
                    226:                            blkp->bi_bp);
                    227:                else {
                    228:                        bp = getblk(vp, blkp->bi_lbn, bsize, 0, 0);
                    229:                        if (!(bp->b_flags & (B_DELWRI | B_DONE | B_CACHE)) &&
                    230:                            (error = copyin(blkp->bi_bp, bp->b_data,
                    231:                            bsize)))
                    232:                                goto err2;
                    233:                        if (error = VOP_BWRITE(bp))
                    234:                                goto err2;
                    235:                }
                    236:                while (lfs_gatherblock(sp, bp, NULL));
                    237:        }
                    238:        if (sp->vp) {
                    239:                if (sp->fip->fi_nblocks == 0) {
                    240:                        DEC_FINFO(sp);
                    241:                        sp->sum_bytes_left +=
                    242:                            sizeof(FINFO) - sizeof(daddr_t);
                    243:                } else
                    244:                        lfs_updatemeta(sp);
                    245: 
                    246:                lfs_writeinode(fs, sp, ip);
                    247:                lfs_vunref(vp);
                    248:        }
                    249:        (void) lfs_writeseg(fs, sp);
                    250:        lfs_segunlock(fs);
                    251:        free(start, M_SEGMENT);
                    252:        return (error);
                    253: 
                    254: /*
                    255:  * XXX
                    256:  * If we come in to error 2, we might have indirect blocks that were
                    257:  * updated and now have bad block pointers.  I don't know what to do
                    258:  * about this.
                    259:  */
                    260: 
                    261: err2:  lfs_vunref(vp);
                    262:        /* Free up fakebuffers */
                    263:        for (bpp = --sp->cbpp; bpp >= sp->bpp; --bpp)
                    264:                if ((*bpp)->b_flags & B_CALL) {
                    265:                        brelvp(*bpp);
                    266:                        free(*bpp, M_SEGMENT);
                    267:                } else
                    268:                        brelse(*bpp);
                    269:        lfs_segunlock(fs);
                    270: err1:  
                    271:        free(start, M_SEGMENT);
                    272:        return (error);
                    273: }
                    274: 
                    275: /*
                    276:  * lfs_bmapv:
                    277:  *
                    278:  * This will fill in the current disk address for arrays of blocks.
                    279:  *
                    280:  *  0 on success
                    281:  * -1/errno is return on error.
                    282:  */
                    283: int
                    284: lfs_bmapv(p, uap, retval)
                    285:        struct proc *p;
                    286:        struct lfs_bmapv_args /* {
                    287:                syscallarg(fsid_t *) fsidp;
                    288:                syscallarg(struct block_info *) blkiov;
                    289:                syscallarg(int) blkcnt;
                    290:        } */ *uap;
                    291:        register_t *retval;
                    292: {
                    293:        BLOCK_INFO *blkp;
                    294:        struct mount *mntp;
                    295:        struct vnode *vp;
                    296:        fsid_t fsid;
                    297:        void *start;
                    298:        daddr_t daddr;
                    299:        int cnt, error, step;
                    300: 
                    301:        if (error = suser(p->p_ucred, &p->p_acflag))
                    302:                return (error);
                    303: 
                    304:        if (error = copyin(SCARG(uap, fsidp), &fsid, sizeof(fsid_t)))
                    305:                return (error);
                    306:        if ((mntp = getvfs(&fsid)) == NULL)
                    307:                return (EINVAL);
                    308: 
                    309:        cnt = SCARG(uap, blkcnt);
                    310: //     start = blkp = malloc(cnt * sizeof(BLOCK_INFO), M_SEGMENT, M_WAITOK);
                    311:        MALLOC(blkp, BLOCK_INFO *, cnt * sizeof(BLOCK_INFO), M_SEGMENT, M_WAITOK);
                    312:        start = blkp;
                    313:        if (error = copyin(SCARG(uap, blkiov), blkp,
                    314:            cnt * sizeof(BLOCK_INFO))) {
                    315:                free(blkp, M_SEGMENT);
                    316:                return (error);
                    317:        }
                    318: 
                    319:        for (step = cnt; step--; ++blkp) {
                    320:                if (blkp->bi_lbn == LFS_UNUSED_LBN)
                    321:                        continue;
                    322:                /* Could be a deadlock ? */
                    323:                if (VFS_VGET(mntp, blkp->bi_inode, &vp))
                    324:                        daddr = LFS_UNUSED_DADDR;
                    325:                else {
                    326:                        if (VOP_BMAP(vp, blkp->bi_lbn, NULL, &daddr, NULL))
                    327:                                daddr = LFS_UNUSED_DADDR;
                    328:                        vput(vp);
                    329:                }
                    330:                blkp->bi_daddr = daddr;
                    331:         }
                    332:        copyout(start, SCARG(uap, blkiov), cnt * sizeof(BLOCK_INFO));
                    333:        free(start, M_SEGMENT);
                    334:        return (0);
                    335: }
                    336: 
                    337: /*
                    338:  * lfs_segclean:
                    339:  *
                    340:  * Mark the segment clean.
                    341:  *
                    342:  *  0 on success
                    343:  * -1/errno is return on error.
                    344:  */
                    345: int
                    346: lfs_segclean(p, uap, retval)
                    347:        struct proc *p;
                    348:        struct lfs_segclean_args /* {
                    349:                syscallarg(fsid_t *) fsidp;
                    350:                syscallarg(u_long) segment;
                    351:        } */ *uap;
                    352:        register_t *retval;
                    353: {
                    354:        CLEANERINFO *cip;
                    355:        SEGUSE *sup;
                    356:        struct buf *bp;
                    357:        struct mount *mntp;
                    358:        struct lfs *fs;
                    359:        fsid_t fsid;
                    360:        int error;
                    361: 
                    362:        if (error = suser(p->p_ucred, &p->p_acflag))
                    363:                return (error);
                    364: 
                    365:        if (error = copyin(SCARG(uap, fsidp), &fsid, sizeof(fsid_t)))
                    366:                return (error);
                    367:        if ((mntp = getvfs(&fsid)) == NULL)
                    368:                return (EINVAL);
                    369: 
                    370:        fs = VFSTOUFS(mntp)->um_lfs;
                    371: 
                    372:        if (datosn(fs, fs->lfs_curseg) == SCARG(uap, segment))
                    373:                return (EBUSY);
                    374: 
                    375:        LFS_SEGENTRY(sup, fs, SCARG(uap, segment), bp);
                    376:        if (sup->su_flags & SEGUSE_ACTIVE) {
                    377:                brelse(bp);
                    378:                return (EBUSY);
                    379:        }
                    380:        fs->lfs_avail += fsbtodb(fs, fs->lfs_ssize) - 1;
                    381:        fs->lfs_bfree += (sup->su_nsums * LFS_SUMMARY_SIZE / DEV_BSIZE) +
                    382:            sup->su_ninos * btodb(fs->lfs_bsize);
                    383:        sup->su_flags &= ~SEGUSE_DIRTY;
                    384:        (void) VOP_BWRITE(bp);
                    385: 
                    386:        LFS_CLEANERINFO(cip, fs, bp);
                    387:        ++cip->clean;
                    388:        --cip->dirty;
                    389:        (void) VOP_BWRITE(bp);
                    390:        wakeup(&fs->lfs_avail);
                    391:        return (0);
                    392: }
                    393: 
                    394: /*
                    395:  * lfs_segwait:
                    396:  *
                    397:  * This will block until a segment in file system fsid is written.  A timeout
                    398:  * in milliseconds may be specified which will awake the cleaner automatically.
                    399:  * An fsid of -1 means any file system, and a timeout of 0 means forever.
                    400:  *
                    401:  *  0 on success
                    402:  *  1 on timeout
                    403:  * -1/errno is return on error.
                    404:  */
                    405: int
                    406: lfs_segwait(p, uap, retval)
                    407:        struct proc *p;
                    408:        struct lfs_segwait_args /* {
                    409:                syscallarg(fsid_t *) fsidp;
                    410:                syscallarg(struct timeval *) tv;
                    411:        } */ *uap;
                    412:        register_t *retval;
                    413: {
                    414:        extern int lfs_allclean_wakeup;
                    415:        struct mount *mntp;
                    416:        struct timeval atv;
                    417:        fsid_t fsid;
                    418:        void *addr;
                    419:        u_long timeout;
                    420:        int error, s;
                    421: 
                    422:        if (error = suser(p->p_ucred, &p->p_acflag)) {
                    423:                return (error);
                    424: }
                    425: #ifdef WHEN_QUADS_WORK
                    426:        if (error = copyin(SCARG(uap, fsidp), &fsid, sizeof(fsid_t)))
                    427:                return (error);
                    428:        if (fsid == (fsid_t)-1)
                    429:                addr = &lfs_allclean_wakeup;
                    430:        else {
                    431:                if ((mntp = getvfs(&fsid)) == NULL)
                    432:                        return (EINVAL);
                    433:                addr = &VFSTOUFS(mntp)->um_lfs->lfs_nextseg;
                    434:        }
                    435: #else
                    436:        if (error = copyin(SCARG(uap, fsidp), &fsid, sizeof(fsid_t)))
                    437:                return (error);
                    438:        if ((mntp = getvfs(&fsid)) == NULL)
                    439:                addr = &lfs_allclean_wakeup;
                    440:        else
                    441:                addr = &VFSTOUFS(mntp)->um_lfs->lfs_nextseg;
                    442: #endif
                    443: 
                    444:        if (SCARG(uap, tv)) {
                    445:                if (error =
                    446:                    copyin(SCARG(uap, tv), &atv, sizeof(struct timeval)))
                    447:                        return (error);
                    448:                if (itimerfix(&atv))
                    449:                        return (EINVAL);
                    450:                s = splclock();
                    451:                timeradd(&atv, &time, &atv);
                    452:                timeout = hzto(&atv);
                    453:                splx(s);
                    454:        } else
                    455:                timeout = 0;
                    456: 
                    457:        error = tsleep(addr, PCATCH | PUSER, "segment", timeout);
                    458:        return (error == ERESTART ? EINTR : 0);
                    459: }
                    460: 
                    461: /*
                    462:  * VFS_VGET call specialized for the cleaner.  The cleaner already knows the
                    463:  * daddr from the ifile, so don't look it up again.  If the cleaner is
                    464:  * processing IINFO structures, it may have the ondisk inode already, so
                    465:  * don't go retrieving it again.
                    466:  */
                    467: int
                    468: lfs_fastvget(mp, ino, daddr, vpp, dinp)
                    469:        struct mount *mp;
                    470:        ino_t ino;
                    471:        daddr_t daddr;
                    472:        struct vnode **vpp;
                    473:        struct dinode *dinp;
                    474: {
                    475:        register struct inode *ip;
                    476:        struct vnode *vp;
                    477:        struct ufsmount *ump;
                    478:        struct buf *bp;
                    479:        dev_t dev;
                    480:        int error;
                    481: 
                    482:        ump = VFSTOUFS(mp);
                    483:        dev = ump->um_dev;
                    484:        /*
                    485:         * This is playing fast and loose.  Someone may have the inode
                    486:         * locked, in which case they are going to be distinctly unhappy
                    487:         * if we trash something.
                    488:         */
                    489:        if ((*vpp = ufs_ihashlookup(dev, ino)) != NULL) {
                    490:                lfs_vref(*vpp);
                    491:                if ((*vpp)->v_flag & VXLOCK)
                    492:                        printf ("Cleaned vnode VXLOCKED\n");
                    493:                ip = VTOI(*vpp);
                    494:                if (ip->i_flag & IN_LOCKED)
                    495:                        printf("cleaned vnode locked\n");
                    496:                if (!(ip->i_flag & IN_MODIFIED)) {
                    497:                        ++ump->um_lfs->lfs_uinodes;
                    498:                        ip->i_flag |= IN_MODIFIED;
                    499:                }
                    500:                ip->i_flag |= IN_MODIFIED;
                    501:                return (0);
                    502:        }
                    503: 
                    504:        /* Allocate new vnode/inode. */
                    505:        if (error = lfs_vcreate(mp, ino, &vp)) {
                    506:                *vpp = NULL;
                    507:                return (error);
                    508:        }
                    509: 
                    510:        /*
                    511:         * Put it onto its hash chain and lock it so that other requests for
                    512:         * this inode will block if they arrive while we are sleeping waiting
                    513:         * for old data structures to be purged or for the contents of the
                    514:         * disk portion of this inode to be read.
                    515:         */
                    516:        ip = VTOI(vp);
                    517:        ufs_ihashins(ip);
                    518: 
                    519:        /*
                    520:         * XXX
                    521:         * This may not need to be here, logically it should go down with
                    522:         * the i_devvp initialization.
                    523:         * Ask Kirk.
                    524:         */
                    525:        ip->i_lfs = ump->um_lfs;
                    526: 
                    527:        /* Read in the disk contents for the inode, copy into the inode. */
                    528:        if (dinp)
                    529:                if (error = copyin(dinp, &ip->i_din, sizeof(struct dinode)))
                    530:                        return (error);
                    531:        else {
                    532:                if (error = bread(ump->um_devvp, daddr,
                    533:                    (int)ump->um_lfs->lfs_bsize, NOCRED, &bp)) {
                    534:                        /*
                    535:                         * The inode does not contain anything useful, so it
                    536:                         * would be misleading to leave it on its hash chain.
                    537:                         * Iput() will return it to the free list.
                    538:                         */
                    539:                        ufs_ihashrem(ip);
                    540: 
                    541:                        /* Unlock and discard unneeded inode. */
                    542:                        lfs_vunref(vp);
                    543:                        brelse(bp);
                    544:                        *vpp = NULL;
                    545:                        return (error);
                    546:                }
                    547:                ip->i_din =
                    548:                    *lfs_ifind(ump->um_lfs, ino, (struct dinode *)bp->b_data);
                    549:                brelse(bp);
                    550:        }
                    551: 
                    552:        /* Inode was just read from user space or disk, make sure it's locked */
                    553:        ip->i_flag |= IN_LOCKED;
                    554: 
                    555:        /*
                    556:         * Initialize the vnode from the inode, check for aliases.  In all
                    557:         * cases re-init ip, the underlying vnode/inode may have changed.
                    558:         */
                    559:        if (error = ufs_vinit(mp, lfs_specop_p, LFS_FIFOOPS, &vp)) {
                    560:                lfs_vunref(vp);
                    561:                *vpp = NULL;
                    562:                return (error);
                    563:        }
                    564:        /*
                    565:         * Finish inode initialization now that aliasing has been resolved.
                    566:         */
                    567:        ip->i_devvp = ump->um_devvp;
                    568:        ip->i_flag |= IN_MODIFIED;
                    569:        ++ump->um_lfs->lfs_uinodes;
                    570:        VREF(ip->i_devvp);
                    571:        *vpp = vp;
                    572:        return (0);
                    573: }
                    574: struct buf *
                    575: lfs_fakebuf(vp, lbn, size, uaddr)
                    576:        struct vnode *vp;
                    577:        int lbn;
                    578:        size_t size;
                    579:        caddr_t uaddr;
                    580: {
                    581:        struct buf *bp;
                    582: 
                    583:        bp = lfs_newbuf(vp, lbn, 0);
                    584:        bp->b_saveaddr = uaddr;
                    585:        bp->b_bufsize = size;
                    586:        bp->b_bcount = size;
                    587:        bp->b_flags |= B_INVAL;
                    588:        return (bp);
                    589: }

unix.superglobalmegacorp.com

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