2 * Copyright (c) 2007,2020 The DragonFly Project. All rights reserved.
4 * This code is derived from software contributed to The DragonFly Project
5 * by Matthew Dillon <dillon@backplane.com>
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in
15 * the documentation and/or other materials provided with the
17 * 3. Neither the name of The DragonFly Project nor the names of its
18 * contributors may be used to endorse or promote products derived
19 * from this software without specific, prior written permission.
21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25 * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26 * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
27 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
29 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
30 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
31 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
35 * Copyright (c) 1987, 1993
36 * The Regents of the University of California. All rights reserved.
38 * This code is derived from software contributed to Berkeley by
39 * Symmetric Computer Systems.
41 * Redistribution and use in source and binary forms, with or without
42 * modification, are permitted provided that the following conditions
44 * 1. Redistributions of source code must retain the above copyright
45 * notice, this list of conditions and the following disclaimer.
46 * 2. Redistributions in binary form must reproduce the above copyright
47 * notice, this list of conditions and the following disclaimer in the
48 * documentation and/or other materials provided with the distribution.
49 * 3. Neither the name of the University nor the names of its contributors
50 * may be used to endorse or promote products derived from this software
51 * without specific prior written permission.
53 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
54 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
55 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
56 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
57 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
58 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
59 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
60 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
61 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
62 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
65 * @(#)disklabel.c 1.2 (Symmetric) 11/28/85
66 * @(#)disklabel.c 8.2 (Berkeley) 1/7/94
67 * $FreeBSD: src/sbin/disklabel/disklabel.c,v 1.28.2.15 2003/01/24 16:18:16 des Exp $
70 #include <sys/param.h>
74 #include <sys/disklabel64.h>
75 #include <sys/diskslice.h>
76 #include <sys/diskmbr.h>
77 #include <sys/dtype.h>
78 #include <sys/sysctl.h>
83 #include <vfs/ufs/dinode.h>
84 #include <vfs/ufs/fs.h>
97 #include "pathnames.h"
99 extern uint32_t crc32(const void *buf
, size_t size
);
101 static void makelabel(const char *, const char *, struct disklabel64
*);
102 static int writelabel(int, struct disklabel64
*);
103 static int expandlabel(int f
, struct disklabel64
*lp
, int *wbackp
);
104 static void l_perror(const char *);
105 static void display(FILE *, const struct disklabel64
*);
106 static int edit(struct disklabel64
*, int);
107 static int editit(void);
108 static char *skip(char *);
109 static char *word(char *);
110 static int parse_field_val(char **, char **, u_int64_t
*, int);
111 static int getasciilabel(FILE *, struct disklabel64
*);
112 static int getasciipartspec(char *, struct disklabel64
*, int, int, uint32_t);
113 static int getasciipartuuid(char *, struct disklabel64
*, int, int, uint32_t);
114 static int checklabel(struct disklabel64
*);
115 static void Warning(const char *, ...) __printflike(1, 2);
116 static void usage(void);
117 static struct disklabel64
*getvirginlabel(void);
118 static struct disklabel64
*readlabel(int);
119 static struct disklabel64
*makebootarea(int);
121 #define DEFEDITOR _PATH_VI
122 #define streq(a,b) (strcmp(a,b) == 0)
125 static char *specname
;
126 static char tmpfil
[] = PATH_TMPFILE
;
128 static struct disklabel64 lab
;
130 #define MAX_PART ('z')
131 #define MAX_NUM_PARTS (1 + MAX_PART - 'a')
132 static char part_size_type
[MAX_NUM_PARTS
];
133 static char part_offset_type
[MAX_NUM_PARTS
];
134 static int part_set
[MAX_NUM_PARTS
];
136 static int installboot
;
137 static int expandopt
;
138 static int boot1size
;
139 static int boot1lsize
;
140 static int boot2size
;
141 static char *boot1buf
;
142 static char *boot2buf
;
143 static char *boot1path
;
144 static char *boot2path
;
147 UNSPEC
, EDIT
, NOWRITE
, READ
, RESTORE
, WRITE
, WRITEABLE
, WRITEBOOT
,
153 static int disable_write
; /* set to disable writing to disk label */
157 #define OPTIONS "BNRWb:denrs:Vwx"
159 #define OPTIONS "BNRWb:enrs:Vwx"
163 main(int argc
, char *argv
[])
165 struct disklabel64
*lp
;
167 int ch
, f
= 0, flag
, error
= 0;
168 const char *name
= NULL
;
169 const char *dtype
= NULL
;
171 while ((ch
= getopt(argc
, argv
, OPTIONS
)) != -1)
245 dkname
= getdevpath(argv
[0], 0);
247 f
= open(specname
, op
== READ
? O_RDONLY
: O_RDWR
);
249 err(4, "%s", specname
);
265 if (ioctl(f
, DIOCWLABEL
, (char *)&flag
) < 0)
266 err(4, "ioctl DIOCWLABEL");
274 error
= checklabel(lp
);
282 error
= expandlabel(f
, lp
, &wback
);
283 if (checklabel(lp
) == 0 && wback
)
284 error
= writelabel(f
, lp
);
289 if (installboot
&& argc
== 3) {
290 makelabel(argv
[2], 0, &lab
);
294 * We only called makelabel() for its side effect
295 * of setting the bootstrap file names. Discard
296 * all changes to `lab' so that all values in the
297 * final label come from the ASCII label.
299 bzero((char *)&lab
, sizeof(lab
));
303 if (!(t
= fopen(argv
[1], "r")))
304 err(4, "%s", argv
[1]);
305 if (!getasciilabel(t
, &lab
))
307 lp
= makebootarea(f
);
308 bcopy(&lab
.d_magic
, &lp
->d_magic
,
309 sizeof(lab
) - offsetof(struct disklabel64
, d_magic
));
310 error
= writelabel(f
, lp
);
324 makelabel(dtype
, name
, &lab
);
325 lp
= makebootarea(f
);
326 bcopy(&lab
.d_magic
, &lp
->d_magic
,
327 sizeof(lab
) - offsetof(struct disklabel64
, d_magic
));
328 if (checklabel(lp
) == 0)
329 error
= writelabel(f
, lp
);
334 if (ioctl(f
, DIOCWLABEL
, (char *)&flag
) < 0)
335 err(4, "ioctl DIOCWLABEL");
340 struct disklabel64 tlab
;
345 makelabel(argv
[1], 0, &lab
);
346 lp
= makebootarea(f
);
347 bcopy(&tlab
.d_magic
, &lp
->d_magic
,
348 sizeof(tlab
) - offsetof(struct disklabel64
, d_magic
));
349 if (checklabel(lp
) == 0)
350 error
= writelabel(f
, lp
);
358 * Construct a prototype disklabel from /etc/disktab. As a side
359 * effect, set the names of the primary and secondary boot files
363 makelabel(const char *type
, const char *name
, struct disklabel64
*lp
)
365 struct disklabel64
*dp
;
367 if (streq(type
, "auto"))
368 dp
= getvirginlabel();
370 errx(1, "no disktab(5) support yet; only 'auto' allowed");
374 strlcpy((char *)lp
->d_packname
, name
, sizeof(lp
->d_packname
));
378 writelabel(int f
, struct disklabel64
*lp
)
380 struct disklabel64
*blp
;
386 lpsize
= offsetof(struct disklabel64
, d_partitions
[lp
->d_npartitions
]);
387 lpcrcsize
= lpsize
- offsetof(struct disklabel64
, d_magic
);
390 Warning("write to disk label suppressed - label was as follows:");
394 lp
->d_magic
= DISKMAGIC64
;
396 lp
->d_crc
= crc32(&lp
->d_magic
, lpcrcsize
);
399 * Make sure the boot area is not too large
402 int lpbsize
= (int)(lp
->d_pbase
- lp
->d_bbase
);
403 if (lp
->d_pbase
== 0) {
404 errx(1, "no space was set aside in "
405 "the disklabel for boot2!");
407 if (boot2size
> lpbsize
) {
408 errx(1, "label did not reserve enough "
409 "space for boot! %d/%d",
415 * First set the kernel disk label,
416 * then write a label to the raw disk.
417 * If the SDINFO ioctl fails because it is
418 * unimplemented, keep going; otherwise, the kernel
419 * consistency checks may prevent us from changing
420 * the current (in-core) label.
422 if (ioctl(f
, DIOCSDINFO64
, lp
) < 0 &&
423 errno
!= ENODEV
&& errno
!= ENOTTY
) {
424 l_perror("ioctl DIOCSDINFO");
427 lseek(f
, (off_t
)0, SEEK_SET
);
430 * The disklabel embeds areas which we may not
431 * have wanted to change. Merge those areas in
434 blp
= makebootarea(f
);
436 bcopy(&lp
->d_magic
, &blp
->d_magic
,
438 offsetof(struct disklabel64
, d_magic
));
442 * write enable label sector before write
443 * (if necessary), disable after writing.
446 if (ioctl(f
, DIOCWLABEL
, &flag
) < 0)
447 warn("ioctl DIOCWLABEL");
449 r
= write(f
, boot1buf
, boot1lsize
);
450 if (r
!= (ssize_t
)boot1lsize
) {
455 * Output the remainder of the disklabel
458 lseek(f
, lp
->d_bbase
, 0);
459 r
= write(f
, boot2buf
, boot2size
);
460 if (r
!= boot2size
) {
466 ioctl(f
, DIOCWLABEL
, &flag
);
467 } else if (ioctl(f
, DIOCWDINFO64
, lp
) < 0) {
468 l_perror("ioctl DIOCWDINFO64");
476 l_perror(const char *s
)
481 warnx("%s: no disk label on disk;", s
);
482 fprintf(stderr
, "add \"-r\" to install initial label\n");
486 warnx("%s: label magic number or checksum is wrong!", s
);
487 fprintf(stderr
, "(disklabel or kernel is out of date?)\n");
491 warnx("%s: open partition would move or shrink", s
);
495 warnx("%s: the disk already has a label of a different type,\n"
496 "probably a 32 bit disklabel. It must be cleaned out "
507 * Fetch disklabel for disk.
508 * Use ioctl to get label unless -r flag is given.
510 static struct disklabel64
*
513 struct disklabel64
*lp
;
519 * Allocate space for the label. The boot1 code, if any,
520 * is embedded in the label. The label overlaps the boot1
523 lp
= makebootarea(f
);
524 lpcrcsize
= offsetof(struct disklabel64
,
525 d_partitions
[lp
->d_npartitions
]) -
526 offsetof(struct disklabel64
, d_magic
);
529 if (lp
->d_magic
!= DISKMAGIC64
)
530 errx(1, "bad pack magic number");
531 if (lp
->d_npartitions
> MAXPARTITIONS64
||
532 savecrc
!= crc32(&lp
->d_magic
, lpcrcsize
)
534 errx(1, "corrupted disklabel64");
539 * Just use a static structure to hold the label. Note
540 * that DIOCSDINFO64 does not overwrite the boot1 area
541 * even though it is part of the disklabel64 structure.
545 if (ioctl(f
, DIOCGDVIRGIN64
, lp
) < 0) {
546 l_perror("ioctl DIOCGDVIRGIN64");
550 if (ioctl(f
, DIOCGDINFO64
, lp
) < 0) {
551 l_perror("ioctl DIOCGDINFO64");
560 * Construct a boot area for boot1 and boot2 and return the location of
561 * the label within the area. The caller will overwrite the label so
562 * we don't actually have to read it.
564 static struct disklabel64
*
567 struct disklabel64
*lp
;
568 struct partinfo info
;
574 if (ioctl(f
, DIOCGPART
, &info
) == 0)
575 secsize
= info
.media_blksize
;
579 if (boot1buf
== NULL
) {
582 rsize
= roundup2(sizeof(struct disklabel64
), secsize
);
583 boot1size
= offsetof(struct disklabel64
, d_magic
);
585 boot1buf
= malloc(rsize
);
586 bzero(boot1buf
, rsize
);
587 r
= read(f
, boot1buf
, rsize
);
588 if (r
!= (int)rsize
) {
590 err(4, "%s", specname
);
593 lp
= (void *)boot1buf
;
595 if (installboot
== 0)
598 if (boot2buf
== NULL
) {
599 boot2size
= BOOT2SIZE64
;
600 boot2buf
= malloc(boot2size
);
601 bzero(boot2buf
, boot2size
);
605 * If installing the boot code, read it into the appropriate portions
608 if (boot1path
== NULL
)
609 asprintf(&boot1path
, "%s/boot1_64", _PATH_BOOTDIR
);
610 if (boot2path
== NULL
)
611 asprintf(&boot2path
, "%s/boot2_64", _PATH_BOOTDIR
);
613 if ((fd
= open(boot1path
, O_RDONLY
)) < 0)
614 err(4, "%s", boot1path
);
615 if (fstat(fd
, &st
) < 0)
616 err(4, "%s", boot1path
);
617 if (st
.st_size
> boot1size
)
618 err(4, "%s must be exactly %d bytes!", boot1path
, boot1size
);
619 if (read(fd
, boot1buf
, boot1size
) != boot1size
)
620 err(4, "%s must be exactly %d bytes!", boot1path
, boot1size
);
623 if ((fd
= open(boot2path
, O_RDONLY
)) < 0)
624 err(4, "%s", boot2path
);
625 if (fstat(fd
, &st
) < 0)
626 err(4, "%s", boot2path
);
627 if (st
.st_size
> boot2size
)
628 err(4, "%s must be <= %d bytes!", boot2path
, boot2size
);
629 if ((r
= read(fd
, boot2buf
, boot2size
)) < 1)
630 err(4, "%s is empty!", boot2path
);
631 boot2size
= roundup2(r
, secsize
);
635 * XXX dangerously dedicated support goes here XXX
641 display(FILE *f
, const struct disklabel64
*lp
)
643 const struct partition64
*pp
;
650 * Use a human readable block size if possible. This is for
651 * display and editing purposes only.
653 if (lp
->d_align
> 1024)
656 blksize
= lp
->d_align
;
658 fprintf(f
, "# %s:\n", specname
);
660 fprintf(f
, "# Calculated informational fields for the slice:\n");
662 fprintf(f
, "# boot space: %10ju bytes\n",
663 (uintmax_t)(lp
->d_pbase
- lp
->d_bbase
));
664 fprintf(f
, "# data space: %10ju blocks\t# %6.2f MB (%ju bytes)\n",
665 (uintmax_t)(lp
->d_pstop
- lp
->d_pbase
) / blksize
,
666 (double)(lp
->d_pstop
- lp
->d_pbase
) / 1024.0 / 1024.0,
667 (uintmax_t)(lp
->d_pstop
- lp
->d_pbase
));
669 fprintf(f
, "# NOTE: The partition data base and stop are physically\n");
670 fprintf(f
, "# aligned instead of slice-relative aligned.\n");
672 fprintf(f
, "# All byte equivalent offsets must be aligned.\n");
675 uuid_to_string(&lp
->d_stor_uuid
, &str
, NULL
);
676 fprintf(f
, "diskid: %s\n", str
? str
: "<unknown>");
679 fprintf(f
, "label: %s\n", lp
->d_packname
);
680 fprintf(f
, "boot2 data base: 0x%012jx\n", (uintmax_t)lp
->d_bbase
);
681 fprintf(f
, "partitions data base: 0x%012jx\n", (uintmax_t)lp
->d_pbase
);
682 fprintf(f
, "partitions data stop: 0x%012jx\n", (uintmax_t)lp
->d_pstop
);
683 fprintf(f
, "backup label: 0x%012jx\n", (uintmax_t)lp
->d_abase
);
684 fprintf(f
, "total size: 0x%012jx\t# %6.2f MB\n",
685 (uintmax_t)lp
->d_total_size
,
686 (double)lp
->d_total_size
/ 1024.0 / 1024.0);
687 fprintf(f
, "alignment: %u\n", lp
->d_align
);
688 fprintf(f
, "display block size: %u\t# for partition display and edit only\n",
692 fprintf(f
, "%u partitions:\n", lp
->d_npartitions
);
693 fprintf(f
, "# size offset fstype fsuuid\n");
695 for (part
= 0; part
< lp
->d_npartitions
; part
++) {
696 pp
= &lp
->d_partitions
[part
];
697 const u_long onemeg
= 1024 * 1024;
699 if (pp
->p_bsize
== 0)
702 fprintf(f
, " %c: ", 'a' + part
);
704 if (pp
->p_bsize
% lp
->d_align
)
705 fprintf(f
, "%10s ", "ILLEGAL");
707 fprintf(f
, "%10ju ", (uintmax_t)pp
->p_bsize
/ blksize
);
709 if ((pp
->p_boffset
- lp
->d_pbase
) % lp
->d_align
)
710 fprintf(f
, "%10s ", "ILLEGAL");
713 (uintmax_t)(pp
->p_boffset
- lp
->d_pbase
) / blksize
);
715 if (pp
->p_fstype
< FSMAXTYPES
)
716 fprintf(f
, "%8.8s", fstypenames
[pp
->p_fstype
]);
718 fprintf(f
, "%8d", pp
->p_fstype
);
719 fprintf(f
, "\t# %11.3fMB", (double)pp
->p_bsize
/ onemeg
);
722 for (part
= 0; part
< lp
->d_npartitions
; part
++) {
723 pp
= &lp
->d_partitions
[part
];
725 if (pp
->p_bsize
== 0)
728 if (uuid_is_nil(&lp
->d_stor_uuid
, NULL
) == 0) {
729 fprintf(f
, " %c-stor_uuid: ", 'a' + part
);
731 uuid_to_string(&pp
->p_stor_uuid
, &str
, NULL
);
733 fprintf(f
, "%s", str
);
740 fprintf(f
, "# EXAMPLE\n");
741 fprintf(f
, "#a: 4g 0 4.2BSD\n");
742 fprintf(f
, "#a: * * 4.2BSD\n");
749 edit(struct disklabel64
*lp
, int f
)
752 struct disklabel64 label
;
755 if ((fd
= mkstemp(tmpfil
)) == -1 ||
756 (fp
= fdopen(fd
, "w")) == NULL
) {
757 warnx("can't create %s", tmpfil
);
765 fp
= fopen(tmpfil
, "r");
767 warnx("can't reopen %s for reading", tmpfil
);
770 bzero((char *)&label
, sizeof(label
));
771 if (getasciilabel(fp
, &label
)) {
773 if (writelabel(f
, lp
) == 0) {
780 printf("re-edit the label? [y]: "); fflush(stdout
);
782 if (c
!= EOF
&& c
!= (int)'\n')
783 while (getchar() != (int)'\n')
799 omask
= sigblock(sigmask(SIGINT
)|sigmask(SIGQUIT
)|sigmask(SIGHUP
));
800 while ((pid
= fork()) < 0) {
801 if (errno
== EPROCLIM
) {
802 warnx("you have too many processes");
805 if (errno
!= EAGAIN
) {
815 if ((ed
= getenv("EDITOR")) == NULL
)
817 execlp(ed
, ed
, tmpfil
, NULL
);
820 while ((xpid
= wait(&status
)) >= 0)
831 while (*cp
!= '\0' && isspace(*cp
))
833 if (*cp
== '\0' || *cp
== '#')
843 while (*cp
!= '\0' && !isspace(*cp
) && *cp
!= '#')
845 if ((c
= *cp
) != '\0') {
854 * Read an ascii label in from fd f,
855 * in the same format as that put out by display(),
859 getasciilabel(FILE *f
, struct disklabel64
*lp
)
863 char *tp
, line
[BUFSIZ
];
865 uint32_t blksize
= 0;
867 int lineno
= 0, errors
= 0;
870 bzero(&part_set
, sizeof(part_set
));
871 bzero(&part_size_type
, sizeof(part_size_type
));
872 bzero(&part_offset_type
, sizeof(part_offset_type
));
873 while (fgets(line
, sizeof(line
) - 1, f
)) {
875 if ((cp
= strchr(line
,'\n')) != NULL
)
880 tp
= strchr(cp
, ':');
882 fprintf(stderr
, "line %d: syntax error\n", lineno
);
886 *tp
++ = '\0', tp
= skip(tp
);
887 if (sscanf(cp
, "%lu partitions", &v
) == 1) {
888 if (v
== 0 || v
> MAXPARTITIONS64
) {
890 "line %d: bad # of partitions\n", lineno
);
891 lp
->d_npartitions
= MAXPARTITIONS64
;
894 lp
->d_npartitions
= v
;
900 if (streq(cp
, "diskid")) {
902 uuid_from_string(tp
, &lp
->d_stor_uuid
, &status
);
903 if (status
!= uuid_s_ok
) {
905 "line %d: %s: illegal UUID\n",
911 if (streq(cp
, "label")) {
912 strlcpy((char *)lp
->d_packname
, tp
, sizeof(lp
->d_packname
));
916 if (streq(cp
, "alignment")) {
917 v
= strtoul(tp
, NULL
, 0);
918 if (v
<= 0 || (v
& DEV_BMASK
) != 0 || v
> 1024*1024) {
920 "line %d: %s: bad alignment\n",
928 if (streq(cp
, "total size")) {
929 vv
= strtoull(tp
, NULL
, 0);
930 if (vv
== 0 || vv
== (uint64_t)-1) {
931 fprintf(stderr
, "line %d: %s: bad %s\n",
935 lp
->d_total_size
= vv
;
939 if (streq(cp
, "boot2 data base")) {
940 vv
= strtoull(tp
, NULL
, 0);
941 if (vv
== 0 || vv
== (uint64_t)-1) {
942 fprintf(stderr
, "line %d: %s: bad %s\n",
950 if (streq(cp
, "partitions data base")) {
951 vv
= strtoull(tp
, NULL
, 0);
952 if (vv
== 0 || vv
== (uint64_t)-1) {
953 fprintf(stderr
, "line %d: %s: bad %s\n",
961 if (streq(cp
, "partitions data stop")) {
962 vv
= strtoull(tp
, NULL
, 0);
963 if (vv
== 0 || vv
== (uint64_t)-1) {
964 fprintf(stderr
, "line %d: %s: bad %s\n",
972 if (streq(cp
, "backup label")) {
973 vv
= strtoull(tp
, NULL
, 0);
974 if (vv
== 0 || vv
== (uint64_t)-1) {
975 fprintf(stderr
, "line %d: %s: bad %s\n",
983 if (streq(cp
, "display block size")) {
984 v
= strtoul(tp
, NULL
, 0);
985 if (v
<= 0 || (v
& DEV_BMASK
) != 0 || v
> 1024*1024) {
986 fprintf(stderr
, "line %d: %s: bad %s\n",
995 /* the ':' was removed above */
998 * Handle main partition data, e.g. a:, b:, etc.
1000 if (*cp
< 'a' || *cp
> MAX_PART
) {
1002 "line %d: %s: Unknown disklabel field\n", lineno
,
1008 /* Process a partition specification line. */
1010 if (part
>= lp
->d_npartitions
) {
1012 "line %d: partition name out of range a-%c: %s\n",
1013 lineno
, 'a' + lp
->d_npartitions
- 1, cp
);
1019 fprintf(stderr
, "block size to use for partition "
1020 "display was not specified!\n");
1025 if (strcmp(cp
+ 1, "-stor_uuid") == 0) {
1026 if (getasciipartuuid(tp
, lp
, part
, lineno
, blksize
)) {
1031 } else if (cp
[1] == 0) {
1033 if (getasciipartspec(tp
, lp
, part
, lineno
, blksize
)) {
1039 fprintf(stderr
, "line %d: %s: Unknown disklabel field\n",
1044 errors
+= checklabel(lp
);
1045 return (errors
== 0);
1049 parse_field_val(char **tp
, char **cp
, u_int64_t
*vv
, int lineno
)
1053 if (*tp
== NULL
|| **tp
== 0) {
1054 fprintf(stderr
, "line %d: too few numeric fields\n", lineno
);
1059 *vv
= strtoull(*cp
, &tmp
, 0);
1060 if (*vv
== ULLONG_MAX
) {
1061 fprintf(stderr
, "line %d: illegal number\n", lineno
);
1071 * Read a partition line into partition `part' in the specified disklabel.
1072 * Return 0 on success, 1 on failure.
1075 getasciipartspec(char *tp
, struct disklabel64
*lp
, int part
,
1076 int lineno
, uint32_t blksize
)
1078 struct partition64
*pp
;
1086 pp
= &lp
->d_partitions
[part
];
1092 r
= parse_field_val(&tp
, &cp
, &vv
, lineno
);
1122 r
= 0; /* eat the suffix */
1125 Warning("unknown size specifier '%c' (*/%%/K/M/G/T are valid)",
1130 part_size_type
[part
] = r
;
1131 if (vv
== 0 && r
!= '*') {
1133 "line %d: %s: bad partition size (0)\n", lineno
, cp
);
1136 pp
->p_bsize
= vv
* mpx
;
1141 r
= parse_field_val(&tp
, &cp
, &vv
, lineno
);
1144 part_offset_type
[part
] = r
;
1150 pp
->p_boffset
= vv
* blksize
+ lp
->d_pbase
;
1154 "line %d: %s: bad suffix on partition offset (%c)\n",
1164 "line %d: no filesystem type was specified\n", lineno
);
1169 for (cpp
= fstypenames
; cpp
< &fstypenames
[FSMAXTYPES
]; cpp
++) {
1170 if (*cpp
&& strcasecmp(*cpp
, cp
) == 0)
1174 pp
->p_fstype
= cpp
- fstypenames
;
1177 v
= strtoul(cp
, NULL
, 0);
1180 if (v
>= FSMAXTYPES
) {
1182 "line %d: Warning, unknown filesystem type %s\n",
1191 fprintf(stderr
, "line %d: Warning, extra data on line\n",
1198 getasciipartuuid(char *tp
, struct disklabel64
*lp
, int part
,
1199 int lineno
, uint32_t blksize __unused
)
1201 struct partition64
*pp
;
1205 pp
= &lp
->d_partitions
[part
];
1209 uuid_from_string(cp
, &pp
->p_stor_uuid
, &status
);
1210 if (status
!= uuid_s_ok
) {
1211 fprintf(stderr
, "line %d: Illegal storage uuid specification\n",
1219 * Check disklabel for errors and fill in
1220 * derived fields according to supplied values.
1223 checklabel(struct disklabel64
*lp
)
1225 struct partition64
*pp
;
1228 u_int64_t total_size
;
1229 u_int64_t current_offset
;
1230 u_long total_percent
;
1231 int seen_default_offset
;
1234 struct partition64
*pp2
;
1237 if (lp
->d_align
< 512 ||
1238 (lp
->d_align
^ (lp
->d_align
- 1)) != lp
->d_align
* 2 - 1) {
1239 Warning("Illegal alignment specified: %u\n", lp
->d_align
);
1242 if (lp
->d_npartitions
> MAXPARTITIONS64
) {
1243 Warning("number of partitions (%u) > MAXPARTITIONS (%d)",
1244 lp
->d_npartitions
, MAXPARTITIONS64
);
1247 off
= offsetof(struct disklabel64
, d_partitions
[lp
->d_npartitions
]);
1248 off
= (off
+ lp
->d_align
- 1) & ~(int64_t)(lp
->d_align
- 1);
1250 if (lp
->d_bbase
< off
|| lp
->d_bbase
% lp
->d_align
) {
1251 Warning("illegal boot2 data base ");
1256 * pbase can be unaligned slice-relative but will be
1257 * aligned physically.
1259 if (lp
->d_pbase
< lp
->d_bbase
) {
1260 Warning("illegal partition data base");
1263 if (lp
->d_pstop
< lp
->d_pbase
) {
1264 Warning("illegal partition data stop");
1267 if (lp
->d_pstop
> lp
->d_total_size
) {
1268 printf("%012jx\n%012jx\n",
1269 (uintmax_t)lp
->d_pstop
, (uintmax_t)lp
->d_total_size
);
1270 Warning("disklabel control info is beyond the total size");
1274 (lp
->d_abase
< lp
->d_pstop
||
1275 lp
->d_abase
> lp
->d_total_size
- off
)) {
1276 Warning("illegal backup label location");
1280 /* first allocate space to the partitions, then offsets */
1281 total_size
= 0; /* in bytes */
1282 total_percent
= 0; /* in percent */
1284 /* find all fixed partitions */
1285 for (i
= 0; i
< (int)lp
->d_npartitions
; i
++) {
1286 pp
= &lp
->d_partitions
[i
];
1288 if (part_size_type
[i
] == '*') {
1289 if (part_offset_type
[i
] != '*') {
1290 if (total_size
< pp
->p_boffset
)
1291 total_size
= pp
->p_boffset
;
1293 if (hog_part
!= -1) {
1294 Warning("Too many '*' partitions (%c and %c)",
1295 hog_part
+ 'a',i
+ 'a');
1303 if (part_size_type
[i
] == '%') {
1305 * don't count %'s yet
1307 total_percent
+= size
;
1310 * Value has already been converted
1313 if (size
% lp
->d_align
!= 0) {
1314 Warning("partition %c's size is not properly aligned",
1322 /* handle % partitions - note %'s don't need to add up to 100! */
1323 if (total_percent
!= 0) {
1327 free_space
= (int64_t)(lp
->d_pstop
- lp
->d_pbase
- total_size
);
1328 free_space
&= ~(u_int64_t
)(lp
->d_align
- 1);
1330 space_left
= free_space
;
1331 if (total_percent
> 100) {
1332 fprintf(stderr
,"total percentage %lu is greater than 100\n",
1337 if (free_space
> 0) {
1338 for (i
= 0; i
< (int)lp
->d_npartitions
; i
++) {
1339 pp
= &lp
->d_partitions
[i
];
1340 if (part_set
[i
] && part_size_type
[i
] == '%') {
1341 /* careful of overflows! and integer roundoff */
1342 pp
->p_bsize
= ((double)pp
->p_bsize
/100) * free_space
;
1343 pp
->p_bsize
= (pp
->p_bsize
+ lp
->d_align
- 1) & ~(u_int64_t
)(lp
->d_align
- 1);
1344 if ((int64_t)pp
->p_bsize
> space_left
)
1345 pp
->p_bsize
= (u_int64_t
)space_left
;
1346 total_size
+= pp
->p_bsize
;
1347 space_left
-= pp
->p_bsize
;
1351 fprintf(stderr
, "%jd bytes available to give to "
1352 "'*' and '%%' partitions\n",
1353 (intmax_t)free_space
);
1355 /* fix? set all % partitions to size 0? */
1358 /* give anything remaining to the hog partition */
1359 if (hog_part
!= -1) {
1360 off
= lp
->d_pstop
- lp
->d_pbase
- total_size
;
1361 off
&= ~(u_int64_t
)(lp
->d_align
- 1);
1362 lp
->d_partitions
[hog_part
].p_bsize
= off
;
1363 total_size
= lp
->d_pstop
- lp
->d_pbase
;
1366 /* Now set the offsets for each partition */
1367 current_offset
= lp
->d_pbase
;
1368 seen_default_offset
= 0;
1369 for (i
= 0; i
< (int)lp
->d_npartitions
; i
++) {
1371 pp
= &lp
->d_partitions
[i
];
1372 if (pp
->p_bsize
== 0)
1375 if (part_offset_type
[i
] == '*') {
1376 pp
->p_boffset
= current_offset
;
1377 seen_default_offset
= 1;
1379 /* allow them to be out of order for old-style tables */
1380 if (pp
->p_boffset
< current_offset
&&
1381 seen_default_offset
&&
1382 pp
->p_fstype
!= FS_VINUM
) {
1384 "Offset 0x%012jx for partition %c overlaps previous partition which ends at 0x%012jx\n",
1385 (uintmax_t)pp
->p_boffset
,
1387 (uintmax_t)current_offset
);
1389 "Labels with any *'s for offset must be in ascending order by sector\n");
1391 } else if (pp
->p_boffset
!= current_offset
&&
1392 seen_default_offset
) {
1394 * this may give unneeded warnings if
1395 * partitions are out-of-order
1398 "Offset 0x%012jx for partition %c doesn't match expected value 0x%012jx",
1399 pp
->p_boffset
, i
+ 'a',
1400 (intmax_t)current_offset
);
1403 current_offset
= pp
->p_boffset
+ pp
->p_bsize
;
1407 for (i
= 0; i
< (int)lp
->d_npartitions
; i
++) {
1409 pp
= &lp
->d_partitions
[i
];
1410 if (pp
->p_bsize
== 0 && pp
->p_boffset
!= 0)
1411 Warning("partition %c: size 0, but offset 0x%012jx",
1412 part
, (intmax_t)pp
->p_boffset
);
1413 if (pp
->p_bsize
== 0) {
1417 if (uuid_is_nil(&pp
->p_stor_uuid
, NULL
))
1418 uuid_create(&pp
->p_stor_uuid
, NULL
);
1420 if (pp
->p_boffset
< lp
->d_pbase
) {
1422 "partition %c: offset out of bounds (%jd)\n",
1423 part
, (intmax_t)(pp
->p_boffset
- lp
->d_pbase
));
1426 if (pp
->p_boffset
> lp
->d_pstop
) {
1428 "partition %c: offset out of bounds (%jd)\n",
1429 part
, (intmax_t)(pp
->p_boffset
- lp
->d_pbase
));
1432 if (pp
->p_boffset
+ pp
->p_bsize
> lp
->d_pstop
) {
1434 "partition %c: size out of bounds (%jd)\n",
1435 part
, (intmax_t)(pp
->p_boffset
- lp
->d_pbase
));
1439 /* check for overlaps */
1440 /* this will check for all possible overlaps once and only once */
1441 for (j
= 0; j
< i
; j
++) {
1442 pp2
= &lp
->d_partitions
[j
];
1443 if (pp
->p_fstype
!= FS_VINUM
&&
1444 pp2
->p_fstype
!= FS_VINUM
&&
1445 part_set
[i
] && part_set
[j
]) {
1446 if (pp2
->p_boffset
< pp
->p_boffset
+ pp
->p_bsize
&&
1447 (pp2
->p_boffset
+ pp2
->p_bsize
> pp
->p_boffset
||
1448 pp2
->p_boffset
>= pp
->p_boffset
)) {
1449 fprintf(stderr
,"partitions %c and %c overlap!\n",
1456 for (; i
< (int)lp
->d_npartitions
; i
++) {
1458 pp
= &lp
->d_partitions
[i
];
1459 if (pp
->p_bsize
|| pp
->p_boffset
)
1460 Warning("unused partition %c: size 0x%012jx "
1462 'a' + i
, (intmax_t)pp
->p_bsize
,
1463 (intmax_t)pp
->p_boffset
);
1469 expandlabel(int f
, struct disklabel64
*lp
, int *wbackp
)
1471 const uint64_t onemeg
= 1024 * 1024;
1476 struct partinfo info
;
1477 struct partition64
*part
;
1478 struct partition64
*best
;
1482 if (ioctl(f
, DIOCGPART
, &info
) == 0) {
1483 st
.st_size
= info
.media_size
;
1484 } else if (fstat(f
, &st
) == 0) {
1487 fprintf(stderr
, "disklabel64: cannot ioctl/stat device\n");
1490 abase
= st
.st_size
- 4096;
1491 pstop
= abase
& ~(onemeg
- 1);
1494 printf("partitions data stop: %ld -> %ld\n", lp
->d_pstop
, pstop
);
1495 printf("backup label: %ld -> %ld\n", lp
->d_abase
, abase
);
1496 printf("total size: %ld -> %ld\n", lp
->d_total_size
, tsize
);
1499 * This directive does not shrink disklabels!!!
1501 if (lp
->d_pstop
> pstop
||
1502 lp
->d_abase
> abase
||
1503 lp
->d_total_size
> tsize
) {
1504 fprintf(stderr
, "disklabel64: cannot expand "
1505 "disklabel because it would "
1511 if (lp
->d_pstop
== pstop
&&
1512 lp
->d_abase
== abase
&&
1513 lp
->d_total_size
== tsize
) {
1514 printf("disklabel64: expand: "
1515 "no change in disklabel "
1518 lp
->d_pstop
= pstop
;
1519 lp
->d_abase
= abase
;
1520 lp
->d_total_size
= tsize
;
1525 * Resize the last partition
1527 if (expandopt
> 1) {
1529 for (n
= 0; n
< lp
->d_npartitions
; ++n
) {
1530 part
= &lp
->d_partitions
[n
];
1531 if (best
== NULL
|| best
->p_boffset
< part
->p_boffset
)
1535 bsize
= lp
->d_pstop
- best
->p_boffset
;
1536 if (best
->p_bsize
> bsize
) {
1537 printf("disklabel64: cannot expand "
1538 "partition because it would "
1541 } else if (best
->p_bsize
== bsize
) {
1542 printf("disklabel64: expand: "
1543 "no change in partition "
1546 printf("disklabel64: expand: increase part "
1547 "%ld to %ld bytes\n",
1550 best
->p_bsize
= bsize
;
1560 * When operating on a "virgin" disk, try getting an initial label
1561 * from the associated device driver. This might work for all device
1562 * drivers that are able to fetch some initial device parameters
1563 * without even having access to a (BSD) disklabel, like SCSI disks,
1564 * most IDE drives, or vn devices.
1566 * The device name must be given in its "canonical" form.
1568 static struct disklabel64 dlab
;
1570 static struct disklabel64
*
1571 getvirginlabel(void)
1573 struct disklabel64
*dl
= &dlab
;
1576 if ((f
= open(dkname
, O_RDONLY
)) == -1) {
1577 warn("cannot open %s", dkname
);
1582 * Generate a virgin disklabel via ioctl
1584 if (ioctl(f
, DIOCGDVIRGIN64
, dl
) < 0) {
1585 l_perror("ioctl DIOCGDVIRGIN64");
1595 Warning(const char *fmt
, ...)
1599 fprintf(stderr
, "Warning, ");
1601 vfprintf(stderr
, fmt
, ap
);
1602 fprintf(stderr
, "\n");
1609 fprintf(stderr
, "%s\n%s\n%s\n%s\n%s\n%s\n%s\n%s\n%s\n%s\n%s\n%s\n%s\n%s\n%s\n%s\n",
1610 "usage: disklabel64 [-r] disk",
1611 "\t\t(to read label)",
1612 " disklabel64 -w [-r] [-n] disk [type [packid]]",
1613 "\t\t(to write label with existing boot program)",
1614 " disklabel64 -e [-r] [-n] disk",
1615 "\t\t(to edit label)",
1616 " disklabel64 -R [-r] [-n] disk protofile",
1617 "\t\t(to restore label with existing boot program)",
1618 " disklabel64 -B [-n] [-b boot1 -s boot2] disk [type]",
1619 "\t\t(to install boot program with existing label)",
1620 " disklabel64 -w -B [-n] [-b boot1 -s boot2] disk [type [packid]]",
1621 "\t\t(to write label and boot program)",
1622 " disklabel64 -R -B [-n] [-b boot1 -s boot2] disk protofile [type]",
1623 "\t\t(to restore label and boot program)",
1624 " disklabel64 [-NW] disk",
1625 "\t\t(to write disable/enable label)");