File: [local] / src / usr.sbin / installboot / i386_softraid.c (download)
Revision 1.21, Mon Nov 7 15:56:09 2022 UTC (19 months ago) by kn
Branch: MAIN
CVS Tags: OPENBSD_7_5_BASE, OPENBSD_7_5, OPENBSD_7_4_BASE, OPENBSD_7_4, OPENBSD_7_3_BASE, OPENBSD_7_3, HEAD Changes since 1.20: +3 -33 lines
Merge duplicate MD code into MI sr_open_chunk()
It does not have the prettiest signature, but nicely folds identical copies
into softraid.c, which then allows us to reuse sr_open_chunk() yet again in
an upcoming diff to make -p softraid aware (fixes arm64 installations).
Regress keeps passing.
Works fine on amd64, arm64 and sparc64.
"looks fine" mlarkin for whom this unbreaks a fresh arm64 softraid install
|
/* $OpenBSD: i386_softraid.c,v 1.21 2022/11/07 15:56:09 kn Exp $ */
/*
* Copyright (c) 2012 Joel Sing <jsing@openbsd.org>
* Copyright (c) 2010 Otto Moerbeek <otto@drijf.net>
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#include <sys/param.h> /* DEV_BSIZE */
#include <sys/disklabel.h>
#include <sys/dkio.h>
#include <sys/ioctl.h>
#include <sys/stat.h>
#include <dev/biovar.h>
#include <dev/softraidvar.h>
#include <ufs/ufs/dinode.h>
#include <err.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "installboot.h"
#include "i386_installboot.h"
void sr_install_bootblk(int, int, int);
void sr_install_bootldr(int, char *);
void
sr_install_bootblk(int devfd, int vol, int disk)
{
struct bioc_disk bd;
struct disklabel dl;
struct partition *pp;
uint32_t poffset;
char *dev;
char part, efipart;
int diskfd;
diskfd = sr_open_chunk(devfd, vol, disk, &bd, &dev, &part);
if (diskfd == -1)
return;
/* Get and check disklabel. */
if (ioctl(diskfd, DIOCGDINFO, &dl) == -1)
err(1, "disklabel: %s", dev);
if (dl.d_magic != DISKMAGIC)
err(1, "bad disklabel magic=0x%08x", dl.d_magic);
/* Warn on unknown disklabel types. */
if (dl.d_type == 0)
warnx("disklabel type unknown");
efipart = findgptefisys(diskfd, &dl);
if (efipart != -1) {
write_filesystem(&dl, (char)efipart);
return;
}
/* Determine poffset and set symbol value. */
pp = &dl.d_partitions[part - 'a'];
if (pp->p_offseth != 0)
errx(1, "partition offset too high");
poffset = pp->p_offset; /* Offset of RAID partition. */
poffset += SR_BOOT_LOADER_OFFSET; /* SR boot loader area. */
sym_set_value(pbr_symbols, "_p_offset", poffset);
if (verbose)
fprintf(stderr, "%s%c: %s boot blocks on %s, part offset %u\n",
bd.bd_vendor, part,
(nowrite ? "would install" : "installing"), dev, poffset);
/* Write boot blocks to device. */
write_bootblocks(diskfd, dev, &dl);
close(diskfd);
}
void
sr_install_bootldr(int devfd, char *dev)
{
struct bioc_installboot bb;
struct stat sb;
struct ufs1_dinode *ino_p;
uint32_t bootsize, inodeblk, inodedbl;
uint16_t bsize = SR_FS_BLOCKSIZE;
uint16_t nblocks;
uint8_t bshift = 5; /* fragsize == blocksize */
int fd, i;
u_char *p;
/*
* Install boot loader into softraid boot loader storage area.
*
* In order to allow us to reuse the existing biosboot we construct
* a fake FFS filesystem with a single inode, which points to the
* boot loader.
*/
nblocks = howmany(SR_BOOT_LOADER_SIZE, SR_FS_BLOCKSIZE / DEV_BSIZE);
inodeblk = nblocks - 1;
bootsize = nblocks * SR_FS_BLOCKSIZE;
p = calloc(1, bootsize);
if (p == NULL)
err(1, NULL);
fd = open(stage2, O_RDONLY);
if (fd == -1)
err(1, NULL);
if (fstat(fd, &sb) == -1)
err(1, NULL);
nblocks = howmany(sb.st_blocks, SR_FS_BLOCKSIZE / DEV_BSIZE);
if (sb.st_blocks * S_BLKSIZE > bootsize -
(int)(sizeof(struct ufs1_dinode)))
errx(1, "boot code will not fit");
/* We only need to fill the direct block array. */
ino_p = (struct ufs1_dinode *)&p[bootsize - sizeof(struct ufs1_dinode)];
ino_p->di_mode = sb.st_mode;
ino_p->di_nlink = 1;
ino_p->di_inumber = 0xfeebfaab;
ino_p->di_size = read(fd, p, sb.st_blocks * S_BLKSIZE);
ino_p->di_blocks = nblocks;
for (i = 0; i < nblocks; i++)
ino_p->di_db[i] = i;
inodedbl = ((u_char*)&ino_p->di_db[0] -
&p[bootsize - SR_FS_BLOCKSIZE]) + INODEOFF;
memset(&bb, 0, sizeof(bb));
bb.bb_bootldr = p;
bb.bb_bootldr_size = bootsize;
bb.bb_bootblk = "XXX";
bb.bb_bootblk_size = sizeof("XXX");
strncpy(bb.bb_dev, dev, sizeof(bb.bb_dev));
if (verbose)
fprintf(stderr, "%s: %s boot loader on softraid volume\n", dev,
(nowrite ? "would install" : "installing"));
if (!nowrite) {
if (ioctl(devfd, BIOCINSTALLBOOT, &bb) == -1)
errx(1, "softraid installboot failed");
sr_status(&bb.bb_bio.bio_status);
}
/*
* Set the values that will need to go into biosboot
* (the partition boot record, a.k.a. the PBR).
*/
sym_set_value(pbr_symbols, "_fs_bsize_p", (bsize / 16));
sym_set_value(pbr_symbols, "_fs_bsize_s", (bsize / 512));
sym_set_value(pbr_symbols, "_fsbtodb", bshift);
sym_set_value(pbr_symbols, "_inodeblk", inodeblk);
sym_set_value(pbr_symbols, "_inodedbl", inodedbl);
sym_set_value(pbr_symbols, "_nblocks", nblocks);
sym_set_value(pbr_symbols, "_blkincr", 0);
if (verbose)
fprintf(stderr, "%s is %d blocks x %d bytes\n",
stage2, nblocks, bsize);
free(p);
close(fd);
}