2.11BSD/sys/sys/ufs_inode.c

Compare this file to the similar file:
Show the results in this format:

/*
 * Copyright (c) 1986 Regents of the University of California.
 * All rights reserved.  The Berkeley software License Agreement
 * specifies the terms and conditions for redistribution.
 *
 *	@(#)ufs_inode.c	1.7 (2.11BSD GTE) 1997/2/7
 */

#include "param.h"
#include "../machine/seg.h"

#include "user.h"
#include "proc.h"
#include "inode.h"
#include "fs.h"
#include "mount.h"
#include "kernel.h"
#include "buf.h"
#include "text.h"
#include "systm.h"
#ifdef QUOTA
#include "quota.h"
#endif
#include "syslog.h"

#define	INOHSZ	16		/* must be power of two */
#define	INOHASH(dev,ino)	(((dev)+(ino))&(INOHSZ-1))

union ihead {				/* inode LRU cache, stolen */
	union  ihead *ih_head[2];
	struct inode *ih_chain[2];
} ihead[INOHSZ];

struct inode *ifreeh, **ifreet;

/*
 * Initialize hash links for inodes
 * and build inode free list.
 */
ihinit()
{
	register int i;
	register struct inode *ip = inode;
	register union  ihead *ih = ihead;

	for (i = INOHSZ; --i >= 0; ih++) {
		ih->ih_head[0] = ih;
		ih->ih_head[1] = ih;
	}
	ifreeh = ip;
	ifreet = &ip->i_freef;
	ip->i_freeb = &ifreeh;
	ip->i_forw = ip;
	ip->i_back = ip;
	for (i = ninode; --i > 0; ) {
		++ip;
		ip->i_forw = ip;
		ip->i_back = ip;
		*ifreet = ip;
		ip->i_freeb = ifreet;
		ifreet = &ip->i_freef;
	}
	ip->i_freef = NULL;
}

/*
 * Find an inode if it is incore.
 */
struct inode *
ifind(dev, ino)
	register dev_t dev;
	register ino_t ino;
{
	register struct inode *ip;
	union ihead *ih;

	ih = &ihead[INOHASH(dev, ino)];
	for (ip = ih->ih_chain[0]; ip != (struct inode *)ih; ip = ip->i_forw)
		if (ino == ip->i_number && dev == ip->i_dev)
			return(ip);
	return((struct inode *)NULL);
}

/*
 * Look up an inode by device,inumber.
 * If it is in core (in the inode structure),
 * honor the locking protocol.
 * If it is not in core, read it in from the
 * specified device.
 * If the inode is mounted on, perform
 * the indicated indirection.
 * In all cases, a pointer to a locked
 * inode structure is returned.
 *
 * panic: no imt -- if the mounted file
 *	system is not in the mount table.
 *	"cannot happen"
 */
