diff options
author | Kenneth R Westerback <krw@cvs.openbsd.org> | 2015-08-24 23:03:12 +0000 |
---|---|---|
committer | Kenneth R Westerback <krw@cvs.openbsd.org> | 2015-08-24 23:03:12 +0000 |
commit | 9653a62608467bfc6a6065591822692d9a32177e (patch) | |
tree | 98365b772423500a87407f24436357912c782ad3 | |
parent | ec1c24011d8acaf0a39824f31ead16d43933b2c4 (diff) |
Fix GPT code to work with non-DEV_BSIZE disks.
Part of GPT refactoring that was backed out at c2k15.
ok deraadt@
-rw-r--r-- | sys/kern/subr_disk.c | 38 |
1 files changed, 17 insertions, 21 deletions
diff --git a/sys/kern/subr_disk.c b/sys/kern/subr_disk.c index adc5de72876..4fc92de0bba 100644 --- a/sys/kern/subr_disk.c +++ b/sys/kern/subr_disk.c @@ -1,4 +1,4 @@ -/* $OpenBSD: subr_disk.c,v 1.188 2015/08/12 22:37:32 krw Exp $ */ +/* $OpenBSD: subr_disk.c,v 1.189 2015/08/24 23:03:11 krw Exp $ */ /* $NetBSD: subr_disk.c,v 1.17 1996/03/16 23:17:08 christos Exp $ */ /* @@ -647,15 +647,12 @@ readgptlabel(struct buf *bp, void (*strat)(struct buf *), if (lp->d_secsize == 0) return (ENOSPC); /* disk too small */ - /* - * XXX: We should not trust the primary header and instead - * use the last LBA of the disk, as defined in the standard. - */ - for (part_blkno = GPTSECTOR; ; part_blkno = gh.gh_lba_alt, - altheader = 1) { + for (part_blkno = DL_SECTOBLK(lp, GPTSECTOR); ; + part_blkno = DL_SECTOBLK(lp, DL_GETDSIZE(lp)-1), altheader = 1) { uint32_t ghsize; uint32_t ghpartsize; uint32_t ghpartnum; + uint32_t ghpartspersec; /* read header record */ bp->b_blkno = DL_BLKTOSEC(lp, part_blkno) * DL_BLKSPERSEC(lp); @@ -677,6 +674,7 @@ readgptlabel(struct buf *bp, void (*strat)(struct buf *), bcopy(bp->b_data + offset, &gh, sizeof(gh)); ghsize = letoh32(gh.gh_size); ghpartsize = letoh32(gh.gh_part_size); + ghpartspersec = lp->d_secsize / ghpartsize; ghpartnum = letoh32(gh.gh_part_num); if (letoh64(gh.gh_sig) != GPTSIGNATURE) @@ -706,7 +704,7 @@ readgptlabel(struct buf *bp, void (*strat)(struct buf *), * Header size must be greater than or equal to 92 and less * than or equal to the logical block size. */ - if (ghsize < GPTMINHDRSIZE || ghsize > DEV_BSIZE) + if (ghsize < GPTMINHDRSIZE || ghsize > lp->d_secsize) return (EINVAL); if (letoh64(gh.gh_lba_start) >= DL_GETDSIZE(lp) || @@ -718,33 +716,32 @@ readgptlabel(struct buf *bp, void (*strat)(struct buf *), * Size per partition entry shall be 128*(2**n) with n >= 0. * We don't support partition entries larger than block size. */ - if (ghpartsize % GPTMINPARTSIZE - || ghpartsize > DEV_BSIZE - || GPT_PARTSPERSEC(&gh) == 0) { + if (ghpartsize % GPTMINPARTSIZE || ghpartsize > lp->d_secsize + || ghpartspersec == 0) { DPRINTF("invalid partition size\n"); return (EINVAL); } /* XXX: we don't support multiples of GPTMINPARTSIZE yet */ - if (letoh32(gh.gh_part_size) != GPTMINPARTSIZE) { + if (ghpartsize != GPTMINPARTSIZE) { DPRINTF("partition sizes larger than %d bytes are not " "supported", GPTMINPARTSIZE); return (EINVAL); } /* read GPT partition entry array */ - gp = mallocarray(ghpartnum, sizeof(struct gpt_partition), M_DEVBUF, M_NOWAIT|M_ZERO); + gp = mallocarray(ghpartnum, sizeof(struct gpt_partition), + M_DEVBUF, M_NOWAIT|M_ZERO); if (gp == NULL) return (ENOMEM); gpsz = ghpartnum * sizeof(struct gpt_partition); /* - * XXX: Fails if # of partition entries is no multiple of - * GPT_PARTSPERSEC(&gh) + * XXX: Fails if # of partition entries is not a multiple of + * ghpartspersec. */ - for (i = 0; i < ghpartnum / GPT_PARTSPERSEC(&gh); - i++) { - part_blkno = letoh64(gh.gh_part_lba) + i; + for (i = 0; i < ghpartnum / ghpartspersec; i++) { + part_blkno = DL_SECTOBLK(lp, letoh64(gh.gh_part_lba)+i); /* read partition record */ bp->b_blkno = DL_BLKTOSEC(lp, part_blkno) * DL_BLKSPERSEC(lp); @@ -763,9 +760,8 @@ readgptlabel(struct buf *bp, void (*strat)(struct buf *), return (error); } - bcopy(bp->b_data + offset, gp + - i * GPT_PARTSPERSEC(&gh), GPT_PARTSPERSEC(&gh) * - sizeof(struct gpt_partition)); + bcopy(bp->b_data + offset, gp + i * ghpartspersec, + ghpartspersec * sizeof(struct gpt_partition)); } if (gpt_chk_parts(&gh, gp)) { |