/* $NetBSD: cd9660.c,v 1.2.2.2 2024/06/22 10:57:10 martin Exp $ */ /*- * Copyright (c) 2005 Izumi Tsutsui. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #if HAVE_NBTOOL_CONFIG_H #include "nbtool_config.h" #endif #include <sys/cdefs.h> #if defined(__RCSID) && !defined(__lint) __RCSID("$NetBSD: cd9660.c,v 1.2.2.2 2024/06/22 10:57:10 martin Exp $"); #endif /* !__lint */ #include <sys/param.h> #if !HAVE_NBTOOL_CONFIG_H #include <sys/mount.h> #endif #if !HAVE_NBTOOL_CONFIG_H || HAVE_SYS_ENDIAN_H #include <sys/endian.h> #endif #include <assert.h> #include <err.h> #include <errno.h> #include <fcntl.h> #include <stdarg.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <dirent.h> #include <fs/cd9660/iso.h> #include <fs/cd9660/cd9660_extern.h> #include "installboot.h" #define roundup(x, y) ((((x)+((y)-1))/(y))*(y)) #define MAXLEN 16 int cd9660_match(ib_params *params) { int rv, blocksize; struct iso_primary_descriptor ipd; assert(params != NULL); assert(params->fstype != NULL); assert(params->fsfd != -1); rv = pread(params->fsfd, &ipd, sizeof(ipd), ISO_DEFAULT_BLOCK_SIZE * 16); if (rv == -1) { warn("Reading primary descriptor in `%s'", params->filesystem); return 0; } else if (rv != sizeof(ipd)) { warnx("Reading primary descriptor in `%s': short read", params->filesystem); return 0; } if (ipd.type[0] != ISO_VD_PRIMARY || strncmp(ipd.id, ISO_STANDARD_ID, sizeof(ipd.id)) != 0 || ipd.version[0] != 1) { warnx("Filesystem `%s' is not ISO9660 format", params->filesystem); return 0; } blocksize = isonum_723((u_char *)ipd.logical_block_size); if (blocksize != ISO_DEFAULT_BLOCK_SIZE) { warnx("Invalid blocksize %d in `%s'", blocksize, params->filesystem); return 0; } params->fstype->blocksize = blocksize; params->fstype->needswap = 0; return 1; } int cd9660_findstage2(ib_params *params, uint32_t *maxblk, ib_block *blocks) { uint8_t buf[ISO_DEFAULT_BLOCK_SIZE]; char name[ISO_MAXNAMLEN]; char *stage2; off_t loc; int rv, blocksize, found; u_int i; struct iso_primary_descriptor ipd; struct iso_directory_record *idr; assert(params != NULL); assert(params->stage2 != NULL); assert(maxblk != NULL); assert(blocks != NULL); #if 0 if (params->flags & IB_STAGE2START) return hardcode_stage2(params, maxblk, blocks); #endif /* The secondary bootstrap must be clearly in /. */ strlcpy(name, params->stage2, ISO_MAXNAMLEN); stage2 = name; if (stage2[0] == '/') stage2++; if (strchr(stage2, '/') != NULL) { warnx("The secondary bootstrap `%s' must be in / " "on filesystem `%s'", params->stage2, params->filesystem); return 0; } if (strchr(stage2, '.') == NULL) { /* * XXX should fix isofncmp()? */ strlcat(name, ".", ISO_MAXNAMLEN); } rv = pread(params->fsfd, &ipd, sizeof(ipd), ISO_DEFAULT_BLOCK_SIZE * 16); if (rv == -1) { warn("Reading primary descriptor in `%s'", params->filesystem); return 0; } else if (rv != sizeof(ipd)) { warnx("Reading primary descriptor in `%s': short read", params->filesystem); return 0; } blocksize = isonum_723((u_char *)ipd.logical_block_size); idr = (void *)ipd.root_directory_record; loc = (off_t)isonum_733(idr->extent) * blocksize; rv = pread(params->fsfd, buf, blocksize, loc); if (rv == -1) { warn("Reading root directory record in `%s'", params->filesystem); return 0; } else if (rv != sizeof(ipd)) { warnx("Reading root directory record in `%s': short read", params->filesystem); return 0; } found = 0; for (i = 0; i < blocksize - sizeof(struct iso_directory_record); i += (u_char)idr->length[0]) { idr = (void *)&buf[i]; #ifdef DEBUG printf("i = %d, idr->length[0] = %3d\n", i, (u_char)idr->length[0]); #endif /* check end of entries */ if (idr->length[0] == 0) { #ifdef DEBUG printf("end of entries\n"); #endif break; } if (idr->flags[0] & 2) { /* skip directory entries */ #ifdef DEBUG printf("skip directory entry\n"); #endif continue; } if (idr->name_len[0] == 1 && (idr->name[0] == 0 || idr->name[0] == 1)) { /* skip "." and ".." */ #ifdef DEBUG printf("skip dot dot\n"); #endif continue; } #ifdef DEBUG { int j; printf("filename:"); for (j = 0; j < isonum_711(idr->name_len); j++) printf("%c", idr->name[j]); printf("\n"); } #endif if (isofncmp((u_char *)stage2, strlen(stage2), (u_char *)idr->name, isonum_711((u_char *)idr->name_len), 0) == 0) { found = 1; /* ISO filesystem always has contiguous file blocks */ blocks[0].block = (int64_t)isonum_733(idr->extent); blocks[0].blocksize = roundup(isonum_733(idr->size), blocksize); *maxblk = 1; #ifdef DEBUG printf("block = %ld, blocksize = %ld\n", (long)blocks[0].block, blocks[0].blocksize); #endif break; } } if (found == 0) { warnx("Can't find secondary bootstrap `%s' in filesystem `%s'", params->stage2, params->filesystem); return 0; } return 1; }