struct inode *
iget(dev, fs, ino)
	dev_t dev;
	register struct fs *fs;
	ino_t ino;
{
	register struct inode *ip;
	union ihead *ih;
	struct buf *bp;
	struct dinode *dp;
#ifdef EXTERNALITIMES
	struct icommon2 xic2;
#endif
#ifdef	QUOTA
	struct dquot **xdq;
#endif

loop:
	ih = &ihead[INOHASH(dev, ino)];
	for (ip = ih->ih_chain[0]; ip != (struct inode *)ih; ip = ip->i_forw)
		if (ino == ip->i_number && dev == ip->i_dev) {
			/*
			 * Following is essentially an inline expanded
			 * copy of igrab(), expanded inline for speed,
			 * and so that the test for a mounted on inode
			 * can be deferred until after we are sure that
			 * the inode isn't busy.
			 */
			if ((ip->i_flag&ILOCKED) != 0) {
				ip->i_flag |= IWANT;
				sleep((caddr_t)ip, PINOD);
				goto loop;
			}
			if ((ip->i_flag&IMOUNT) != 0) {
				register struct mount *mp;

				for (mp = &mount[0]; mp < &mount[NMOUNT]; mp++)
					if(mp->m_inodp == ip) {
						dev = mp->m_dev;
						fs = &mp->m_filsys;
						ino = ROOTINO;
						goto loop;
					}
				panic("no imt");
			}
			if (ip->i_count == 0) {		/* ino on free list */
				register struct inode *iq;

				if (iq = ip->i_freef)
					iq->i_freeb = ip->i_freeb;
				else
					ifreet = ip->i_freeb;
				*ip->i_freeb = iq;
				ip->i_freef = NULL;
				ip->i_freeb = NULL;
			}
			ip->i_count++;
			ip->i_flag |= ILOCKED;
			return(ip);
		}

	if ((ip = ifreeh) == NULL) {
		tablefull("inode");
		u.u_error = ENFILE;
		return(NULL);
	}
	if (ip->i_count)
		panic("free inode isn't");
{
	register struct inode *iq;

	if (iq = ip->i_freef)
		iq->i_freeb = &ifreeh;
	ifreeh = iq;
}
	ip->i_freef = NULL;
	ip->i_freeb = NULL;
	/*
	 * Now to take inode off the hash chain it was on
	 * (initially, or after an iflush, it is on a "hash chain"
	 * consisting entirely of itself, and pointed to by no-one,
	 * but that doesn't matter), and put it on the chain for
	 * its new (ino, dev) pair
	 */
	remque(ip);
	insque(ip, ih);
	ip->i_dev = dev;
	ip->i_fs = fs;
	ip->i_number = ino;
	cacheinval(ip);
	ip->i_flag = ILOCKED;
	ip->i_count++;
	ip->i_lastr = 0;
#ifdef QUOTA
	QUOTAMAP();
	xdq = &ix_dquot[ip - inode];
	dqrele(*xdq);
	QUOTAUNMAP();
#endif
	bp = bread(dev, itod(ino));
	/*
	 * Check I/O errors
	 */
	if ((bp->b_flags&B_ERROR) != 0) {
		brelse(bp);
		/*
		 * the inode doesn't contain anything useful, so it would
		 * be misleading to leave it on its hash chain.
		 * 'iput' will take care of putting it back on the free list.
		 */
		remque(ip);
		ip->i_forw = ip;
		ip->i_back = ip;
		/*
		 * we also loose its inumber, just in case (as iput
		 * doesn't do that any more) - but as it isn't on its
		 * hash chain, I doubt if this is really necessary .. kre
		 * (probably the two methods are interchangable)
		 */
		ip->i_number = 0;
#ifdef QUOTA
		QUOTAMAP();
		*xdq = NODQUOT;
		QUOTAUNMAP();
#endif
		iput(ip);
		return(NULL);
	}
	dp = (struct dinode *)mapin(bp);
	dp += itoo(ino);
	ip->i_ic1 = dp->di_ic1;
	ip->i_flags = dp->di_flags;
#ifdef EXTERNALITIMES
	xic2 = dp->di_ic2;
#else
	ip->i_ic2 = dp->di_ic2;
#endif
	bcopy(dp->di_addr, ip->i_addr, NADDR * sizeof (daddr_t));
	mapout(bp);
	brelse(bp);
#ifdef EXTERNALITIMES
	mapseg5(xitimes, xitdesc);
	((struct icommon2 *)SEG5)[ip-inode] = xic2;
	normalseg5();
#endif
#ifdef QUOTA
	QUOTAMAP();
	if	(ip->i_mode == 0)
		*xdq = NODQUOT;
	else
		*xdq = inoquota(ip);
	QUOTAUNMAP();
#endif
	return (ip);
}

/*
 * Convert a pointer to an inode into a reference to an inode.
 *
 * This is basically the internal piece of iget (after the
 * inode pointer is located) but without the test for mounted
 * filesystems.  It is caller's responsibility to check that
 * the inode pointer is valid.
 */
