1*da22d85dSmlelstv /* $NetBSD: setup.c,v 1.39 2019/03/31 13:16:52 mlelstv Exp $ */
2bf07c871Sagc
3bf07c871Sagc /*
4bf07c871Sagc * Copyright (c) 1980, 1986, 1993
5bf07c871Sagc * The Regents of the University of California. All rights reserved.
6bf07c871Sagc *
7bf07c871Sagc * Redistribution and use in source and binary forms, with or without
8bf07c871Sagc * modification, are permitted provided that the following conditions
9bf07c871Sagc * are met:
10bf07c871Sagc * 1. Redistributions of source code must retain the above copyright
11bf07c871Sagc * notice, this list of conditions and the following disclaimer.
12bf07c871Sagc * 2. Redistributions in binary form must reproduce the above copyright
13bf07c871Sagc * notice, this list of conditions and the following disclaimer in the
14bf07c871Sagc * documentation and/or other materials provided with the distribution.
15bf07c871Sagc * 3. Neither the name of the University nor the names of its contributors
16bf07c871Sagc * may be used to endorse or promote products derived from this software
17bf07c871Sagc * without specific prior written permission.
18bf07c871Sagc *
19bf07c871Sagc * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
20bf07c871Sagc * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21bf07c871Sagc * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22bf07c871Sagc * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
23bf07c871Sagc * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24bf07c871Sagc * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
25bf07c871Sagc * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26bf07c871Sagc * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27bf07c871Sagc * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28bf07c871Sagc * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29bf07c871Sagc * SUCH DAMAGE.
30bf07c871Sagc */
318f7c2b37Sbouyer
328f7c2b37Sbouyer /*
338f7c2b37Sbouyer * Copyright (c) 1997 Manuel Bouyer.
348f7c2b37Sbouyer *
358f7c2b37Sbouyer * Redistribution and use in source and binary forms, with or without
368f7c2b37Sbouyer * modification, are permitted provided that the following conditions
378f7c2b37Sbouyer * are met:
388f7c2b37Sbouyer * 1. Redistributions of source code must retain the above copyright
398f7c2b37Sbouyer * notice, this list of conditions and the following disclaimer.
408f7c2b37Sbouyer * 2. Redistributions in binary form must reproduce the above copyright
418f7c2b37Sbouyer * notice, this list of conditions and the following disclaimer in the
428f7c2b37Sbouyer * documentation and/or other materials provided with the distribution.
438f7c2b37Sbouyer *
442f853da9Sbouyer * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
452f853da9Sbouyer * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
462f853da9Sbouyer * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
472f853da9Sbouyer * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
482f853da9Sbouyer * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
492f853da9Sbouyer * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
502f853da9Sbouyer * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
512f853da9Sbouyer * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
522f853da9Sbouyer * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
532f853da9Sbouyer * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
548f7c2b37Sbouyer */
558f7c2b37Sbouyer
564b836889Slukem #include <sys/cdefs.h>
578f7c2b37Sbouyer #ifndef lint
588f7c2b37Sbouyer #if 0
598f7c2b37Sbouyer static char sccsid[] = "@(#)setup.c 8.5 (Berkeley) 11/23/94";
608f7c2b37Sbouyer #else
61*da22d85dSmlelstv __RCSID("$NetBSD: setup.c,v 1.39 2019/03/31 13:16:52 mlelstv Exp $");
628f7c2b37Sbouyer #endif
638f7c2b37Sbouyer #endif /* not lint */
648f7c2b37Sbouyer
657b57bf6dSchristos #define FSTYPENAMES
668f7c2b37Sbouyer #include <sys/param.h>
678f7c2b37Sbouyer #include <sys/time.h>
68bbddb141Smlelstv #include <sys/bitops.h>
698f7c2b37Sbouyer #include <ufs/ext2fs/ext2fs_dinode.h>
708f7c2b37Sbouyer #include <ufs/ext2fs/ext2fs.h>
718f7c2b37Sbouyer #include <sys/stat.h>
728f7c2b37Sbouyer #include <sys/ioctl.h>
7325aaa28eSmlelstv #include <sys/disk.h>
748f7c2b37Sbouyer #include <sys/file.h>
758f7c2b37Sbouyer
768f7c2b37Sbouyer #include <errno.h>
778f7c2b37Sbouyer #include <stdio.h>
788f7c2b37Sbouyer #include <stdlib.h>
798f7c2b37Sbouyer #include <string.h>
808f7c2b37Sbouyer #include <ctype.h>
818f7c2b37Sbouyer
82a5764f71Sjdolecek #include <util.h>
83a5764f71Sjdolecek
848f7c2b37Sbouyer #include "fsck.h"
858f7c2b37Sbouyer #include "extern.h"
868f7c2b37Sbouyer #include "fsutil.h"
8725aaa28eSmlelstv #include "partutil.h"
88481ad7b0Slukem #include "exitvalues.h"
898f7c2b37Sbouyer
9097880e96Schristos void badsb(int, const char *);
91ccde05f0Sxtraeme int calcsb(const char *, int, struct m_ext2fs *);
92ccde05f0Sxtraeme static int readsb(int);
938f7c2b37Sbouyer
9425aaa28eSmlelstv /*
9525aaa28eSmlelstv * For file systems smaller than SMALL_FSSIZE we use the S_DFL_* defaults,
9625aaa28eSmlelstv * otherwise if less than MEDIUM_FSSIZE use M_DFL_*, otherwise use
9725aaa28eSmlelstv * L_DFL_*.
9825aaa28eSmlelstv */
9925aaa28eSmlelstv #define SMALL_FSSIZE ((4 * 1024 * 1024) / secsize) /* 4MB */
10025aaa28eSmlelstv #define S_DFL_BSIZE 1024
10125aaa28eSmlelstv #define MEDIUM_FSSIZE ((512 * 1024 * 1024) / secsize) /* 512MB */
10225aaa28eSmlelstv #define M_DFL_BSIZE 1024
10325aaa28eSmlelstv #define L_DFL_BSIZE 4096
10425aaa28eSmlelstv
1058f7c2b37Sbouyer int
setup(const char * dev)106ccde05f0Sxtraeme setup(const char *dev)
1078f7c2b37Sbouyer {
1084b836889Slukem long cg, asked, i;
1098f7c2b37Sbouyer long bmapsize;
11025aaa28eSmlelstv struct disk_geom geo;
11125aaa28eSmlelstv struct dkwedge_info dkw;
1128f7c2b37Sbouyer off_t sizepb;
1138f7c2b37Sbouyer struct stat statb;
1148f7c2b37Sbouyer struct m_ext2fs proto;
1158f7c2b37Sbouyer int doskipclean;
1168f7c2b37Sbouyer u_int64_t maxfilesize;
1178f7c2b37Sbouyer
1188f7c2b37Sbouyer havesb = 0;
1198f7c2b37Sbouyer fswritefd = -1;
1208f7c2b37Sbouyer doskipclean = skipclean;
1218f7c2b37Sbouyer if (stat(dev, &statb) < 0) {
1228f7c2b37Sbouyer printf("Can't stat %s: %s\n", dev, strerror(errno));
12337d769edStsutsui return 0;
1248f7c2b37Sbouyer }
1258f7c2b37Sbouyer if (!S_ISCHR(statb.st_mode)) {
1268f7c2b37Sbouyer pfatal("%s is not a character device", dev);
1278f7c2b37Sbouyer if (reply("CONTINUE") == 0)
12837d769edStsutsui return 0;
1298f7c2b37Sbouyer }
1308f7c2b37Sbouyer if ((fsreadfd = open(dev, O_RDONLY)) < 0) {
1318f7c2b37Sbouyer printf("Can't open %s: %s\n", dev, strerror(errno));
13237d769edStsutsui return 0;
1338f7c2b37Sbouyer }
1348f7c2b37Sbouyer if (preen == 0)
1358f7c2b37Sbouyer printf("** %s", dev);
1368f7c2b37Sbouyer if (nflag || (fswritefd = open(dev, O_WRONLY)) < 0) {
1378f7c2b37Sbouyer fswritefd = -1;
1388f7c2b37Sbouyer if (preen)
1398f7c2b37Sbouyer pfatal("NO WRITE ACCESS");
1408f7c2b37Sbouyer printf(" (NO WRITE)");
1418f7c2b37Sbouyer }
1428f7c2b37Sbouyer if (preen == 0)
1438f7c2b37Sbouyer printf("\n");
1448f7c2b37Sbouyer fsmodified = 0;
1458f7c2b37Sbouyer lfdir = 0;
1468f7c2b37Sbouyer initbarea(&sblk);
1478f7c2b37Sbouyer initbarea(&asblk);
1487052d78bSbouyer sblk.b_un.b_buf = malloc(SBSIZE);
1497052d78bSbouyer asblk.b_un.b_buf = malloc(SBSIZE);
1508f7c2b37Sbouyer if (sblk.b_un.b_buf == NULL || asblk.b_un.b_buf == NULL)
151481ad7b0Slukem errexit("cannot allocate space for superblock");
15225aaa28eSmlelstv if (getdiskinfo(dev, fsreadfd, NULL, &geo, &dkw) != -1)
15325aaa28eSmlelstv dev_bsize = secsize = geo.dg_secsize;
1548f7c2b37Sbouyer else
1558f7c2b37Sbouyer dev_bsize = secsize = DEV_BSIZE;
1568f7c2b37Sbouyer /*
1578f7c2b37Sbouyer * Read in the superblock, looking for alternates if necessary
1588f7c2b37Sbouyer */
1598f7c2b37Sbouyer if (readsb(1) == 0) {
1608f7c2b37Sbouyer if (bflag || preen || calcsb(dev, fsreadfd, &proto) == 0)
16137d769edStsutsui return 0;
1628f7c2b37Sbouyer if (reply("LOOK FOR ALTERNATE SUPERBLOCKS") == 0)
16337d769edStsutsui return 0;
1648f7c2b37Sbouyer for (cg = 1; cg < proto.e2fs_ncg; cg++) {
1652737439dSdholland bflag = EXT2_FSBTODB(&proto,
16637d769edStsutsui cg * proto.e2fs.e2fs_bpg +
16737d769edStsutsui proto.e2fs.e2fs_first_dblock);
1688f7c2b37Sbouyer if (readsb(0) != 0)
1698f7c2b37Sbouyer break;
1708f7c2b37Sbouyer }
1718f7c2b37Sbouyer if (cg >= proto.e2fs_ncg) {
1728f7c2b37Sbouyer printf("%s %s\n%s %s\n%s %s\n",
1738f7c2b37Sbouyer "SEARCH FOR ALTERNATE SUPER-BLOCK",
1748f7c2b37Sbouyer "FAILED. YOU MUST USE THE",
1758f7c2b37Sbouyer "-b OPTION TO FSCK_FFS TO SPECIFY THE",
1768f7c2b37Sbouyer "LOCATION OF AN ALTERNATE",
1778f7c2b37Sbouyer "SUPER-BLOCK TO SUPPLY NEEDED",
1788f7c2b37Sbouyer "INFORMATION; SEE fsck_ext2fs(8).");
17937d769edStsutsui return 0;
1808f7c2b37Sbouyer }
1818f7c2b37Sbouyer doskipclean = 0;
1828f7c2b37Sbouyer pwarn("USING ALTERNATE SUPERBLOCK AT %d\n", bflag);
1838f7c2b37Sbouyer }
1848f7c2b37Sbouyer if (debug)
1858f7c2b37Sbouyer printf("state = %d\n", sblock.e2fs.e2fs_state);
1868f7c2b37Sbouyer if (sblock.e2fs.e2fs_state == E2FS_ISCLEAN) {
1878f7c2b37Sbouyer if (doskipclean) {
1888f7c2b37Sbouyer pwarn("%sile system is clean; not checking\n",
1898f7c2b37Sbouyer preen ? "f" : "** F");
19037d769edStsutsui return -1;
1918f7c2b37Sbouyer }
1928f7c2b37Sbouyer if (!preen)
1938f7c2b37Sbouyer pwarn("** File system is already clean\n");
1948f7c2b37Sbouyer }
1958f7c2b37Sbouyer maxfsblock = sblock.e2fs.e2fs_bcount;
1968f7c2b37Sbouyer maxino = sblock.e2fs_ncg * sblock.e2fs.e2fs_ipg;
1978f7c2b37Sbouyer sizepb = sblock.e2fs_bsize;
198dcd34a91Sdholland maxfilesize = sblock.e2fs_bsize * EXT2FS_NDADDR - 1;
199dcd34a91Sdholland for (i = 0; i < EXT2FS_NIADDR; i++) {
200f1333577Sdholland sizepb *= EXT2_NINDIR(&sblock);
2018f7c2b37Sbouyer maxfilesize += sizepb;
2028f7c2b37Sbouyer }
2038f7c2b37Sbouyer /*
2048f7c2b37Sbouyer * Check and potentially fix certain fields in the super block.
2058f7c2b37Sbouyer */
20668475dcbSchristos if (/* (sblock.e2fs.e2fs_rbcount < 0) || */
2078f7c2b37Sbouyer (sblock.e2fs.e2fs_rbcount > sblock.e2fs.e2fs_bcount)) {
2088f7c2b37Sbouyer pfatal("IMPOSSIBLE RESERVED BLOCK COUNT=%d IN SUPERBLOCK",
2098f7c2b37Sbouyer sblock.e2fs.e2fs_rbcount);
2108f7c2b37Sbouyer if (reply("SET TO DEFAULT") == 1) {
21137d769edStsutsui sblock.e2fs.e2fs_rbcount =
21237d769edStsutsui sblock.e2fs.e2fs_bcount * MINFREE / 100;
2138f7c2b37Sbouyer sbdirty();
2147052d78bSbouyer dirty(&asblk);
2158f7c2b37Sbouyer }
2168f7c2b37Sbouyer }
2178f7c2b37Sbouyer if (sblock.e2fs.e2fs_bpg != sblock.e2fs.e2fs_fpg) {
2188f7c2b37Sbouyer pfatal("WRONG FPG=%d (BPG=%d) IN SUPERBLOCK",
2198f7c2b37Sbouyer sblock.e2fs.e2fs_fpg, sblock.e2fs.e2fs_bpg);
2208f7c2b37Sbouyer return 0;
2218f7c2b37Sbouyer }
2228f7c2b37Sbouyer if (asblk.b_dirty && !bflag) {
2237052d78bSbouyer copyback_sb(&asblk);
2248f7c2b37Sbouyer flush(fswritefd, &asblk);
2258f7c2b37Sbouyer }
2268f7c2b37Sbouyer /*
2278f7c2b37Sbouyer * read in the summary info.
2288f7c2b37Sbouyer */
2298f7c2b37Sbouyer
2308f7c2b37Sbouyer sblock.e2fs_gd = malloc(sblock.e2fs_ngdb * sblock.e2fs_bsize);
231c3d84360Sbouyer if (sblock.e2fs_gd == NULL)
232481ad7b0Slukem errexit("out of memory");
2338f7c2b37Sbouyer asked = 0;
2348f7c2b37Sbouyer for (i = 0; i < sblock.e2fs_ngdb; i++) {
23537d769edStsutsui if (bread(fsreadfd,
23637d769edStsutsui (char *)&sblock.e2fs_gd[i * sblock.e2fs_bsize /
23737d769edStsutsui sizeof(struct ext2_gd)],
2382737439dSdholland EXT2_FSBTODB(&sblock, ((sblock.e2fs_bsize > 1024) ? 0 : 1) +
23937d769edStsutsui i + 1),
2408f7c2b37Sbouyer sblock.e2fs_bsize) != 0 && !asked) {
2418f7c2b37Sbouyer pfatal("BAD SUMMARY INFORMATION");
2428f7c2b37Sbouyer if (reply("CONTINUE") == 0)
243481ad7b0Slukem exit(FSCK_EXIT_CHECK_FAILED);
2448f7c2b37Sbouyer asked++;
2458f7c2b37Sbouyer }
2468f7c2b37Sbouyer }
2478f7c2b37Sbouyer /*
2488f7c2b37Sbouyer * allocate and initialize the necessary maps
2498f7c2b37Sbouyer */
2508f7c2b37Sbouyer bmapsize = roundup(howmany(maxfsblock, NBBY), sizeof(int16_t));
25137d769edStsutsui blockmap = calloc((unsigned int)bmapsize, sizeof(char));
2528f7c2b37Sbouyer if (blockmap == NULL) {
2538f7c2b37Sbouyer printf("cannot alloc %u bytes for blockmap\n",
25437d769edStsutsui (unsigned int)bmapsize);
2558f7c2b37Sbouyer goto badsblabel;
2568f7c2b37Sbouyer }
25737d769edStsutsui statemap = calloc((unsigned int)(maxino + 2), sizeof(char));
2588f7c2b37Sbouyer if (statemap == NULL) {
2598f7c2b37Sbouyer printf("cannot alloc %u bytes for statemap\n",
26037d769edStsutsui (unsigned int)(maxino + 1));
2618f7c2b37Sbouyer goto badsblabel;
2628f7c2b37Sbouyer }
26337d769edStsutsui typemap = calloc((unsigned int)(maxino + 1), sizeof(char));
2645fb6bc4eSbouyer if (typemap == NULL) {
2655fb6bc4eSbouyer printf("cannot alloc %u bytes for typemap\n",
26637d769edStsutsui (unsigned int)(maxino + 1));
2675fb6bc4eSbouyer goto badsblabel;
2685fb6bc4eSbouyer }
26937d769edStsutsui lncntp = calloc((unsigned)(maxino + 1), sizeof(int16_t));
2708f7c2b37Sbouyer if (lncntp == NULL) {
2718f7c2b37Sbouyer printf("cannot alloc %u bytes for lncntp\n",
27237d769edStsutsui (unsigned int)((maxino + 1) * sizeof(int16_t)));
2738f7c2b37Sbouyer goto badsblabel;
2748f7c2b37Sbouyer }
2758f7c2b37Sbouyer for (numdirs = 0, cg = 0; cg < sblock.e2fs_ncg; cg++) {
2767052d78bSbouyer numdirs += fs2h16(sblock.e2fs_gd[cg].ext2bgd_ndirs);
2778f7c2b37Sbouyer }
2788f7c2b37Sbouyer inplast = 0;
2798f7c2b37Sbouyer listmax = numdirs + 10;
28037d769edStsutsui inpsort = calloc((unsigned int)listmax, sizeof(struct inoinfo *));
28137d769edStsutsui inphead = calloc((unsigned int)numdirs, sizeof(struct inoinfo *));
2828f7c2b37Sbouyer if (inpsort == NULL || inphead == NULL) {
2838f7c2b37Sbouyer printf("cannot alloc %u bytes for inphead\n",
28437d769edStsutsui (unsigned int)(numdirs * sizeof(struct inoinfo *)));
2858f7c2b37Sbouyer goto badsblabel;
2868f7c2b37Sbouyer }
2878f7c2b37Sbouyer bufinit();
28837d769edStsutsui return 1;
2898f7c2b37Sbouyer
2908f7c2b37Sbouyer badsblabel:
2918f7c2b37Sbouyer ckfini(0);
29237d769edStsutsui return 0;
2938f7c2b37Sbouyer }
2948f7c2b37Sbouyer
2958f7c2b37Sbouyer /*
2967052d78bSbouyer * Read in the super block and its summary info, convert to host byte order.
2978f7c2b37Sbouyer */
2988f7c2b37Sbouyer static int
readsb(int listerr)299ccde05f0Sxtraeme readsb(int listerr)
3008f7c2b37Sbouyer {
3018f7c2b37Sbouyer daddr_t super = bflag ? bflag : SBOFF / dev_bsize;
3028f7c2b37Sbouyer
3037052d78bSbouyer if (bread(fsreadfd, (char *)sblk.b_un.b_fs, super, (long)SBSIZE) != 0)
30437d769edStsutsui return 0;
3058f7c2b37Sbouyer sblk.b_bno = super;
3068f7c2b37Sbouyer sblk.b_size = SBSIZE;
3077052d78bSbouyer
3087052d78bSbouyer /* Copy the superblock in memory */
30909d4663fSbouyer e2fs_sbload(sblk.b_un.b_fs, &sblock.e2fs);
3107052d78bSbouyer
3118f7c2b37Sbouyer /*
3128f7c2b37Sbouyer * run a few consistency checks of the super block
3138f7c2b37Sbouyer */
3148f7c2b37Sbouyer if (sblock.e2fs.e2fs_magic != E2FS_MAGIC) {
31537d769edStsutsui badsb(listerr, "MAGIC NUMBER WRONG");
31637d769edStsutsui return 0;
3178f7c2b37Sbouyer }
3188f7c2b37Sbouyer if (sblock.e2fs.e2fs_log_bsize > 2) {
31937d769edStsutsui badsb(listerr, "BAD LOG_BSIZE");
32037d769edStsutsui return 0;
3218f7c2b37Sbouyer }
322f5925335Stsutsui if (sblock.e2fs.e2fs_rev > E2FS_REV0 &&
323f5925335Stsutsui (!powerof2(sblock.e2fs.e2fs_inode_size) ||
324b720c3e2Sjdolecek sblock.e2fs.e2fs_inode_size < EXT2_REV0_DINODE_SIZE ||
325f5925335Stsutsui sblock.e2fs.e2fs_inode_size >
326f5925335Stsutsui (1024 << sblock.e2fs.e2fs_log_bsize))) {
327f5925335Stsutsui badsb(listerr, "BAD INODE_SIZE");
328f5925335Stsutsui return 0;
329f5925335Stsutsui }
3308f7c2b37Sbouyer
3318f7c2b37Sbouyer /* compute the dynamic fields of the in-memory sb */
3328f7c2b37Sbouyer /* compute dynamic sb infos */
3338f7c2b37Sbouyer sblock.e2fs_ncg =
3348f7c2b37Sbouyer howmany(sblock.e2fs.e2fs_bcount - sblock.e2fs.e2fs_first_dblock,
3358f7c2b37Sbouyer sblock.e2fs.e2fs_bpg);
336bbddb141Smlelstv sblock.e2fs_fsbtodb = sblock.e2fs.e2fs_log_bsize + ilog2(1024 / dev_bsize);
3378f7c2b37Sbouyer sblock.e2fs_bsize = 1024 << sblock.e2fs.e2fs_log_bsize;
3388f7c2b37Sbouyer sblock.e2fs_bshift = LOG_MINBSIZE + sblock.e2fs.e2fs_log_bsize;
3398f7c2b37Sbouyer sblock.e2fs_qbmask = sblock.e2fs_bsize - 1;
3408f7c2b37Sbouyer sblock.e2fs_bmask = ~sblock.e2fs_qbmask;
3418f7c2b37Sbouyer sblock.e2fs_ngdb = howmany(sblock.e2fs_ncg,
3428f7c2b37Sbouyer sblock.e2fs_bsize / sizeof(struct ext2_gd));
343f5925335Stsutsui sblock.e2fs_ipb = sblock.e2fs_bsize / EXT2_DINODE_SIZE(&sblock);
344672990c3Stsutsui sblock.e2fs_itpg = howmany(sblock.e2fs.e2fs_ipg, sblock.e2fs_ipb);
3458f7c2b37Sbouyer
3468f7c2b37Sbouyer /*
3478f7c2b37Sbouyer * Compute block size that the filesystem is based on,
3488f7c2b37Sbouyer * according to fsbtodb, and adjust superblock block number
3498f7c2b37Sbouyer * so we can tell if this is an alternate later.
3508f7c2b37Sbouyer */
3518f7c2b37Sbouyer super *= dev_bsize;
3522737439dSdholland dev_bsize = sblock.e2fs_bsize / EXT2_FSBTODB(&sblock, 1);
3538f7c2b37Sbouyer sblk.b_bno = super / dev_bsize;
3547052d78bSbouyer
355672990c3Stsutsui if (sblock.e2fs_ncg == 1) {
356672990c3Stsutsui /* no alternate superblock; assume it's okay */
357672990c3Stsutsui havesb = 1;
35837d769edStsutsui return 1;
359672990c3Stsutsui }
3607052d78bSbouyer getblk(&asblk, 1 * sblock.e2fs.e2fs_bpg + sblock.e2fs.e2fs_first_dblock,
3617052d78bSbouyer (long)SBSIZE);
3627052d78bSbouyer if (asblk.b_errs)
36337d769edStsutsui return 0;
3648f7c2b37Sbouyer if (bflag) {
3658f7c2b37Sbouyer havesb = 1;
36637d769edStsutsui return 1;
3678f7c2b37Sbouyer }
3688f7c2b37Sbouyer
3698f7c2b37Sbouyer /*
3708f7c2b37Sbouyer * Set all possible fields that could differ, then do check
3718f7c2b37Sbouyer * of whole super block against an alternate super block.
3728f7c2b37Sbouyer * When an alternate super-block is specified this check is skipped.
3738f7c2b37Sbouyer */
3747052d78bSbouyer asblk.b_un.b_fs->e2fs_rbcount = sblk.b_un.b_fs->e2fs_rbcount;
3757052d78bSbouyer asblk.b_un.b_fs->e2fs_fbcount = sblk.b_un.b_fs->e2fs_fbcount;
3767052d78bSbouyer asblk.b_un.b_fs->e2fs_ficount = sblk.b_un.b_fs->e2fs_ficount;
3777052d78bSbouyer asblk.b_un.b_fs->e2fs_mtime = sblk.b_un.b_fs->e2fs_mtime;
3787052d78bSbouyer asblk.b_un.b_fs->e2fs_wtime = sblk.b_un.b_fs->e2fs_wtime;
3797052d78bSbouyer asblk.b_un.b_fs->e2fs_mnt_count = sblk.b_un.b_fs->e2fs_mnt_count;
38037d769edStsutsui asblk.b_un.b_fs->e2fs_max_mnt_count =
38137d769edStsutsui sblk.b_un.b_fs->e2fs_max_mnt_count;
3827052d78bSbouyer asblk.b_un.b_fs->e2fs_state = sblk.b_un.b_fs->e2fs_state;
3837052d78bSbouyer asblk.b_un.b_fs->e2fs_beh = sblk.b_un.b_fs->e2fs_beh;
3847052d78bSbouyer asblk.b_un.b_fs->e2fs_lastfsck = sblk.b_un.b_fs->e2fs_lastfsck;
3857052d78bSbouyer asblk.b_un.b_fs->e2fs_fsckintv = sblk.b_un.b_fs->e2fs_fsckintv;
3867052d78bSbouyer asblk.b_un.b_fs->e2fs_ruid = sblk.b_un.b_fs->e2fs_ruid;
3877052d78bSbouyer asblk.b_un.b_fs->e2fs_rgid = sblk.b_un.b_fs->e2fs_rgid;
3881bb2b4ddSbouyer asblk.b_un.b_fs->e2fs_block_group_nr =
3891bb2b4ddSbouyer sblk.b_un.b_fs->e2fs_block_group_nr;
390d53c382dSws asblk.b_un.b_fs->e2fs_features_rocompat &= ~EXT2F_ROCOMPAT_LARGEFILE;
391d53c382dSws asblk.b_un.b_fs->e2fs_features_rocompat |=
392d53c382dSws sblk.b_un.b_fs->e2fs_features_rocompat & EXT2F_ROCOMPAT_LARGEFILE;
393bbddb141Smlelstv memcpy(asblk.b_un.b_fs->e2fs_fsmnt, sblk.b_un.b_fs->e2fs_fsmnt,
394bbddb141Smlelstv sizeof(asblk.b_un.b_fs->e2fs_fsmnt));
395*da22d85dSmlelstv asblk.b_un.b_fs->e4fs_kbytes_written =
396*da22d85dSmlelstv sblk.b_un.b_fs->e4fs_kbytes_written;
39709d4663fSbouyer if (sblock.e2fs.e2fs_rev > E2FS_REV0 &&
398d2d8cddeSjdolecek ((sblock.e2fs.e2fs_features_incompat & ~EXT2F_INCOMPAT_SUPP_FSCK) ||
399d2d8cddeSjdolecek (sblock.e2fs.e2fs_features_rocompat & ~EXT2F_ROCOMPAT_SUPP_FSCK))) {
4001bb2b4ddSbouyer if (debug) {
4011bb2b4ddSbouyer printf("compat 0x%08x, incompat 0x%08x, compat_ro "
4021bb2b4ddSbouyer "0x%08x\n",
40309d4663fSbouyer sblock.e2fs.e2fs_features_compat,
40409d4663fSbouyer sblock.e2fs.e2fs_features_incompat,
40509d4663fSbouyer sblock.e2fs.e2fs_features_rocompat);
406a5764f71Sjdolecek
407a5764f71Sjdolecek if ((sblock.e2fs.e2fs_features_rocompat & ~EXT2F_ROCOMPAT_SUPP_FSCK)) {
408a5764f71Sjdolecek char buf[512];
409a5764f71Sjdolecek
410a5764f71Sjdolecek snprintb(buf, sizeof(buf), EXT2F_ROCOMPAT_BITS,
411a5764f71Sjdolecek sblock.e2fs.e2fs_features_rocompat & ~EXT2F_ROCOMPAT_SUPP_FSCK);
412a5764f71Sjdolecek printf("unsupported rocompat features: %s\n", buf);
413a5764f71Sjdolecek }
414a5764f71Sjdolecek if ((sblock.e2fs.e2fs_features_incompat & ~EXT2F_INCOMPAT_SUPP_FSCK)) {
415a5764f71Sjdolecek char buf[512];
416a5764f71Sjdolecek
417a5764f71Sjdolecek snprintb(buf, sizeof(buf), EXT2F_INCOMPAT_BITS,
418a5764f71Sjdolecek sblock.e2fs.e2fs_features_incompat & ~EXT2F_INCOMPAT_SUPP_FSCK);
419a5764f71Sjdolecek printf("unsupported incompat features: %s\n", buf);
420a5764f71Sjdolecek }
4211bb2b4ddSbouyer }
4225fb6bc4eSbouyer badsb(listerr, "INCOMPATIBLE FEATURE BITS IN SUPER BLOCK");
4231bb2b4ddSbouyer return 0;
4241bb2b4ddSbouyer }
4257052d78bSbouyer if (memcmp(sblk.b_un.b_fs, asblk.b_un.b_fs, SBSIZE)) {
4268f7c2b37Sbouyer if (debug) {
4277052d78bSbouyer u_int32_t *nlp, *olp, *endlp;
4288f7c2b37Sbouyer
4298f7c2b37Sbouyer printf("superblock mismatches\n");
4307052d78bSbouyer nlp = (u_int32_t *)asblk.b_un.b_fs;
4317052d78bSbouyer olp = (u_int32_t *)sblk.b_un.b_fs;
43237d769edStsutsui endlp = olp + (SBSIZE / sizeof(*olp));
4338f7c2b37Sbouyer for ( ; olp < endlp; olp++, nlp++) {
4348f7c2b37Sbouyer if (*olp == *nlp)
4358f7c2b37Sbouyer continue;
43637d769edStsutsui printf("offset %ld, original %ld, "
43737d769edStsutsui "alternate %ld\n",
4387052d78bSbouyer (long)(olp - (u_int32_t *)sblk.b_un.b_fs),
4397052d78bSbouyer (long)fs2h32(*olp),
4407052d78bSbouyer (long)fs2h32(*nlp));
4418f7c2b37Sbouyer }
4428f7c2b37Sbouyer }
4438f7c2b37Sbouyer badsb(listerr,
44437d769edStsutsui "VALUES IN SUPER BLOCK DISAGREE WITH "
44537d769edStsutsui "THOSE IN FIRST ALTERNATE");
44637d769edStsutsui return 0;
4478f7c2b37Sbouyer }
4488f7c2b37Sbouyer havesb = 1;
44937d769edStsutsui return 1;
4508f7c2b37Sbouyer }
4518f7c2b37Sbouyer
4528f7c2b37Sbouyer void
copyback_sb(struct bufarea * bp)453ccde05f0Sxtraeme copyback_sb(struct bufarea *bp)
4547052d78bSbouyer {
4557052d78bSbouyer /* Copy the in-memory superblock back to buffer */
45668fb2791Stsutsui bp->b_un.b_fs->e2fs_icount = h2fs32(sblock.e2fs.e2fs_icount);
45768fb2791Stsutsui bp->b_un.b_fs->e2fs_bcount = h2fs32(sblock.e2fs.e2fs_bcount);
45868fb2791Stsutsui bp->b_un.b_fs->e2fs_rbcount = h2fs32(sblock.e2fs.e2fs_rbcount);
45968fb2791Stsutsui bp->b_un.b_fs->e2fs_fbcount = h2fs32(sblock.e2fs.e2fs_fbcount);
46068fb2791Stsutsui bp->b_un.b_fs->e2fs_ficount = h2fs32(sblock.e2fs.e2fs_ficount);
4617052d78bSbouyer bp->b_un.b_fs->e2fs_first_dblock =
46268fb2791Stsutsui h2fs32(sblock.e2fs.e2fs_first_dblock);
46368fb2791Stsutsui bp->b_un.b_fs->e2fs_log_bsize = h2fs32(sblock.e2fs.e2fs_log_bsize);
46468fb2791Stsutsui bp->b_un.b_fs->e2fs_fsize = h2fs32(sblock.e2fs.e2fs_fsize);
46568fb2791Stsutsui bp->b_un.b_fs->e2fs_bpg = h2fs32(sblock.e2fs.e2fs_bpg);
46668fb2791Stsutsui bp->b_un.b_fs->e2fs_fpg = h2fs32(sblock.e2fs.e2fs_fpg);
46768fb2791Stsutsui bp->b_un.b_fs->e2fs_ipg = h2fs32(sblock.e2fs.e2fs_ipg);
46868fb2791Stsutsui bp->b_un.b_fs->e2fs_mtime = h2fs32(sblock.e2fs.e2fs_mtime);
46968fb2791Stsutsui bp->b_un.b_fs->e2fs_wtime = h2fs32(sblock.e2fs.e2fs_wtime);
47068fb2791Stsutsui bp->b_un.b_fs->e2fs_lastfsck = h2fs32(sblock.e2fs.e2fs_lastfsck);
47168fb2791Stsutsui bp->b_un.b_fs->e2fs_fsckintv = h2fs32(sblock.e2fs.e2fs_fsckintv);
47268fb2791Stsutsui bp->b_un.b_fs->e2fs_creator = h2fs32(sblock.e2fs.e2fs_creator);
47368fb2791Stsutsui bp->b_un.b_fs->e2fs_rev = h2fs32(sblock.e2fs.e2fs_rev);
47468fb2791Stsutsui bp->b_un.b_fs->e2fs_mnt_count = h2fs16(sblock.e2fs.e2fs_mnt_count);
4757052d78bSbouyer bp->b_un.b_fs->e2fs_max_mnt_count =
47668fb2791Stsutsui h2fs16(sblock.e2fs.e2fs_max_mnt_count);
47768fb2791Stsutsui bp->b_un.b_fs->e2fs_magic = h2fs16(sblock.e2fs.e2fs_magic);
47868fb2791Stsutsui bp->b_un.b_fs->e2fs_state = h2fs16(sblock.e2fs.e2fs_state);
47968fb2791Stsutsui bp->b_un.b_fs->e2fs_beh = h2fs16(sblock.e2fs.e2fs_beh);
48068fb2791Stsutsui bp->b_un.b_fs->e2fs_ruid = h2fs16(sblock.e2fs.e2fs_ruid);
48168fb2791Stsutsui bp->b_un.b_fs->e2fs_rgid = h2fs16(sblock.e2fs.e2fs_rgid);
4827052d78bSbouyer }
4837052d78bSbouyer
4847052d78bSbouyer void
badsb(int listerr,const char * s)48597880e96Schristos badsb(int listerr, const char *s)
4868f7c2b37Sbouyer {
4878f7c2b37Sbouyer
4888f7c2b37Sbouyer if (!listerr)
4898f7c2b37Sbouyer return;
4908f7c2b37Sbouyer if (preen)
4918f7c2b37Sbouyer printf("%s: ", cdevname());
4928f7c2b37Sbouyer pfatal("BAD SUPER BLOCK: %s\n", s);
4938f7c2b37Sbouyer }
4948f7c2b37Sbouyer
4958f7c2b37Sbouyer /*
4968f7c2b37Sbouyer * Calculate a prototype superblock based on information in the disk label.
4978f7c2b37Sbouyer * When done the cgsblock macro can be calculated and the fs_ncg field
4988f7c2b37Sbouyer * can be used. Do NOT attempt to use other macros without verifying that
4998f7c2b37Sbouyer * their needed information is available!
5008f7c2b37Sbouyer */
5018f7c2b37Sbouyer
5028f7c2b37Sbouyer int
calcsb(const char * dev,int devfd,struct m_ext2fs * fs)503ccde05f0Sxtraeme calcsb(const char *dev, int devfd, struct m_ext2fs *fs)
5048f7c2b37Sbouyer {
50525aaa28eSmlelstv struct dkwedge_info dkw;
50625aaa28eSmlelstv struct disk_geom geo;
5078f7c2b37Sbouyer
50825aaa28eSmlelstv if (getdiskinfo(dev, devfd, NULL, &geo, &dkw) == -1)
50925aaa28eSmlelstv pfatal("%s: CANNOT FIGURE OUT FILE SYSTEM PARTITION\n", dev);
51025aaa28eSmlelstv if (dkw.dkw_parent[0] == '\0') {
5118f7c2b37Sbouyer pfatal("%s: CANNOT FIGURE OUT FILE SYSTEM PARTITION\n", dev);
51237d769edStsutsui return 0;
5138f7c2b37Sbouyer }
51425aaa28eSmlelstv
5158f7c2b37Sbouyer memset(fs, 0, sizeof(struct m_ext2fs));
51625aaa28eSmlelstv
51725aaa28eSmlelstv if (dkw.dkw_size < (uint64_t)SMALL_FSSIZE)
51825aaa28eSmlelstv fs->e2fs_bsize = S_DFL_BSIZE;
51925aaa28eSmlelstv else if (dkw.dkw_size < (uint64_t)MEDIUM_FSSIZE)
52025aaa28eSmlelstv fs->e2fs_bsize = M_DFL_BSIZE;
52125aaa28eSmlelstv else
52225aaa28eSmlelstv fs->e2fs_bsize = L_DFL_BSIZE;
52325aaa28eSmlelstv
52425aaa28eSmlelstv fs->e2fs.e2fs_log_bsize = ilog2(fs->e2fs_bsize / 1024);
52525aaa28eSmlelstv fs->e2fs.e2fs_bcount = (fs->e2fs_bsize * DEV_BSIZE) / fs->e2fs_bsize;
5267052d78bSbouyer fs->e2fs.e2fs_first_dblock = (fs->e2fs.e2fs_log_bsize == 0) ? 1 : 0;
5278f7c2b37Sbouyer fs->e2fs.e2fs_bpg = fs->e2fs_bsize * NBBY;
5288f7c2b37Sbouyer fs->e2fs_bshift = LOG_MINBSIZE + fs->e2fs.e2fs_log_bsize;
5298f7c2b37Sbouyer fs->e2fs_qbmask = fs->e2fs_bsize - 1;
5308f7c2b37Sbouyer fs->e2fs_bmask = ~fs->e2fs_qbmask;
5318f7c2b37Sbouyer fs->e2fs_ncg =
5328f7c2b37Sbouyer howmany(fs->e2fs.e2fs_bcount - fs->e2fs.e2fs_first_dblock,
5338f7c2b37Sbouyer fs->e2fs.e2fs_bpg);
5348f7c2b37Sbouyer fs->e2fs_fsbtodb = fs->e2fs.e2fs_log_bsize + 1;
5358f7c2b37Sbouyer fs->e2fs_ngdb = howmany(fs->e2fs_ncg,
5368f7c2b37Sbouyer fs->e2fs_bsize / sizeof(struct ext2_gd));
5378f7c2b37Sbouyer
53837d769edStsutsui return 1;
5398f7c2b37Sbouyer }
5408f7c2b37Sbouyer
54109d4663fSbouyer daddr_t
cgoverhead(int c)542ccde05f0Sxtraeme cgoverhead(int c)
54309d4663fSbouyer {
54409d4663fSbouyer int overh;
54537d769edStsutsui overh =
54637d769edStsutsui 1 /* block bitmap */ +
54709d4663fSbouyer 1 /* inode bitmap */ +
54809d4663fSbouyer sblock.e2fs_itpg;
54909d4663fSbouyer if (sblock.e2fs.e2fs_rev > E2FS_REV0 &&
55009d4663fSbouyer sblock.e2fs.e2fs_features_rocompat & EXT2F_ROCOMPAT_SPARSESUPER) {
55109d4663fSbouyer if (cg_has_sb(c) == 0)
55209d4663fSbouyer return overh;
55309d4663fSbouyer }
55437d769edStsutsui overh += 1 /* superblock */ + sblock.e2fs_ngdb;
55509d4663fSbouyer return overh;
55609d4663fSbouyer }
557