/* $OpenBSD: mbr.c,v 1.105 2021/11/09 14:02:09 krw Exp $ */ /* * Copyright (c) 1997 Tobias Weingartner * * 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 /* DEV_BSIZE */ #include #include #include #include #include #include #include #include "part.h" #include "disk.h" #include "misc.h" #include "mbr.h" #include "gpt.h" struct dos_mbr default_dmbr; void mbr_to_dos_mbr(const struct mbr *, struct dos_mbr *); void dos_mbr_to_mbr(const struct dos_mbr *, const uint64_t, const uint64_t, struct mbr *); void MBR_init(struct mbr *mbr) { struct dos_partition dp; struct prt bootprt, obsdprt; uint64_t adj; daddr_t daddr; memset(&gmbr, 0, sizeof(gmbr)); memset(&gh, 0, sizeof(gh)); memset(&gp, 0, sizeof(gp)); if (mbr->mbr_lba_self != 0) { /* Extended MBR - save lba's, set sig, zap everything else. */ memset(mbr->mbr_code, 0, sizeof(mbr->mbr_code)); memset(mbr->mbr_prt, 0, sizeof(mbr->mbr_prt)); mbr->mbr_signature = DOSMBR_SIGNATURE; return; } memset(&obsdprt, 0, sizeof(obsdprt)); memset(&bootprt, 0, sizeof(bootprt)); memcpy(&dp, &default_dmbr.dmbr_parts[0], sizeof(dp)); PRT_parse(&dp, 0, 0, &bootprt); if (bootprt.prt_flag != DOSACTIVE) obsdprt.prt_flag = DOSACTIVE; /* Use whole disk. Reserve first track, or first cyl, if possible. */ obsdprt.prt_id = DOSPTYP_OPENBSD; if (disk.dk_heads > 1) obsdprt.prt_shead = 1; else obsdprt.prt_shead = 0; if (disk.dk_heads < 2 && disk.dk_cylinders > 1) obsdprt.prt_scyl = 1; else obsdprt.prt_scyl = 0; obsdprt.prt_ssect = 1; /* Go right to the end */ obsdprt.prt_ecyl = disk.dk_cylinders - 1; obsdprt.prt_ehead = disk.dk_heads - 1; obsdprt.prt_esect = disk.dk_sectors; /* Fix up start/length fields */ PRT_fix_BN(&obsdprt, 3); #if defined(__powerpc__) || defined(__mips__) /* Now fix up for the MS-DOS boot partition on PowerPC/MIPS. */ bootprt.prt_flag = DOSACTIVE; /* Boot from dos part */ bootprt.prt_flag = 0; obsdprt.prt_ns += obsdprt.prt_bs; obsdprt.prt_bs = bootprt.prt_bs + bootprt.prt_ns; obsdprt.prt_ns -= obsdprt.prt_bs; PRT_fix_CHS(&obsdprt); if ((obsdprt.prt_shead != 1) || (obsdprt.prt_ssect != 1)) { /* align the partition on a cylinder boundary */ obsdprt.prt_shead = 0; obsdprt.prt_ssect = 1; obsdprt.prt_scyl += 1; } /* Fix up start/length fields */ PRT_fix_BN(&obsdprt, 3); #else if (disk.dk_bootprt.prt_ns > 0) { bootprt = disk.dk_bootprt; PRT_fix_CHS(&bootprt); obsdprt.prt_ns += obsdprt.prt_bs; obsdprt.prt_bs = bootprt.prt_bs + bootprt.prt_ns; obsdprt.prt_ns -= obsdprt.prt_bs; PRT_fix_CHS(&obsdprt); } #endif /* Start OpenBSD MBR partition on a power of 2 block number. */ daddr = 1; while (daddr < DL_SECTOBLK(&dl, obsdprt.prt_bs)) daddr *= 2; adj = DL_BLKTOSEC(&dl, daddr) - obsdprt.prt_bs; obsdprt.prt_bs += adj; obsdprt.prt_ns -= adj; PRT_fix_CHS(&obsdprt); memset(mbr, 0, sizeof(*mbr)); memcpy(mbr->mbr_code, default_dmbr.dmbr_boot, sizeof(mbr->mbr_code)); mbr->mbr_prt[0] = bootprt; mbr->mbr_prt[3] = obsdprt; mbr->mbr_signature = DOSMBR_SIGNATURE; } void dos_mbr_to_mbr(const struct dos_mbr *dmbr, const uint64_t lba_self, const uint64_t lba_firstembr, struct mbr *mbr) { struct dos_partition dos_parts[NDOSPART]; int i; memcpy(mbr->mbr_code, dmbr->dmbr_boot, sizeof(mbr->mbr_code)); mbr->mbr_lba_self = lba_self; mbr->mbr_lba_firstembr = lba_firstembr; mbr->mbr_signature = letoh16(dmbr->dmbr_sign); memcpy(dos_parts, dmbr->dmbr_parts, sizeof(dos_parts)); for (i = 0; i < NDOSPART; i++) PRT_parse(&dos_parts[i], lba_self, lba_firstembr, &mbr->mbr_prt[i]); } void mbr_to_dos_mbr(const struct mbr *mbr, struct dos_mbr *dos_mbr) { struct dos_partition dos_partition; int i; memcpy(dos_mbr->dmbr_boot, mbr->mbr_code, sizeof(dos_mbr->dmbr_boot)); dos_mbr->dmbr_sign = htole16(DOSMBR_SIGNATURE); for (i = 0; i < NDOSPART; i++) { PRT_make(&mbr->mbr_prt[i], mbr->mbr_lba_self, mbr->mbr_lba_firstembr, &dos_partition); memcpy(&dos_mbr->dmbr_parts[i], &dos_partition, sizeof(dos_mbr->dmbr_parts[i])); } } void MBR_print(const struct mbr *mbr, const char *units) { int i; DISK_printgeometry("s"); printf("Offset: %lld\t", (long long)mbr->mbr_lba_self); printf("Signature: 0x%X\n", (int)mbr->mbr_signature); PRT_print(0, NULL, units); for (i = 0; i < NDOSPART; i++) PRT_print(i, &mbr->mbr_prt[i], units); } int MBR_read(const uint64_t lba_self, const uint64_t lba_firstembr, struct mbr *mbr) { struct dos_mbr dos_mbr; char *secbuf; secbuf = DISK_readsectors(lba_self, 1); if (secbuf == NULL) return -1; memcpy(&dos_mbr, secbuf, sizeof(dos_mbr)); free(secbuf); dos_mbr_to_mbr(&dos_mbr, lba_self, lba_firstembr, mbr); return 0; } int MBR_write(const struct mbr *mbr) { struct dos_mbr dos_mbr; char *secbuf; int rslt; secbuf = DISK_readsectors(mbr->mbr_lba_self, 1); if (secbuf == NULL) return -1; mbr_to_dos_mbr(mbr, &dos_mbr); memcpy(secbuf, &dos_mbr, sizeof(dos_mbr)); rslt = DISK_writesectors(secbuf, mbr->mbr_lba_self, 1); free(secbuf); if (rslt) return -1; /* Refresh in-kernel disklabel from the updated disk information. */ if (ioctl(disk.dk_fd, DIOCRLDINFO, 0) == -1) warn("DIOCRLDINFO"); return 0; }