igrab(ip)
	register struct inode *ip;
{
	while ((ip->i_flag&ILOCKED) != 0) {
		ip->i_flag |= IWANT;
		sleep((caddr_t)ip, PINOD);
	}
	if (ip->i_count == 0) {		/* ino on free list */
		register struct inode *iq;

		if (iq = ip->i_freef)
			iq->i_freeb = ip->i_freeb;
		else
			ifreet = ip->i_freeb;
		*ip->i_freeb = iq;
		ip->i_freef = NULL;
		ip->i_freeb = NULL;
	}
	ip->i_count++;
	ip->i_flag |= ILOCKED;
}

/*
 * Decrement reference count of
 * an inode structure.
 * On the last reference,
 * write the inode out and if necessary,
 * truncate and deallocate the file.
 */
iput(ip)
	register struct inode *ip;
{

#ifdef notnow
	/*
	 * This code requires a lot of workarounds, you have to change
	 * lots of places to gratuitously lock just so we can unlock it.
	 * Not worth it.  -- KB
	 */
	if ((ip->i_flag & ILOCKED) == 0)
		panic("iput");
#endif
	IUNLOCK(ip);
	irele(ip);
}

irele(ip)
	register struct inode *ip;
{
	if (ip->i_count == 1) {
		ip->i_flag |= ILOCKED;
		if (ip->i_nlink <= 0 && ip->i_fs->fs_ronly == 0) {
#ifdef QUOTA
			QUOTAMAP();
			(void) chkiq(ip->i_dev, ip, ip->i_uid, 0);
			dqrele(ix_dquot[ip - inode]);
			ix_dquot[ip - inode] = NODQUOT;
			QUOTAUNMAP();
#endif
			itrunc(ip, (u_long)0, 0);
			ip->i_mode = 0;
			ip->i_rdev = 0;
			ip->i_flag |= IUPD|ICHG;
			ifree(ip, ip->i_number);
		}
		IUPDAT(ip, &time, &time, 0);
		IUNLOCK(ip);
		ip->i_flag = 0;
		/*
		 * Put the inode on the end of the free list.
		 * Possibly in some cases it would be better to
		 * put the inode at the head of the free list,
		 * (eg: where i_mode == 0 || i_number == 0)
		 * but I will think about that later .. kre
		 * (i_number is rarely 0 - only after an i/o error in iget,
		 * where i_mode == 0, the inode will probably be wanted
		 * again soon for an ialloc, so possibly we should keep it)
		 */
		if (ifreeh) {
			*ifreet = ip;
			ip->i_freeb = ifreet;
		} else {
			ifreeh = ip;
			ip->i_freeb = &ifreeh;
		}
		ip->i_freef = NULL;
		ifreet = &ip->i_freef;
	} else if (!(ip->i_flag & ILOCKED))
		ITIMES(ip, &time, &time);
	ip->i_count--;
}

/*
 * Check accessed and update flags on
 * an inode structure.
 * If any are on, update the inode
 * with the current time.
 * If waitfor set, then must insure
 * i/o order so wait for the write to complete.
 */
iupdat(ip, ta, tm, waitfor)
	struct inode *ip;
	struct timeval *ta, *tm;
	int waitfor;
{
	register struct buf *bp;
	register struct dinode *dp;
#ifdef EXTERNALITIMES
	struct icommon2 xic2, *xicp2;
#endif
	register struct inode *tip = ip;

	if ((tip->i_flag & (IUPD|IACC|ICHG|IMOD)) == 0)
		return;
	if (tip->i_fs->fs_ronly)
		return;
	bp = bread(tip->i_dev, itod(tip->i_number));
	if (bp->b_flags & B_ERROR) {
		brelse(bp);
		return;
	}
#ifdef EXTERNALITIMES
	mapseg5(xitimes, xitdesc);
	xicp2 = &((struct icommon2 *)SEG5)[ip - inode];
	if (tip->i_flag & IACC)
		xicp2->ic_atime = ta->tv_sec;
	if (tip->i_flag & IUPD)
		xicp2->ic_mtime = tm->tv_sec;
	if (tip->i_flag & ICHG)
		xicp2->ic_ctime = time.tv_sec;
	xic2 = *xicp2;
	normalseg5();
#else
	if (tip->i_flag&IACC)
		tip->i_atime = ta->tv_sec;
	if (tip->i_flag&IUPD)
		tip->i_mtime = tm->tv_sec;
	if (tip->i_flag&ICHG)
		tip->i_ctime = time.tv_sec;
#endif
	tip->i_flag &= ~(IUPD|IACC|ICHG|IMOD);
	dp = (struct dinode *)mapin(bp) + itoo(tip->i_number);
	dp->di_ic1 = tip->i_ic1;
	dp->di_flags = tip->i_flags;
#ifdef EXTERNALITIMES
	dp->di_ic2 = xic2;
#else
	dp->di_ic2 = tip->i_ic2;
#endif
	bcopy(ip->i_addr, dp->di_addr, NADDR * sizeof (daddr_t));
	mapout(bp);
	if (waitfor && ((ip->i_fs->fs_flags & MNT_ASYNC) == 0))
		bwrite(bp);
	else
		bdwrite(bp);
}

