#include <sys/param.h>
#include <sys/reboot.h>
#include <sys/disklabel.h>
#include <lib/libz/zlib.h>
#include <isofs/cd9660/iso.h>
#include "libsa.h"
#include <efi.h>
extern EFI_BOOT_SERVICES *BS;
extern int debug;
#include "disk.h"
#include "efidev.h"
#define EFI_BLKSPERSEC(_ed) ((_ed)->blkio->Media->BlockSize / DEV_BSIZE)
#define EFI_SECTOBLK(_ed, _n) ((_n) * EFI_BLKSPERSEC(_ed))
static EFI_STATUS
efid_io(int, efi_diskinfo_t, u_int, int, void *);
static int efid_diskio(int, struct diskinfo *, u_int, int, void *);
const char * efi_getdisklabel(efi_diskinfo_t, struct disklabel *);
static int efi_getdisklabel_cd9660(efi_diskinfo_t, struct disklabel *);
static u_int findopenbsd(efi_diskinfo_t, const char **);
static u_int findopenbsd_gpt(efi_diskinfo_t, const char **);
static int gpt_chk_mbr(struct dos_partition *, u_int64_t);
void
efid_init(struct diskinfo *dip, void *handle)
{
EFI_BLOCK_IO *blkio = handle;
memset(dip, 0, sizeof(struct diskinfo));
dip->ed.blkio = blkio;
dip->ed.mediaid = blkio->Media->MediaId;
dip->diskio = efid_diskio;
dip->strategy = efistrategy;
if (efi_getdisklabel(&dip->ed, &dip->disklabel) == NULL)
dip->flags |= DISKINFO_FLAG_GOODLABEL;
}
static EFI_STATUS
efid_io(int rw, efi_diskinfo_t ed, u_int off, int nsect, void *buf)
{
u_int blks, start, end;
EFI_PHYSICAL_ADDRESS addr;
EFI_STATUS status;
caddr_t data;
size_t size;
blks = EFI_BLKSPERSEC(ed);
if (blks == 0)
return (EFI_UNSUPPORTED);
start = off / blks;
end = (off + nsect + blks - 1) / blks;
size = (end - start) * ed->blkio->Media->BlockSize;
status = BS->AllocatePages(AllocateAnyPages, EfiLoaderData,
EFI_SIZE_TO_PAGES(size), &addr);
if (EFI_ERROR(status))
goto on_eio;
data = (caddr_t)(uintptr_t)addr;
switch (rw) {
case F_READ:
status = ed->blkio->ReadBlocks(ed->blkio, ed->mediaid, start,
size, data);
if (EFI_ERROR(status))
goto on_eio;
memcpy(buf, data + DEV_BSIZE * (off - start * blks),
DEV_BSIZE * nsect);
break;
case F_WRITE:
if (ed->blkio->Media->ReadOnly)
goto on_eio;
if (off % blks != 0 || nsect % blks != 0) {
status = ed->blkio->ReadBlocks(ed->blkio, ed->mediaid,
start, size, data);
if (EFI_ERROR(status))
goto on_eio;
}
memcpy(data + DEV_BSIZE * (off - start * blks), buf,
DEV_BSIZE * nsect);
status = ed->blkio->WriteBlocks(ed->blkio, ed->mediaid, start,
size, data);
if (EFI_ERROR(status))
goto on_eio;
break;
}
on_eio:
BS->FreePages(addr, EFI_SIZE_TO_PAGES(size));
return (status);
}
static int
efid_diskio(int rw, struct diskinfo *dip, u_int off, int nsect, void *buf)
{
EFI_STATUS status;
status = efid_io(rw, &dip->ed, off, nsect, buf);
return ((EFI_ERROR(status))? -1 : 0);
}
static int
gpt_chk_mbr(struct dos_partition *dp, u_int64_t dsize)
{
struct dos_partition *dp2;
int efi, found, i;
u_int32_t psize;
found = efi = 0;
for (dp2=dp, i=0; i < NDOSPART; i++, dp2++) {
if (dp2->dp_typ == DOSPTYP_UNUSED)
continue;
found++;
if (dp2->dp_typ != DOSPTYP_EFI)
continue;
if (letoh32(dp2->dp_start) != GPTSECTOR)
continue;
psize = letoh32(dp2->dp_size);
if (psize <= (dsize - GPTSECTOR) || psize == UINT32_MAX)
efi++;
}
if (found == 1 && efi == 1)
return (0);
return (1);
}
static u_int
findopenbsd(efi_diskinfo_t ed, const char **err)
{
EFI_STATUS status;
struct dos_mbr mbr;
struct dos_partition *dp;
u_int mbroff = DOSBBSECTOR;
u_int mbr_eoff = DOSBBSECTOR;
int i, maxebr = DOS_MAXEBR, nextebr;
again:
if (!maxebr--) {
*err = "too many extended partitions";
return (-1);
}
bzero(&mbr, sizeof(mbr));
status = efid_io(F_READ, ed, mbroff, 1, &mbr);
if (EFI_ERROR(status)) {
*err = "Disk I/O Error";
return (-1);
}
nextebr = 0;
for (i = 0; i < NDOSPART; i++) {
dp = &mbr.dmbr_parts[i];
if (!dp->dp_size)
continue;
#ifdef BIOS_DEBUG
if (debug)
printf("found partition %u: "
"type %u (0x%x) offset %u (0x%x)\n",
(int)(dp - mbr.dmbr_parts),
dp->dp_typ, dp->dp_typ,
dp->dp_start, dp->dp_start);
#endif
if (dp->dp_typ == DOSPTYP_OPENBSD) {
if (dp->dp_start > (dp->dp_start + mbroff))
continue;
return (dp->dp_start + mbroff);
}
if (!nextebr && (dp->dp_typ == DOSPTYP_EXTEND ||
dp->dp_typ == DOSPTYP_EXTENDL)) {
nextebr = dp->dp_start + mbr_eoff;
if (nextebr < dp->dp_start)
nextebr = (u_int)-1;
if (mbr_eoff == DOSBBSECTOR)
mbr_eoff = dp->dp_start;
}
}
if (nextebr && nextebr != (u_int)-1) {
mbroff = nextebr;
goto again;
}
return (-1);
}
static u_int
findopenbsd_gpt(efi_diskinfo_t ed, const char **err)
{
EFI_STATUS status;
struct gpt_header gh;
int i, part, found;
uint64_t lba;
uint32_t orig_csum, new_csum;
uint32_t ghsize, ghpartsize, ghpartnum, ghpartspersec;
uint32_t gpsectors;
const char openbsd_uuid_code[] = GPT_UUID_OPENBSD;
struct gpt_partition gp;
static struct uuid *openbsd_uuid = NULL, openbsd_uuid_space;
static u_char buf[4096];
if (openbsd_uuid == NULL) {
memcpy(&openbsd_uuid_space, openbsd_uuid_code,
sizeof(openbsd_uuid_space));
openbsd_uuid_space.time_low =
betoh32(openbsd_uuid_space.time_low);
openbsd_uuid_space.time_mid =
betoh16(openbsd_uuid_space.time_mid);
openbsd_uuid_space.time_hi_and_version =
betoh16(openbsd_uuid_space.time_hi_and_version);
openbsd_uuid = &openbsd_uuid_space;
}
if (EFI_BLKSPERSEC(ed) > 8) {
*err = "disk sector > 4096 bytes\n";
return (-1);
}
lba = GPTSECTOR;
status = efid_io(F_READ, ed, EFI_SECTOBLK(ed, lba), EFI_BLKSPERSEC(ed),
buf);
if (EFI_ERROR(status)) {
*err = "Disk I/O Error";
return (-1);
}
memcpy(&gh, buf, sizeof(gh));
if (letoh64(gh.gh_sig) != GPTSIGNATURE) {
*err = "bad GPT signature\n";
return (-1);
}
if (letoh32(gh.gh_rev) != GPTREVISION) {
*err = "bad GPT revision\n";
return (-1);
}
ghsize = letoh32(gh.gh_size);
if (ghsize < GPTMINHDRSIZE || ghsize > sizeof(struct gpt_header)) {
*err = "bad GPT header size\n";
return (-1);
}
orig_csum = gh.gh_csum;
gh.gh_csum = 0;
new_csum = crc32(0, (unsigned char *)&gh, ghsize);
gh.gh_csum = orig_csum;
if (letoh32(orig_csum) != new_csum) {
*err = "bad GPT header checksum\n";
return (-1);
}
lba = letoh64(gh.gh_part_lba);
ghpartsize = letoh32(gh.gh_part_size);
ghpartspersec = ed->blkio->Media->BlockSize / ghpartsize;
ghpartnum = letoh32(gh.gh_part_num);
gpsectors = (ghpartnum + ghpartspersec - 1) / ghpartspersec;
new_csum = crc32(0L, Z_NULL, 0);
found = 0;
for (i = 0; i < gpsectors; i++, lba++) {
status = efid_io(F_READ, ed, EFI_SECTOBLK(ed, lba),
EFI_BLKSPERSEC(ed), buf);
if (EFI_ERROR(status)) {
*err = "Disk I/O Error";
return (-1);
}
for (part = 0; part < ghpartspersec; part++) {
if (ghpartnum == 0)
break;
new_csum = crc32(new_csum, buf + part * sizeof(gp),
sizeof(gp));
ghpartnum--;
if (found)
continue;
memcpy(&gp, buf + part * sizeof(gp), sizeof(gp));
if (memcmp(&gp.gp_type, openbsd_uuid,
sizeof(struct uuid)) == 0)
found = 1;
}
}
if (new_csum != letoh32(gh.gh_part_csum)) {
*err = "bad GPT entries checksum\n";
return (-1);
}
if (found) {
lba = letoh64(gp.gp_lba_start);
if (lba > UINT_MAX || EFI_SECTOBLK(ed, lba) > UINT_MAX) {
*err = "OpenBSD Partition LBA > 2**32 - 1";
return (-1);
}
return (u_int)lba;
}
return (-1);
}
const char *
efi_getdisklabel(efi_diskinfo_t ed, struct disklabel *label)
{
u_int start = 0;
uint8_t buf[DEV_BSIZE];
struct dos_partition dosparts[NDOSPART];
EFI_STATUS status;
const char *err = NULL;
int error;
memset(buf, 0, sizeof(buf));
status = efid_io(F_READ, ed, DOSBBSECTOR, 1, buf);
if (EFI_ERROR(status))
return ("Disk I/O Error");
if (buf[510] != 0x55 || buf[511] != 0xaa) {
if (efi_getdisklabel_cd9660(ed, label) == 0)
return (NULL);
return ("invalid MBR signature");
}
memcpy(dosparts, buf+DOSPARTOFF, sizeof(dosparts));
if (gpt_chk_mbr(dosparts, ed->blkio->Media->LastBlock + 1) == 0) {
start = findopenbsd_gpt(ed, &err);
if (start == (u_int)-1) {
if (err != NULL)
return (err);
return ("no OpenBSD GPT partition");
}
} else {
start = findopenbsd(ed, &err);
if (start == (u_int)-1) {
if (err != NULL)
return (err);
return "no OpenBSD MBR partition\n";
}
}
#ifdef BIOS_DEBUG
if (debug)
printf("loading disklabel @ %u\n", start + DOS_LABELSECTOR);
#endif
error = efid_io(F_READ, ed, EFI_SECTOBLK(ed, start) + DOS_LABELSECTOR,
1, buf);
if (error)
return "failed to read disklabel";
return (getdisklabel(buf, label));
}
static int
efi_getdisklabel_cd9660(efi_diskinfo_t ed, struct disklabel *label)
{
uint8_t buf[DEV_BSIZE];
EFI_STATUS status;
status = efid_io(F_READ, ed, 64, 1, buf);
if (EFI_ERROR(status))
return -1;
if (buf[0] != ISO_VD_PRIMARY || bcmp(buf + 1, ISO_STANDARD_ID, 5) != 0)
return -1;
label->d_secsize = 2048;
label->d_ntracks = 1;
label->d_nsectors = 100;
label->d_ncylinders = 1;
label->d_secpercyl = label->d_ntracks * label->d_nsectors;
strncpy(label->d_typename, "ATAPI CD-ROM", sizeof(label->d_typename));
label->d_type = DTYPE_ATAPI;
strncpy(label->d_packname, "fictitious", sizeof(label->d_packname));
DL_SETDSIZE(label, 100);
DL_SETPOFFSET(&label->d_partitions[0], 0);
DL_SETPSIZE(&label->d_partitions[0], 100);
label->d_partitions[0].p_fstype = FS_UNUSED;
DL_SETPOFFSET(&label->d_partitions[RAW_PART], 0);
DL_SETPSIZE(&label->d_partitions[RAW_PART], 100);
label->d_partitions[RAW_PART].p_fstype = FS_UNUSED;
label->d_npartitions = MAXPARTITIONS;
label->d_magic = DISKMAGIC;
label->d_magic2 = DISKMAGIC;
label->d_checksum = dkcksum(label);
return (0);
}
int
efiopen(struct open_file *f, ...)
{
struct diskinfo *dip = NULL;
va_list ap;
u_int unit, part;
int i = 0;
va_start(ap, f);
unit = va_arg(ap, u_int);
part = va_arg(ap, u_int);
va_end(ap);
if (part >= MAXPARTITIONS)
return (ENXIO);
TAILQ_FOREACH(dip, &disklist, list) {
if (i == unit)
break;
i++;
}
if (dip == NULL)
return (ENXIO);
if ((dip->flags & DISKINFO_FLAG_GOODLABEL) == 0)
return (ENXIO);
dip->part = part;
bootdev_dip = dip;
f->f_devdata = dip;
return 0;
}
int
efistrategy(void *devdata, int rw, daddr_t blk, size_t size, void *buf,
size_t *rsize)
{
struct diskinfo *dip = (struct diskinfo *)devdata;
int error = 0;
size_t nsect;
nsect = (size + DEV_BSIZE - 1) / DEV_BSIZE;
blk += DL_SECTOBLK(&dip->disklabel,
dip->disklabel.d_partitions[dip->part].p_offset);
if (blk < 0)
error = EINVAL;
else
error = efid_diskio(rw, dip, blk, nsect, buf);
if (rsize != NULL)
*rsize = nsect * DEV_BSIZE;
return (error);
}
int
eficlose(struct open_file *f)
{
f->f_devdata = NULL;
return 0;
}
int
efiioctl(struct open_file *f, u_long cmd, void *data)
{
return 0;
}