#include <sys/param.h>
#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 gpt_partition gp;
struct partition *pp;
uint32_t poffset;
char *dev;
char part;
int gpart, efipart;
int diskfd;
diskfd = sr_open_chunk(devfd, vol, disk, &bd, &dev, &part);
if (diskfd == -1)
return;
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);
if (dl.d_type == 0)
warnx("disklabel type unknown");
efipart = findgptefisys(diskfd, &dl, &gpart, &gp);
if (efipart != -1) {
write_filesystem(&dl, (char)efipart, gpart, &gp);
return;
}
pp = &dl.d_partitions[DL_PARTNAME2NUM(part)];
if (pp->p_offseth != 0)
errx(1, "partition offset too high");
poffset = pp->p_offset;
poffset += SR_BOOT_LOADER_OFFSET;
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_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;
int fd, i;
u_char *p;
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");
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);
}
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);
}