#define	SINGLE	0	/* index of single indirect block */
#define	DOUBLE	1	/* index of double indirect block */
#define	TRIPLE	2	/* index of triple indirect block */
/*
 * Truncate the inode ip to at most
 * length size.  Free affected disk
 * blocks -- the blocks of the file
 * are removed in reverse order.
 *
 * NB: triple indirect blocks are untested.
 */
itrunc(oip,length, ioflags)
	register struct inode *oip;
	u_long length;
	int	ioflags;
{
	daddr_t lastblock;
	register int i;
	register struct inode *ip;
	daddr_t bn, lastiblock[NIADDR];
	struct buf *bp;
	int offset, level;
	struct inode tip;
	int aflags;
#ifdef QUOTA
	long bytesreleased;
#endif

	aflags = B_CLRBUF;
	if (ioflags & IO_SYNC)
		aflags |= B_SYNC;

	/*
	 * special hack for pipes, since size for them isn't the size of
	 * the file, it's the amount currently waiting for transfer.  It's
	 * unclear that this will work, though, because pipes can (although
	 * rarely do) get bigger than MAXPIPSIZ.  Don't think it worked
	 * in V7 either, I don't really understand what's going on.
	 */
	if (oip->i_flag & IPIPE)
		oip->i_size = MAXPIPSIZ;
	else if (oip->i_size == length)
		goto updret;

	/*
	 * Lengthen the size of the file. We must ensure that the
	 * last byte of the file is allocated. Since the smallest
	 * value of osize is 0, length will be at least 1.
	 */
	if (oip->i_size < length) {
		bn = bmap(oip, lblkno(length - 1), B_WRITE, aflags);
		if (u.u_error || bn < 0)
			return;
#ifdef	QUOTA
		bytesreleased = oip->i_size - length;
#endif
		oip->i_size = length;
		goto doquotaupd;
	}

	/*
	 * Calculate index into inode's block list of
	 * last direct and indirect blocks (if any)
	 * which we want to keep.  Lastblock is -1 when
	 * the file is truncated to 0.
	 */
	lastblock = lblkno(length + DEV_BSIZE - 1) - 1;
	lastiblock[SINGLE] = lastblock - NDADDR;
	lastiblock[DOUBLE] = lastiblock[SINGLE] - NINDIR;
	lastiblock[TRIPLE] = lastiblock[DOUBLE] - NINDIR * NINDIR;
	/*
	 * Update the size of the file. If the file is not being
	 * truncated to a block boundry, the contents of the
	 * partial block following the end of the file must be
	 * zero'ed in case it ever become accessable again because
	 * of subsequent file growth.
	 */
	offset = blkoff(length);
	if (offset) {
		bn = bmap(oip, lblkno(length), B_WRITE, aflags);
		if (u.u_error || bn < 0)
			return;
		bp = bread(oip->i_dev, bn);
		if (bp->b_flags & B_ERROR) {
			u.u_error = EIO;
			brelse(bp);
			return;
		}
		bzero(mapin(bp) + offset, (u_int)(DEV_BSIZE - offset));
		mapout(bp);
		bdwrite(bp);
	}
	/*
	 * Update file and block pointers
	 * on disk before we start freeing blocks.
	 * If we crash before free'ing blocks below,
	 * the blocks will be returned to the free list.
	 * lastiblock values are also normalized to -1
	 * for calls to indirtrunc below.
	 */
	tip = *oip;
#ifdef QUOTA
	bytesreleased = oip->i_size - length;
#endif
	oip->i_size = length;
	for (level = TRIPLE; level >= SINGLE; level--)
		if (lastiblock[level] < 0) {
			oip->i_ib[level] = 0;
			lastiblock[level] = -1;
		}
	for (i = NDADDR - 1; i > lastblock; i--)
		oip->i_db[i] = 0;

	/*
	 * Indirect blocks first.
	 */
	ip = &tip;
	for (level = TRIPLE; level >= SINGLE; level--) {
		bn = ip->i_ib[level];
		if (bn != 0) {
			indirtrunc(ip, bn, lastiblock[level], level, aflags);
			if (lastiblock[level] < 0) {
				ip->i_ib[level] = 0;
				free(ip, bn);
			}
		}
		if (lastiblock[level] >= 0)
			goto done;
	}

	/*
	 * All whole direct blocks.
	 */
	for (i = NDADDR - 1; i > lastblock; i--) {
		bn = ip->i_db[i];
		if (bn == 0)
			continue;
		ip->i_db[i] = 0;
		free(ip, bn);
	}
	if (lastblock < 0)
		goto done;

done:
#ifdef DIAGNOSTIC
/* BEGIN PARANOIA */
	for (level = SINGLE; level <= TRIPLE; level++)
		if (ip->i_ib[level] != oip->i_ib[level])
			panic("itrunc1");
	for (i = 0; i < NDADDR; i++)
		if (ip->i_db[i] != oip->i_db[i])
			panic("itrunc2");
/* END PARANOIA */
#endif

doquotaupd:
#ifdef QUOTA
	QUOTAMAP();
	(void)chkdq(oip, -bytesreleased, 0);
	QUOTAUNMAP();
#endif
updret:
	oip->i_flag |= ICHG|IUPD;
	iupdat(oip, &time, &time, 1);
}

