pass4.c revision 1.7
1/* $NetBSD: pass4.c,v 1.7 2003/03/29 00:09:43 perseant Exp $	 */
2
3/*
4 * Copyright (c) 1980, 1986, 1993
5 *	The Regents of the University of California.  All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 *    notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 *    notice, this list of conditions and the following disclaimer in the
14 *    documentation and/or other materials provided with the distribution.
15 * 3. All advertising materials mentioning features or use of this software
16 *    must display the following acknowledgement:
17 *	This product includes software developed by the University of
18 *	California, Berkeley and its contributors.
19 * 4. Neither the name of the University nor the names of its contributors
20 *    may be used to endorse or promote products derived from this software
21 *    without specific prior written permission.
22 *
23 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
24 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
25 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
27 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
28 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
29 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
30 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
32 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
33 * SUCH DAMAGE.
34 */
35
36#include <sys/param.h>
37#include <sys/time.h>
38#include <sys/mount.h>
39#include <ufs/ufs/inode.h>
40
41#define vnode uvnode
42#define buf ubuf
43#define panic call_panic
44#include <ufs/lfs/lfs.h>
45
46#include <err.h>
47#include <stdlib.h>
48#include <string.h>
49
50#include "bufcache.h"
51#include "vnode.h"
52#include "lfs.h"
53
54#include "fsutil.h"
55#include "fsck.h"
56#include "extern.h"
57
58extern SEGUSE *seg_table;
59
60void
61pass4()
62{
63	register ino_t inumber;
64	register struct zlncnt *zlnp;
65	struct dinode *dp;
66	struct inodesc idesc;
67	int n;
68
69	memset(&idesc, 0, sizeof(struct inodesc));
70	idesc.id_type = ADDR;
71	idesc.id_func = pass4check;
72	for (inumber = ROOTINO; inumber <= lastino; inumber++) {
73		idesc.id_number = inumber;
74		switch (statemap[inumber]) {
75
76		case FSTATE:
77		case DFOUND:
78			n = lncntp[inumber];
79			if (n)
80				adjust(&idesc, (short) n);
81			else {
82				for (zlnp = zlnhead; zlnp; zlnp = zlnp->next)
83					if (zlnp->zlncnt == inumber) {
84						zlnp->zlncnt = zlnhead->zlncnt;
85						zlnp = zlnhead;
86						zlnhead = zlnhead->next;
87						free((char *) zlnp);
88						clri(&idesc, "UNREF", 1);
89						break;
90					}
91			}
92			break;
93
94		case DSTATE:
95			clri(&idesc, "UNREF", 1);
96			break;
97
98		case DCLEAR:
99			dp = ginode(inumber);
100			if (dp->di_size == 0) {
101				clri(&idesc, "ZERO LENGTH", 1);
102				break;
103			}
104			/* fall through */
105		case FCLEAR:
106			clri(&idesc, "BAD/DUP", 1);
107			break;
108
109		case USTATE:
110			break;
111
112		default:
113			err(8, "BAD STATE %d FOR INODE I=%d\n",
114			    statemap[inumber], inumber);
115		}
116	}
117}
118
119int
120pass4check(struct inodesc * idesc)
121{
122	register struct dups *dlp;
123	int ndblks, res = KEEPON;
124	daddr_t blkno = idesc->id_blkno;
125	SEGUSE *sup;
126	struct ubuf *bp;
127	int sn, doanyway = 0;
128
129	sn = dtosn(fs, blkno);
130	/* If preening, bmap is not valid for non-active segs */
131	if (preen && !(seg_table[sn].su_flags & SEGUSE_ACTIVE))
132		doanyway = 1;
133	for (ndblks = fragstofsb(fs, idesc->id_numfrags); ndblks > 0; blkno++, ndblks--) {
134		if (chkrange(blkno, 1)) {
135			res = SKIP;
136		} else if (testbmap(blkno) || doanyway) {
137			for (dlp = duplist; dlp; dlp = dlp->next) {
138				if (dlp->dup != blkno)
139					continue;
140				dlp->dup = duplist->dup;
141				dlp = duplist;
142				duplist = duplist->next;
143				free((char *) dlp);
144				break;
145			}
146			if (dlp == 0) {
147				clrbmap(blkno);
148				LFS_SEGENTRY(sup, fs, sn, bp);
149				sup->su_nbytes -= fsbtob(fs, 1);
150				VOP_BWRITE(bp);
151				seg_table[sn].su_nbytes -= fsbtob(fs, 1);
152				++fs->lfs_bfree;
153				n_blks--;
154			}
155		}
156	}
157	return (res);
158}
159