Home | History | Annotate | Line # | Download | only in installboot
ext2fs.c revision 1.6.12.2
      1  1.6.12.2      tls /*	$NetBSD: ext2fs.c,v 1.6.12.2 2013/06/23 06:29:04 tls Exp $	*/
      2       1.1  tsutsui 
      3       1.1  tsutsui /*
      4       1.1  tsutsui  * Copyright (c) 1997 Manuel Bouyer.
      5       1.1  tsutsui  *
      6       1.1  tsutsui  * Redistribution and use in source and binary forms, with or without
      7       1.1  tsutsui  * modification, are permitted provided that the following conditions
      8       1.1  tsutsui  * are met:
      9       1.1  tsutsui  * 1. Redistributions of source code must retain the above copyright
     10       1.1  tsutsui  *    notice, this list of conditions and the following disclaimer.
     11       1.1  tsutsui  * 2. Redistributions in binary form must reproduce the above copyright
     12       1.1  tsutsui  *    notice, this list of conditions and the following disclaimer in the
     13       1.1  tsutsui  *    documentation and/or other materials provided with the distribution.
     14       1.1  tsutsui  *
     15       1.1  tsutsui  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
     16       1.1  tsutsui  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
     17       1.1  tsutsui  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
     18       1.1  tsutsui  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
     19       1.1  tsutsui  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
     20       1.1  tsutsui  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
     21       1.1  tsutsui  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
     22       1.1  tsutsui  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     23       1.1  tsutsui  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
     24       1.1  tsutsui  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     25       1.1  tsutsui  */
     26       1.1  tsutsui 
     27       1.1  tsutsui /*-
     28       1.1  tsutsui  * Copyright (c) 2002 The NetBSD Foundation, Inc.
     29       1.1  tsutsui  * All rights reserved.
     30       1.1  tsutsui  *
     31       1.1  tsutsui  * This code is derived from software contributed to The NetBSD Foundation
     32       1.1  tsutsui  * by Matt Fredette.
     33       1.1  tsutsui  *
     34       1.1  tsutsui  * Redistribution and use in source and binary forms, with or without
     35       1.1  tsutsui  * modification, are permitted provided that the following conditions
     36       1.1  tsutsui  * are met:
     37       1.1  tsutsui  * 1. Redistributions of source code must retain the above copyright
     38       1.1  tsutsui  *    notice, this list of conditions and the following disclaimer.
     39       1.1  tsutsui  * 2. Redistributions in binary form must reproduce the above copyright
     40       1.1  tsutsui  *    notice, this list of conditions and the following disclaimer in the
     41       1.1  tsutsui  *    documentation and/or other materials provided with the distribution.
     42       1.1  tsutsui  *
     43       1.1  tsutsui  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     44       1.1  tsutsui  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     45       1.1  tsutsui  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     46       1.1  tsutsui  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     47       1.1  tsutsui  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     48       1.1  tsutsui  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     49       1.1  tsutsui  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     50       1.1  tsutsui  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     51       1.1  tsutsui  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     52       1.1  tsutsui  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     53       1.1  tsutsui  * POSSIBILITY OF SUCH DAMAGE.
     54       1.1  tsutsui  */
     55       1.1  tsutsui 
     56       1.1  tsutsui #if HAVE_NBTOOL_CONFIG_H
     57       1.1  tsutsui #include "nbtool_config.h"
     58       1.1  tsutsui #endif
     59       1.1  tsutsui 
     60       1.1  tsutsui #include <sys/cdefs.h>
     61       1.6  tsutsui #if !defined(__lint)
     62  1.6.12.2      tls __RCSID("$NetBSD: ext2fs.c,v 1.6.12.2 2013/06/23 06:29:04 tls Exp $");
     63       1.1  tsutsui #endif	/* !__lint */
     64       1.1  tsutsui 
     65       1.1  tsutsui #include <sys/param.h>
     66       1.1  tsutsui 
     67       1.1  tsutsui #if !HAVE_NBTOOL_CONFIG_H
     68       1.1  tsutsui #include <sys/mount.h>
     69       1.1  tsutsui #endif
     70       1.1  tsutsui 
     71       1.1  tsutsui #include <assert.h>
     72       1.1  tsutsui #include <err.h>
     73       1.1  tsutsui #include <errno.h>
     74       1.1  tsutsui #include <fcntl.h>
     75       1.1  tsutsui #include <stdarg.h>
     76       1.1  tsutsui #include <stdio.h>
     77       1.1  tsutsui #include <stdlib.h>
     78       1.1  tsutsui #include <string.h>
     79       1.1  tsutsui #include <unistd.h>
     80       1.1  tsutsui 
     81       1.1  tsutsui #include "installboot.h"
     82       1.1  tsutsui 
     83       1.1  tsutsui #include <ufs/ext2fs/ext2fs_dinode.h>
     84       1.1  tsutsui #include <ufs/ext2fs/ext2fs_dir.h>
     85       1.1  tsutsui #include <ufs/ext2fs/ext2fs.h>
     86       1.1  tsutsui 
     87       1.1  tsutsui static int	ext2fs_read_disk_block(ib_params *, uint64_t, int, uint8_t []);
     88       1.1  tsutsui static int	ext2fs_read_sblock(ib_params *, struct m_ext2fs *fs);
     89       1.1  tsutsui static int	ext2fs_read_gdblock(ib_params *, struct m_ext2fs *fs);
     90       1.1  tsutsui static int	ext2fs_find_disk_blocks(ib_params *, ino_t,
     91       1.1  tsutsui 		    int (*)(ib_params *, void *, uint64_t, uint32_t), void *);
     92       1.1  tsutsui static int	ext2fs_findstage2_ino(ib_params *, void *, uint64_t, uint32_t);
     93       1.1  tsutsui static int	ext2fs_findstage2_blocks(ib_params *, void *, uint64_t,
     94       1.1  tsutsui 		    uint32_t);
     95       1.1  tsutsui 
     96       1.1  tsutsui 
     97       1.1  tsutsui /* This reads a disk block from the file system. */
     98       1.1  tsutsui /* XXX: should be shared with ffs.c? */
     99       1.1  tsutsui static int
    100       1.1  tsutsui ext2fs_read_disk_block(ib_params *params, uint64_t blkno, int size,
    101       1.1  tsutsui     uint8_t blk[])
    102       1.1  tsutsui {
    103       1.1  tsutsui 	int rv;
    104       1.1  tsutsui 
    105       1.1  tsutsui 	assert(params != NULL);
    106       1.1  tsutsui 	assert(params->filesystem != NULL);
    107       1.1  tsutsui 	assert(params->fsfd != -1);
    108       1.1  tsutsui 	assert(size > 0);
    109       1.1  tsutsui 	assert(blk != NULL);
    110       1.1  tsutsui 
    111       1.5  tsutsui 	rv = pread(params->fsfd, blk, size, blkno * params->sectorsize);
    112       1.1  tsutsui 	if (rv == -1) {
    113       1.1  tsutsui 		warn("Reading block %llu in `%s'",
    114       1.1  tsutsui 		    (unsigned long long)blkno, params->filesystem);
    115       1.1  tsutsui 		return 0;
    116       1.1  tsutsui 	} else if (rv != size) {
    117       1.1  tsutsui 		warnx("Reading block %llu in `%s': short read",
    118       1.1  tsutsui 		    (unsigned long long)blkno, params->filesystem);
    119       1.1  tsutsui 		return 0;
    120       1.1  tsutsui 	}
    121       1.1  tsutsui 
    122       1.1  tsutsui 	return 1;
    123       1.1  tsutsui }
    124       1.1  tsutsui 
    125       1.1  tsutsui static int
    126       1.1  tsutsui ext2fs_read_sblock(ib_params *params, struct m_ext2fs *fs)
    127       1.1  tsutsui {
    128       1.1  tsutsui 	uint8_t sbbuf[SBSIZE];
    129       1.1  tsutsui 
    130       1.5  tsutsui 	if (ext2fs_read_disk_block(params, SBOFF / params->sectorsize, SBSIZE,
    131       1.1  tsutsui 	    sbbuf) == 0)
    132       1.1  tsutsui 
    133       1.1  tsutsui 	e2fs_sbload((void *)sbbuf, &fs->e2fs);
    134       1.1  tsutsui 
    135       1.1  tsutsui 	if (fs->e2fs.e2fs_magic != E2FS_MAGIC)
    136       1.1  tsutsui 		return 0;
    137       1.1  tsutsui 
    138       1.1  tsutsui 	if (fs->e2fs.e2fs_rev > E2FS_REV1 ||
    139       1.1  tsutsui 	    (fs->e2fs.e2fs_rev == E2FS_REV1 &&
    140       1.1  tsutsui 	     (fs->e2fs.e2fs_first_ino != EXT2_FIRSTINO ||
    141       1.1  tsutsui 	      fs->e2fs.e2fs_inode_size != EXT2_DINODE_SIZE ||
    142       1.1  tsutsui 	      (fs->e2fs.e2fs_features_incompat & ~EXT2F_INCOMPAT_SUPP) != 0)))
    143       1.1  tsutsui 		return 0;
    144       1.1  tsutsui 
    145       1.1  tsutsui 	fs->e2fs_ncg =
    146       1.1  tsutsui 	    howmany(fs->e2fs.e2fs_bcount - fs->e2fs.e2fs_first_dblock,
    147       1.1  tsutsui 	    fs->e2fs.e2fs_bpg);
    148       1.1  tsutsui 	/* XXX assume hw bsize = 512 */
    149       1.1  tsutsui 	fs->e2fs_fsbtodb = fs->e2fs.e2fs_log_bsize + 1;
    150       1.1  tsutsui 	fs->e2fs_bsize = MINBSIZE << fs->e2fs.e2fs_log_bsize;
    151       1.1  tsutsui 	fs->e2fs_bshift = LOG_MINBSIZE + fs->e2fs.e2fs_log_bsize;
    152       1.1  tsutsui 	fs->e2fs_qbmask = fs->e2fs_bsize - 1;
    153       1.1  tsutsui 	fs->e2fs_bmask = ~fs->e2fs_qbmask;
    154       1.1  tsutsui 	fs->e2fs_ngdb =
    155       1.1  tsutsui 	    howmany(fs->e2fs_ncg, fs->e2fs_bsize / sizeof(struct ext2_gd));
    156       1.1  tsutsui 	fs->e2fs_ipb = fs->e2fs_bsize / EXT2_DINODE_SIZE;
    157       1.1  tsutsui 	fs->e2fs_itpg = fs->e2fs.e2fs_ipg / fs->e2fs_ipb;
    158       1.1  tsutsui 
    159       1.1  tsutsui 	return 1;
    160       1.1  tsutsui }
    161       1.1  tsutsui 
    162       1.1  tsutsui static int
    163       1.1  tsutsui ext2fs_read_gdblock(ib_params *params, struct m_ext2fs *fs)
    164       1.1  tsutsui {
    165       1.1  tsutsui 	uint8_t gdbuf[MAXBSIZE];
    166       1.1  tsutsui 	uint32_t gdpb;
    167       1.1  tsutsui 	int i;
    168       1.1  tsutsui 
    169       1.1  tsutsui 	gdpb = fs->e2fs_bsize / sizeof(struct ext2_gd);
    170       1.1  tsutsui 
    171       1.1  tsutsui 	for (i = 0; i < fs->e2fs_ngdb; i++) {
    172       1.1  tsutsui 		if (ext2fs_read_disk_block(params, fsbtodb(fs,
    173       1.1  tsutsui 		    fs->e2fs.e2fs_first_dblock + 1 /* superblock */ + i),
    174       1.1  tsutsui 		    SBSIZE, gdbuf) == 0)
    175       1.1  tsutsui 			return 0;
    176       1.1  tsutsui 
    177       1.1  tsutsui 		e2fs_cgload((struct ext2_gd *)gdbuf, &fs->e2fs_gd[gdpb * i],
    178       1.1  tsutsui 		    (i == (fs->e2fs_ngdb - 1)) ?
    179       1.1  tsutsui 		    (fs->e2fs_ncg - gdpb * i) * sizeof(struct ext2_gd):
    180       1.1  tsutsui 		    fs->e2fs_bsize);
    181       1.1  tsutsui 	}
    182       1.1  tsutsui 
    183       1.1  tsutsui 	return 1;
    184       1.1  tsutsui }
    185       1.1  tsutsui 
    186       1.1  tsutsui /*
    187       1.1  tsutsui  * This iterates over the data blocks belonging to an inode,
    188       1.1  tsutsui  * making a callback each iteration with the disk block number
    189       1.1  tsutsui  * and the size.
    190       1.1  tsutsui  */
    191       1.1  tsutsui static int
    192       1.1  tsutsui ext2fs_find_disk_blocks(ib_params *params, ino_t ino,
    193       1.1  tsutsui 	int (*callback)(ib_params *, void *, uint64_t, uint32_t),
    194       1.1  tsutsui 	void *state)
    195       1.1  tsutsui {
    196       1.1  tsutsui 	uint8_t sbbuf[sizeof(struct m_ext2fs)];
    197       1.1  tsutsui 	struct m_ext2fs *fs;
    198       1.1  tsutsui 	uint8_t inodebuf[MAXBSIZE];
    199       1.1  tsutsui 	struct ext2fs_dinode inode_store, *inode;
    200       1.1  tsutsui 	int level_i;
    201       1.1  tsutsui 	int32_t blk, lblk, nblk;
    202       1.1  tsutsui 	int rv;
    203       1.1  tsutsui #define LEVELS 4
    204       1.1  tsutsui 	struct {
    205       1.1  tsutsui 		uint32_t *blknums;
    206       1.1  tsutsui 		unsigned long blkcount;
    207       1.1  tsutsui 		uint8_t diskbuf[MAXBSIZE];
    208       1.1  tsutsui 	} level[LEVELS];
    209       1.1  tsutsui 
    210       1.1  tsutsui 	assert(params != NULL);
    211       1.1  tsutsui 	assert(params->fstype != NULL);
    212       1.1  tsutsui 	assert(callback != NULL);
    213       1.1  tsutsui 	assert(state != NULL);
    214       1.1  tsutsui 
    215       1.1  tsutsui 	/* Read the superblock. */
    216       1.1  tsutsui 	fs = (void *)sbbuf;
    217       1.1  tsutsui 	if (ext2fs_read_sblock(params, fs) == 0)
    218       1.1  tsutsui 		return 0;
    219       1.1  tsutsui 
    220       1.1  tsutsui 	fs->e2fs_gd = malloc(sizeof(struct ext2_gd) * fs->e2fs_ncg);
    221       1.1  tsutsui 	if (fs->e2fs_gd == NULL) {
    222       1.1  tsutsui 		warnx("Can't allocate memofy for group descriptors");
    223       1.1  tsutsui 		return 0;
    224       1.1  tsutsui 	}
    225       1.1  tsutsui 
    226       1.1  tsutsui 	if (ext2fs_read_gdblock(params, fs) == 0) {
    227       1.1  tsutsui 		warnx("Can't read group descriptors");
    228       1.1  tsutsui 		return 0;
    229       1.1  tsutsui 	}
    230       1.1  tsutsui 
    231       1.1  tsutsui 	if (fs->e2fs_ipb <= 0) {
    232       1.1  tsutsui 		warnx("Bad ipb %d in superblock in `%s'",
    233       1.1  tsutsui 		    fs->e2fs_ipb, params->filesystem);
    234       1.1  tsutsui 		return 0;
    235       1.1  tsutsui 	}
    236       1.1  tsutsui 
    237       1.1  tsutsui 	/* Read the inode. */
    238       1.1  tsutsui 	if (ext2fs_read_disk_block(params,
    239       1.1  tsutsui 		fsbtodb(fs, ino_to_fsba(fs, ino)) + params->fstype->offset,
    240       1.1  tsutsui 		fs->e2fs_bsize, inodebuf))
    241       1.1  tsutsui 		return 0;
    242       1.1  tsutsui 	inode = (void *)inodebuf;
    243       1.1  tsutsui 	e2fs_iload(&inode[ino_to_fsbo(fs, ino)], &inode_store);
    244       1.1  tsutsui 	inode = &inode_store;
    245       1.1  tsutsui 
    246       1.1  tsutsui 	/* Get the block count and initialize for our block walk. */
    247       1.1  tsutsui 	nblk = howmany(inode->e2di_size, fs->e2fs_bsize);
    248       1.1  tsutsui 	lblk = 0;
    249       1.1  tsutsui 	level_i = 0;
    250       1.1  tsutsui 	level[0].blknums = &inode->e2di_blocks[0];
    251  1.6.12.1      tls 	level[0].blkcount = UFS_NDADDR;
    252  1.6.12.1      tls 	level[1].blknums = &inode->e2di_blocks[UFS_NDADDR + 0];
    253       1.1  tsutsui 	level[1].blkcount = 1;
    254  1.6.12.1      tls 	level[2].blknums = &inode->e2di_blocks[UFS_NDADDR + 1];
    255       1.1  tsutsui 	level[2].blkcount = 1;
    256  1.6.12.1      tls 	level[3].blknums = &inode->e2di_blocks[UFS_NDADDR + 2];
    257       1.1  tsutsui 	level[3].blkcount = 1;
    258       1.1  tsutsui 
    259       1.1  tsutsui 	/* Walk the data blocks. */
    260       1.1  tsutsui 	while (nblk > 0) {
    261       1.1  tsutsui 
    262       1.1  tsutsui 		/*
    263       1.1  tsutsui 		 * If there are no more blocks at this indirection
    264       1.1  tsutsui 		 * level, move up one indirection level and loop.
    265       1.1  tsutsui 		 */
    266       1.1  tsutsui 		if (level[level_i].blkcount == 0) {
    267       1.1  tsutsui 			if (++level_i == LEVELS)
    268       1.1  tsutsui 				break;
    269       1.1  tsutsui 			continue;
    270       1.1  tsutsui 		}
    271       1.1  tsutsui 
    272       1.1  tsutsui 		/* Get the next block at this level. */
    273       1.1  tsutsui 		blk = fs2h32(*(level[level_i].blknums++));
    274       1.1  tsutsui 		level[level_i].blkcount--;
    275       1.1  tsutsui 
    276       1.1  tsutsui #if 0
    277       1.1  tsutsui 		fprintf(stderr, "ino %lu blk %lu level %d\n", ino, blk,
    278       1.1  tsutsui 		    level_i);
    279       1.1  tsutsui #endif
    280       1.1  tsutsui 
    281       1.1  tsutsui 		/*
    282       1.1  tsutsui 		 * If we're not at the direct level, descend one
    283       1.1  tsutsui 		 * level, read in that level's new block list,
    284       1.1  tsutsui 		 * and loop.
    285       1.1  tsutsui 		 */
    286       1.1  tsutsui 		if (level_i > 0) {
    287       1.1  tsutsui 			level_i--;
    288       1.1  tsutsui 			if (blk == 0)
    289       1.1  tsutsui 				memset(level[level_i].diskbuf, 0, MAXBSIZE);
    290       1.1  tsutsui 			else if (ext2fs_read_disk_block(params,
    291       1.1  tsutsui 				fsbtodb(fs, blk) + params->fstype->offset,
    292       1.1  tsutsui 				fs->e2fs_bsize, level[level_i].diskbuf) == 0)
    293       1.1  tsutsui 				return 0;
    294       1.1  tsutsui 			/* XXX ondisk32 */
    295       1.1  tsutsui 			level[level_i].blknums =
    296       1.1  tsutsui 			    (uint32_t *)level[level_i].diskbuf;
    297  1.6.12.2      tls 			level[level_i].blkcount = EXT2_NINDIR(fs);
    298       1.1  tsutsui 			continue;
    299       1.1  tsutsui 		}
    300       1.1  tsutsui 
    301       1.1  tsutsui 		/* blk is the next direct level block. */
    302       1.1  tsutsui #if 0
    303       1.1  tsutsui 		fprintf(stderr, "ino %lu db %lu blksize %lu\n", ino,
    304  1.6.12.2      tls 		    fsbtodb(fs, blk), ext2_sblksize(fs, inode->di_size, lblk));
    305       1.1  tsutsui #endif
    306       1.1  tsutsui 		rv = (*callback)(params, state,
    307       1.1  tsutsui 		    fsbtodb(fs, blk) + params->fstype->offset, fs->e2fs_bsize);
    308       1.1  tsutsui 		lblk++;
    309       1.1  tsutsui 		nblk--;
    310       1.1  tsutsui 		if (rv != 1)
    311       1.1  tsutsui 			return rv;
    312       1.1  tsutsui 	}
    313       1.1  tsutsui 
    314       1.1  tsutsui 	if (nblk != 0) {
    315       1.1  tsutsui 		warnx("Inode %llu in `%s' ran out of blocks?",
    316       1.1  tsutsui 		    (unsigned long long)ino, params->filesystem);
    317       1.1  tsutsui 		return 0;
    318       1.1  tsutsui 	}
    319       1.1  tsutsui 
    320       1.1  tsutsui 	return 1;
    321       1.1  tsutsui }
    322       1.1  tsutsui 
    323       1.1  tsutsui /*
    324       1.1  tsutsui  * This callback reads a block of the root directory,
    325       1.1  tsutsui  * searches for an entry for the secondary bootstrap,
    326       1.1  tsutsui  * and saves the inode number if one is found.
    327       1.1  tsutsui  */
    328       1.1  tsutsui static int
    329       1.1  tsutsui ext2fs_findstage2_ino(ib_params *params, void *_ino,
    330       1.1  tsutsui 	uint64_t blk, uint32_t blksize)
    331       1.1  tsutsui {
    332       1.1  tsutsui 	uint8_t dirbuf[MAXBSIZE];
    333       1.1  tsutsui 	struct ext2fs_direct *de, *ede;
    334       1.1  tsutsui 	uint32_t ino;
    335       1.1  tsutsui 
    336       1.1  tsutsui 	assert(params != NULL);
    337       1.1  tsutsui 	assert(params->fstype != NULL);
    338       1.1  tsutsui 	assert(params->stage2 != NULL);
    339       1.1  tsutsui 	assert(_ino != NULL);
    340       1.1  tsutsui 
    341       1.1  tsutsui 	/* Skip directory holes. */
    342       1.1  tsutsui 	if (blk == 0)
    343       1.1  tsutsui 		return 1;
    344       1.1  tsutsui 
    345       1.1  tsutsui 	/* Read the directory block. */
    346       1.1  tsutsui 	if (ext2fs_read_disk_block(params, blk, blksize, dirbuf) == 0)
    347       1.1  tsutsui 		return 0;
    348       1.1  tsutsui 
    349       1.1  tsutsui 	/* Loop over the directory entries. */
    350       1.1  tsutsui 	de = (struct ext2fs_direct *)&dirbuf[0];
    351       1.1  tsutsui 	ede = (struct ext2fs_direct *)&dirbuf[blksize];
    352       1.1  tsutsui 	while (de < ede) {
    353       1.1  tsutsui 		ino = fs2h32(de->e2d_ino);
    354       1.1  tsutsui 		if (ino != 0 && strcmp(de->e2d_name, params->stage2) == 0) {
    355       1.1  tsutsui 			*((uint32_t *)_ino) = ino;
    356       1.1  tsutsui 			return (2);
    357       1.1  tsutsui 		}
    358       1.1  tsutsui 		if (fs2h16(de->e2d_reclen) == 0)
    359       1.1  tsutsui 			break;
    360       1.1  tsutsui 		de = (struct ext2fs_direct *)((char *)de +
    361       1.1  tsutsui 		    fs2h16(de->e2d_reclen));
    362       1.1  tsutsui 	}
    363       1.1  tsutsui 
    364       1.1  tsutsui 	return 1;
    365       1.1  tsutsui }
    366       1.1  tsutsui 
    367       1.1  tsutsui struct findblks_state {
    368       1.1  tsutsui 	uint32_t	maxblk;
    369       1.1  tsutsui 	uint32_t	nblk;
    370       1.1  tsutsui 	ib_block	*blocks;
    371       1.1  tsutsui };
    372       1.1  tsutsui 
    373       1.1  tsutsui /* This callback records the blocks of the secondary bootstrap. */
    374       1.1  tsutsui static int
    375       1.1  tsutsui ext2fs_findstage2_blocks(ib_params *params, void *_state,
    376       1.1  tsutsui 	uint64_t blk, uint32_t blksize)
    377       1.1  tsutsui {
    378       1.1  tsutsui 	struct findblks_state *state = _state;
    379       1.1  tsutsui 
    380       1.1  tsutsui 	assert(params != NULL);
    381       1.1  tsutsui 	assert(params->stage2 != NULL);
    382       1.1  tsutsui 	assert(_state != NULL);
    383       1.1  tsutsui 
    384       1.1  tsutsui 	if (state->nblk == state->maxblk) {
    385       1.1  tsutsui 		warnx("Secondary bootstrap `%s' has too many blocks (max %d)",
    386       1.1  tsutsui 		    params->stage2, state->maxblk);
    387       1.1  tsutsui 		return (0);
    388       1.1  tsutsui 	}
    389       1.1  tsutsui 	state->blocks[state->nblk].block = blk;
    390       1.1  tsutsui 	state->blocks[state->nblk].blocksize = blksize;
    391       1.1  tsutsui 	state->nblk++;
    392       1.1  tsutsui 	return 1;
    393       1.1  tsutsui }
    394       1.1  tsutsui 
    395       1.1  tsutsui /*
    396       1.1  tsutsui  *	publicly visible functions
    397       1.1  tsutsui  */
    398       1.1  tsutsui 
    399       1.1  tsutsui int
    400       1.1  tsutsui ext2fs_match(ib_params *params)
    401       1.1  tsutsui {
    402       1.1  tsutsui 	uint8_t sbbuf[sizeof(struct m_ext2fs)];
    403       1.1  tsutsui 	struct m_ext2fs *fs;
    404       1.1  tsutsui 
    405       1.1  tsutsui 	assert(params != NULL);
    406       1.1  tsutsui 	assert(params->fstype != NULL);
    407       1.1  tsutsui 
    408       1.1  tsutsui 	/* Read the superblock. */
    409       1.1  tsutsui 	fs = (void *)sbbuf;
    410       1.1  tsutsui 	if (ext2fs_read_sblock(params, fs) == 0)
    411       1.1  tsutsui 		return 0;
    412       1.1  tsutsui 
    413       1.1  tsutsui 	params->fstype->needswap = 0;
    414       1.1  tsutsui 	params->fstype->blocksize = fs->e2fs_bsize;
    415       1.1  tsutsui 	params->fstype->offset = 0;
    416       1.1  tsutsui 
    417       1.1  tsutsui 	return 1;
    418       1.1  tsutsui }
    419       1.1  tsutsui 
    420       1.1  tsutsui int
    421       1.1  tsutsui ext2fs_findstage2(ib_params *params, uint32_t *maxblk, ib_block *blocks)
    422       1.1  tsutsui {
    423       1.1  tsutsui 	int rv;
    424       1.1  tsutsui 	uint32_t ino;
    425       1.1  tsutsui 	struct findblks_state state;
    426       1.1  tsutsui 
    427       1.1  tsutsui 	assert(params != NULL);
    428       1.1  tsutsui 	assert(params->stage2 != NULL);
    429       1.1  tsutsui 	assert(maxblk != NULL);
    430       1.1  tsutsui 	assert(blocks != NULL);
    431       1.1  tsutsui 
    432       1.1  tsutsui 	if (params->flags & IB_STAGE2START)
    433       1.1  tsutsui 		return hardcode_stage2(params, maxblk, blocks);
    434       1.1  tsutsui 
    435       1.1  tsutsui 	/* The secondary bootstrap must be clearly in /. */
    436       1.1  tsutsui 	if (params->stage2[0] == '/')
    437       1.1  tsutsui 		params->stage2++;
    438       1.1  tsutsui 	if (strchr(params->stage2, '/') != NULL) {
    439       1.1  tsutsui 		warnx("The secondary bootstrap `%s' must be in /",
    440       1.1  tsutsui 		    params->stage2);
    441       1.3      apb 		warnx("(Path must be relative to the file system in `%s')",
    442       1.3      apb 		    params->filesystem);
    443       1.1  tsutsui 		return 0;
    444       1.1  tsutsui 	}
    445       1.1  tsutsui 
    446       1.1  tsutsui 	/* Get the inode number of the secondary bootstrap. */
    447       1.1  tsutsui 	rv = ext2fs_find_disk_blocks(params, EXT2_ROOTINO,
    448       1.1  tsutsui 	    ext2fs_findstage2_ino, &ino);
    449       1.1  tsutsui 	if (rv != 2) {
    450       1.1  tsutsui 		warnx("Could not find secondary bootstrap `%s' in `%s'",
    451       1.1  tsutsui 		    params->stage2, params->filesystem);
    452       1.3      apb 		warnx("(Path must be relative to the file system in `%s')",
    453       1.3      apb 		    params->filesystem);
    454       1.1  tsutsui 		return 0;
    455       1.1  tsutsui 	}
    456       1.1  tsutsui 
    457       1.1  tsutsui 	/* Record the disk blocks of the secondary bootstrap. */
    458       1.1  tsutsui 	state.maxblk = *maxblk;
    459       1.1  tsutsui 	state.nblk = 0;
    460       1.1  tsutsui 	state.blocks = blocks;
    461       1.1  tsutsui 		rv = ext2fs_find_disk_blocks(params, ino,
    462       1.1  tsutsui 		    ext2fs_findstage2_blocks, &state);
    463       1.1  tsutsui 	if (rv == 0)
    464       1.1  tsutsui 		return 0;
    465       1.1  tsutsui 
    466       1.1  tsutsui 	*maxblk = state.nblk;
    467       1.1  tsutsui 	return 1;
    468       1.1  tsutsui }
    469