/*
 * Release blocks associated with the inode ip and
 * stored in the indirect block bn.  Blocks are free'd
 * in LIFO order up to (but not including) lastbn.  If
 * level is greater than SINGLE, the block is an indirect
 * block and recursive calls to indirtrunc must be used to
 * cleanse other indirect blocks.
 *
 * NB: triple indirect blocks are untested.
 */
indirtrunc(ip, bn, lastbn, level, aflags)
	struct inode *ip;
	daddr_t bn, lastbn;
	int level;
	int aflags;
{
	register struct buf *bp;
	daddr_t nb, last;
	long factor;

	/*
	 * Calculate index in current block of last
	 * block to be kept.  -1 indicates the entire
	 * block so we need not calculate the index.
	 */
	switch(level) {
		case SINGLE:
			factor = 1;
			break;
		case DOUBLE:
			factor = NINDIR;
			break;
		case TRIPLE:
			factor = NINDIR * NINDIR;
			break;
	}
	last = lastbn;
	if (lastbn > 0)
		last = last / factor;
	/*
	 * Get buffer of block pointers, zero those
	 * entries corresponding to blocks to be free'd,
	 * and update on disk copy first.
	 */
	{
		register daddr_t *bap;
		register struct buf *cpy;

		bp = bread(ip->i_dev, bn);
		if (bp->b_flags&B_ERROR) {
			brelse(bp);
			return;
		}
		cpy = geteblk();
		copy(bftopaddr(bp), bftopaddr(cpy), btoc(DEV_BSIZE));
		bap = (daddr_t *)mapin(bp);
		bzero((caddr_t)&bap[last + 1],
		    (u_int)(NINDIR - (last + 1)) * sizeof(daddr_t));
		mapout(bp);
		if (aflags & B_SYNC)
			bwrite(bp);
		else
			bawrite(bp);
		bp = cpy;
	}

	/*
	 * Optimized for single indirect blocks, i.e. until a file is
	 * greater than 4K + 256K you don't have to do a mapin/mapout
	 * for every entry.  The mapin/mapout is required since free()
	 * may have to map an item in.  Have to use another routine
	 * since it requires 1K of kernel stack to get around the problem
	 * and that doesn't work well with recursion.
	 */
	if (level == SINGLE)
		trsingle(ip, bp, last, aflags);
	else {
		register daddr_t *bstart, *bstop;

		bstart = (daddr_t *)mapin(bp);
		bstop = &bstart[last];
		bstart += NINDIR - 1;
		/*
		 * Recursively free totally unused blocks.
		 */
		for (;bstart > bstop;--bstart) {
			nb = *bstart;
			if (nb) {
				mapout(bp);
				indirtrunc(ip,nb,(daddr_t)-1, level-1, aflags);
				free(ip, nb);
				mapin(bp);
			}
		}
		mapout(bp);

		/*
		 * Recursively free last partial block.
		 */
		if (lastbn >= 0) {

			mapin(bp);
			nb = *bstop;
			mapout(bp);
			last = lastbn % factor;
			if (nb != 0)
				indirtrunc(ip, nb, last, level - 1, aflags);
		}
	}
	brelse(bp);
}

static
trsingle(ip, bp,last, aflags)
	register struct inode *ip;
	caddr_t bp;
	daddr_t last;
	int aflags;
{
	register daddr_t *bstart, *bstop;
	daddr_t blarray[NINDIR];

	bcopy(mapin(bp),blarray,NINDIR * sizeof(daddr_t));
	mapout(bp);
	bstart = &blarray[NINDIR - 1];
	bstop = &blarray[last];
	for (;bstart > bstop;--bstart)
		if (*bstart)
			free(ip, *bstart);
}

/*
 * remove any inodes in the inode cache belonging to dev
 *
 * There should not be any active ones, return error if any are found
 * (nb: this is a user error, not a system err)
 *
 * Also, count the references to dev by block devices - this really
 * has nothing to do with the object of the procedure, but as we have
 * to scan the inode table here anyway, we might as well get the
 * extra benefit.
 *
 * this is called from sumount() when dev is being unmounted
 */
#ifdef QUOTA
iflush(dev, iq)
	struct inode *iq;
#else
iflush(dev)
#endif
	dev_t dev;
{
	register struct inode *ip;
	register int open = 0;

	for (ip = inode; ip < inodeNINODE; ip++) {
#ifdef QUOTA
		if (ip != iq && ip->i_dev == dev)
#else
		if (ip->i_dev == dev)
#endif
			if (ip->i_count)
				return(-1);
			else {
				remque(ip);
				ip->i_forw = ip;
				ip->i_back = ip;
				/*
				 * as i_count == 0, the inode was on the free
				 * list already, just leave it there, it will
				 * fall off the bottom eventually. We could
				 * perhaps move it to the head of the free
				 * list, but as umounts are done so
				 * infrequently, we would gain very little,
				 * while making the code bigger.
				 */
#ifdef QUOTA
				QUOTAMAP();
				dqrele(ix_dquot[ip - inode]);
				ix_dquot[ip - inode] = NODQUOT;
				QUOTAUNMAP();
#endif
			}
		else if (ip->i_count && (ip->i_mode&IFMT)==IFBLK &&
		    ip->i_rdev == dev)
			open++;
	}
	return (open);
}

/*
 * Lock an inode. If its already locked, set the WANT bit and sleep.
 */
ilock(ip)
	register struct inode *ip;
{

	ILOCK(ip);
}

/*
 * Unlock an inode.  If WANT bit is on, wakeup.
 */
iunlock(ip)
	register struct inode *ip;
{

	IUNLOCK(ip);
}