kernel - Add fuwordadd32(), fuwordadd64()
[dragonfly.git] / sbin / disklabel64 / disklabel64.c
bloba9eb5f531c9accb132632a05514c69539d7d1036
1 /*
2 * Copyright (c) 2007 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
9 * are met:
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
16 * distribution.
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
32 * SUCH DAMAGE.
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
43 * are met:
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
63 * SUCH DAMAGE.
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>
71 #include <sys/file.h>
72 #include <sys/stat.h>
73 #include <sys/wait.h>
74 #define DKTYPENAMES
75 #include <sys/disklabel64.h>
76 #include <sys/diskslice.h>
77 #include <sys/diskmbr.h>
78 #include <sys/dtype.h>
79 #include <sys/sysctl.h>
80 #include <disktab.h>
81 #include <fstab.h>
83 #include <vfs/ufs/dinode.h>
84 #include <vfs/ufs/fs.h>
86 #include <unistd.h>
87 #include <string.h>
88 #include <stdio.h>
89 #include <stdlib.h>
90 #include <signal.h>
91 #include <stdarg.h>
92 #include <stddef.h>
93 #include <ctype.h>
94 #include <err.h>
95 #include <errno.h>
96 #include <uuid.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 void l_perror(const char *);
104 static void display(FILE *, const struct disklabel64 *);
105 static int edit(struct disklabel64 *, int);
106 static int editit(void);
107 static char *skip(char *);
108 static char *word(char *);
109 static int parse_field_val(char **, char **, u_int64_t *, int);
110 static int getasciilabel(FILE *, struct disklabel64 *);
111 static int getasciipartspec(char *, struct disklabel64 *, int, int, uint32_t);
112 static int getasciipartuuid(char *, struct disklabel64 *, int, int, uint32_t);
113 static int checklabel(struct disklabel64 *);
114 static void Warning(const char *, ...) __printflike(1, 2);
115 static void usage(void);
116 static struct disklabel64 *getvirginlabel(void);
117 static struct disklabel64 *readlabel(int);
118 static struct disklabel64 *makebootarea(int);
120 #define DEFEDITOR _PATH_VI
121 #define streq(a,b) (strcmp(a,b) == 0)
123 static char *dkname;
124 static char *specname;
125 static char tmpfil[] = PATH_TMPFILE;
127 static struct disklabel64 lab;
129 #define MAX_PART ('z')
130 #define MAX_NUM_PARTS (1 + MAX_PART - 'a')
131 static char part_size_type[MAX_NUM_PARTS];
132 static char part_offset_type[MAX_NUM_PARTS];
133 static int part_set[MAX_NUM_PARTS];
135 static int installboot; /* non-zero if we should install a boot program */
136 static int boot1size;
137 static int boot1lsize;
138 static int boot2size;
139 static char *boot1buf;
140 static char *boot2buf;
141 static char *boot1path;
142 static char *boot2path;
144 static enum {
145 UNSPEC, EDIT, NOWRITE, READ, RESTORE, WRITE, WRITEABLE, WRITEBOOT
146 } op = UNSPEC;
148 static int rflag;
149 static int Vflag;
150 static int disable_write; /* set to disable writing to disk label */
152 #ifdef DEBUG
153 static int debug;
154 #define OPTIONS "BNRWb:denrs:Vw"
155 #else
156 #define OPTIONS "BNRWb:enrs:Vw"
157 #endif
160 main(int argc, char *argv[])
162 struct disklabel64 *lp;
163 FILE *t;
164 int ch, f = 0, flag, error = 0;
165 const char *name = NULL;
166 const char *dtype = NULL;
168 while ((ch = getopt(argc, argv, OPTIONS)) != -1)
169 switch (ch) {
170 case 'B':
171 ++installboot;
172 break;
173 case 'b':
174 boot1path = optarg;
175 break;
177 case 's':
178 boot2path = optarg;
179 break;
180 case 'N':
181 if (op != UNSPEC)
182 usage();
183 op = NOWRITE;
184 break;
185 case 'n':
186 disable_write = 1;
187 break;
188 case 'R':
189 if (op != UNSPEC)
190 usage();
191 op = RESTORE;
192 break;
193 case 'W':
194 if (op != UNSPEC)
195 usage();
196 op = WRITEABLE;
197 break;
198 case 'e':
199 if (op != UNSPEC)
200 usage();
201 op = EDIT;
202 break;
203 case 'V':
204 ++Vflag;
205 break;
206 case 'r':
207 ++rflag;
208 break;
209 case 'w':
210 if (op != UNSPEC)
211 usage();
212 op = WRITE;
213 break;
214 #ifdef DEBUG
215 case 'd':
216 debug++;
217 break;
218 #endif
219 case '?':
220 default:
221 usage();
223 argc -= optind;
224 argv += optind;
225 if (installboot) {
226 rflag++;
227 if (op == UNSPEC)
228 op = WRITEBOOT;
229 } else {
230 if (op == UNSPEC)
231 op = READ;
232 boot1path = NULL;
233 boot2path = NULL;
235 if (argc < 1)
236 usage();
238 dkname = getdevpath(argv[0], 0);
239 specname = dkname;
240 f = open(specname, op == READ ? O_RDONLY : O_RDWR);
241 if (f < 0)
242 err(4, "%s", specname);
244 switch(op) {
246 case UNSPEC:
247 break;
249 case EDIT:
250 if (argc != 1)
251 usage();
252 lp = readlabel(f);
253 error = edit(lp, f);
254 break;
256 case NOWRITE:
257 flag = 0;
258 if (ioctl(f, DIOCWLABEL, (char *)&flag) < 0)
259 err(4, "ioctl DIOCWLABEL");
260 break;
262 case READ:
263 if (argc != 1)
264 usage();
265 lp = readlabel(f);
266 display(stdout, lp);
267 error = checklabel(lp);
268 break;
270 case RESTORE:
271 if (installboot && argc == 3) {
272 makelabel(argv[2], 0, &lab);
273 argc--;
276 * We only called makelabel() for its side effect
277 * of setting the bootstrap file names. Discard
278 * all changes to `lab' so that all values in the
279 * final label come from the ASCII label.
281 bzero((char *)&lab, sizeof(lab));
283 if (argc != 2)
284 usage();
285 if (!(t = fopen(argv[1], "r")))
286 err(4, "%s", argv[1]);
287 if (!getasciilabel(t, &lab))
288 exit(1);
289 lp = makebootarea(f);
290 bcopy(&lab.d_magic, &lp->d_magic,
291 sizeof(lab) - offsetof(struct disklabel64, d_magic));
292 error = writelabel(f, lp);
293 break;
295 case WRITE:
296 if (argc == 3) {
297 name = argv[2];
298 argc--;
300 if (argc == 2)
301 dtype = argv[1];
302 else if (argc == 1)
303 dtype = "auto";
304 else
305 usage();
306 makelabel(dtype, name, &lab);
307 lp = makebootarea(f);
308 bcopy(&lab.d_magic, &lp->d_magic,
309 sizeof(lab) - offsetof(struct disklabel64, d_magic));
310 if (checklabel(lp) == 0)
311 error = writelabel(f, lp);
312 break;
314 case WRITEABLE:
315 flag = 1;
316 if (ioctl(f, DIOCWLABEL, (char *)&flag) < 0)
317 err(4, "ioctl DIOCWLABEL");
318 break;
320 case WRITEBOOT:
322 struct disklabel64 tlab;
324 lp = readlabel(f);
325 tlab = *lp;
326 if (argc == 2)
327 makelabel(argv[1], 0, &lab);
328 lp = makebootarea(f);
329 bcopy(&tlab.d_magic, &lp->d_magic,
330 sizeof(tlab) - offsetof(struct disklabel64, d_magic));
331 if (checklabel(lp) == 0)
332 error = writelabel(f, lp);
333 break;
336 exit(error);
340 * Construct a prototype disklabel from /etc/disktab. As a side
341 * effect, set the names of the primary and secondary boot files
342 * if specified.
344 static void
345 makelabel(const char *type, const char *name, struct disklabel64 *lp)
347 struct disklabel64 *dp;
349 if (strcmp(type, "auto") == 0)
350 dp = getvirginlabel();
351 else
352 errx(1, "no disktab(5) support yet; only 'auto' allowed");
353 *lp = *dp;
356 * NOTE: boot control files may no longer be specified in disktab.
358 if (name)
359 strlcpy((char *)lp->d_packname, name, sizeof(lp->d_packname));
362 static int
363 writelabel(int f, struct disklabel64 *lp)
365 struct disklabel64 *blp;
366 int flag;
367 int r;
368 size_t lpsize;
369 size_t lpcrcsize;
371 lpsize = offsetof(struct disklabel64, d_partitions[lp->d_npartitions]);
372 lpcrcsize = lpsize - offsetof(struct disklabel64, d_magic);
374 if (disable_write) {
375 Warning("write to disk label suppressed - label was as follows:");
376 display(stdout, lp);
377 return (0);
378 } else {
379 lp->d_magic = DISKMAGIC64;
380 lp->d_crc = 0;
381 lp->d_crc = crc32(&lp->d_magic, lpcrcsize);
382 if (rflag) {
384 * Make sure the boot area is not too large
386 if (boot2buf) {
387 int lpbsize = (int)(lp->d_pbase - lp->d_bbase);
388 if (lp->d_pbase == 0) {
389 errx(1, "no space was set aside in "
390 "the disklabel for boot2!");
392 if (boot2size > lpbsize) {
393 errx(1, "label did not reserve enough "
394 "space for boot! %d/%d",
395 boot2size, lpbsize);
400 * First set the kernel disk label,
401 * then write a label to the raw disk.
402 * If the SDINFO ioctl fails because it is
403 * unimplemented, keep going; otherwise, the kernel
404 * consistency checks may prevent us from changing
405 * the current (in-core) label.
407 if (ioctl(f, DIOCSDINFO64, lp) < 0 &&
408 errno != ENODEV && errno != ENOTTY) {
409 l_perror("ioctl DIOCSDINFO");
410 return (1);
412 lseek(f, (off_t)0, SEEK_SET);
415 * The disklabel embeds areas which we may not
416 * have wanted to change. Merge those areas in
417 * from disk.
419 blp = makebootarea(f);
420 if (blp != lp) {
421 bcopy(&lp->d_magic, &blp->d_magic,
422 sizeof(*lp) -
423 offsetof(struct disklabel64, d_magic));
427 * write enable label sector before write
428 * (if necessary), disable after writing.
430 flag = 1;
431 if (ioctl(f, DIOCWLABEL, &flag) < 0)
432 warn("ioctl DIOCWLABEL");
434 r = write(f, boot1buf, boot1lsize);
435 if (r != (ssize_t)boot1lsize) {
436 warn("write");
437 return (1);
440 * Output the remainder of the disklabel
442 if (boot2buf) {
443 lseek(f, lp->d_bbase, 0);
444 r = write(f, boot2buf, boot2size);
445 if (r != boot2size) {
446 warn("write");
447 return(1);
450 flag = 0;
451 ioctl(f, DIOCWLABEL, &flag);
452 } else if (ioctl(f, DIOCWDINFO64, lp) < 0) {
453 l_perror("ioctl DIOCWDINFO64");
454 return (1);
457 return (0);
460 static void
461 l_perror(const char *s)
463 switch (errno) {
465 case ESRCH:
466 warnx("%s: no disk label on disk;", s);
467 fprintf(stderr, "add \"-r\" to install initial label\n");
468 break;
470 case EINVAL:
471 warnx("%s: label magic number or checksum is wrong!", s);
472 fprintf(stderr, "(disklabel or kernel is out of date?)\n");
473 break;
475 case EBUSY:
476 warnx("%s: open partition would move or shrink", s);
477 break;
479 case ENOATTR:
480 warnx("%s: the disk already has a label of a different type,\n"
481 "probably a 32 bit disklabel. It must be cleaned out "
482 "first.\n", s);
483 break;
485 default:
486 warn(NULL);
487 break;
492 * Fetch disklabel for disk.
493 * Use ioctl to get label unless -r flag is given.
495 static struct disklabel64 *
496 readlabel(int f)
498 struct disklabel64 *lp;
499 u_int32_t savecrc;
500 size_t lpcrcsize;
502 if (rflag) {
504 * Allocate space for the label. The boot1 code, if any,
505 * is embedded in the label. The label overlaps the boot1
506 * code.
508 lp = makebootarea(f);
509 lpcrcsize = offsetof(struct disklabel64,
510 d_partitions[lp->d_npartitions]) -
511 offsetof(struct disklabel64, d_magic);
512 savecrc = lp->d_crc;
513 lp->d_crc = 0;
514 if (lp->d_magic != DISKMAGIC64)
515 errx(1, "bad pack magic number");
516 if (lp->d_npartitions > MAXPARTITIONS64 ||
517 savecrc != crc32(&lp->d_magic, lpcrcsize)
519 errx(1, "corrupted disklabel64");
521 lp->d_crc = savecrc;
522 } else {
524 * Just use a static structure to hold the label. Note
525 * that DIOCSDINFO64 does not overwrite the boot1 area
526 * even though it is part of the disklabel64 structure.
528 lp = &lab;
529 if (Vflag) {
530 if (ioctl(f, DIOCGDVIRGIN64, lp) < 0) {
531 l_perror("ioctl DIOCGDVIRGIN64");
532 exit(4);
534 } else {
535 if (ioctl(f, DIOCGDINFO64, lp) < 0) {
536 l_perror("ioctl DIOCGDINFO64");
537 exit(4);
541 return (lp);
545 * Construct a boot area for boot1 and boot2 and return the location of
546 * the label within the area. The caller will overwrite the label so
547 * we don't actually have to read it.
549 static struct disklabel64 *
550 makebootarea(int f)
552 struct disklabel64 *lp;
553 struct partinfo info;
554 u_int32_t secsize;
555 struct stat st;
556 int fd;
557 int r;
559 if (ioctl(f, DIOCGPART, &info) == 0)
560 secsize = info.media_blksize;
561 else
562 secsize = 512;
564 if (boot1buf == NULL) {
565 size_t rsize;
567 rsize = roundup2(sizeof(struct disklabel64), secsize);
568 boot1size = offsetof(struct disklabel64, d_magic);
569 boot1lsize = rsize;
570 boot1buf = malloc(rsize);
571 bzero(boot1buf, rsize);
572 r = read(f, boot1buf, rsize);
573 if (r != (int)rsize) {
574 free(boot1buf);
575 err(4, "%s", specname);
578 lp = (void *)boot1buf;
580 if (installboot == 0)
581 return(lp);
583 if (boot2buf == NULL) {
584 boot2size = BOOT2SIZE64;
585 boot2buf = malloc(boot2size);
586 bzero(boot2buf, boot2size);
590 * If installing the boot code, read it into the appropriate portions
591 * of the buffer(s)
593 if (boot1path == NULL)
594 asprintf(&boot1path, "%s/boot1_64", _PATH_BOOTDIR);
595 if (boot2path == NULL)
596 asprintf(&boot2path, "%s/boot2_64", _PATH_BOOTDIR);
598 if ((fd = open(boot1path, O_RDONLY)) < 0)
599 err(4, "%s", boot1path);
600 if (fstat(fd, &st) < 0)
601 err(4, "%s", boot1path);
602 if (st.st_size > boot1size)
603 err(4, "%s must be exactly %d bytes!", boot1path, boot1size);
604 if (read(fd, boot1buf, boot1size) != boot1size)
605 err(4, "%s must be exactly %d bytes!", boot1path, boot1size);
606 close(fd);
608 if ((fd = open(boot2path, O_RDONLY)) < 0)
609 err(4, "%s", boot2path);
610 if (fstat(fd, &st) < 0)
611 err(4, "%s", boot2path);
612 if (st.st_size > boot2size)
613 err(4, "%s must be <= %d bytes!", boot2path, boot2size);
614 if ((r = read(fd, boot2buf, boot2size)) < 1)
615 err(4, "%s is empty!", boot2path);
616 boot2size = roundup2(r, secsize);
617 close(fd);
620 * XXX dangerously dedicated support goes here XXX
622 return (lp);
625 static void
626 display(FILE *f, const struct disklabel64 *lp)
628 const struct partition64 *pp;
629 char *str;
630 unsigned int part;
631 int didany;
632 uint32_t blksize;
635 * Use a human readable block size if possible. This is for
636 * display and editing purposes only.
638 if (lp->d_align > 1024)
639 blksize = 1024;
640 else
641 blksize = lp->d_align;
643 fprintf(f, "# %s:\n", specname);
644 fprintf(f, "#\n");
645 fprintf(f, "# Calculated informational fields for the slice:\n");
646 fprintf(f, "#\n");
647 fprintf(f, "# boot space: %10ju bytes\n",
648 (intmax_t)(lp->d_pbase - lp->d_bbase));
649 fprintf(f, "# data space: %10ju blocks\t# %6.2f MB (%ju bytes)\n",
650 (intmax_t)(lp->d_pstop - lp->d_pbase) / blksize,
651 (double)(lp->d_pstop - lp->d_pbase) / 1024.0 / 1024.0,
652 (intmax_t)(lp->d_pstop - lp->d_pbase));
653 fprintf(f, "#\n");
654 fprintf(f, "# NOTE: The partition data base and stop are physically\n");
655 fprintf(f, "# aligned instead of slice-relative aligned.\n");
656 fprintf(f, "#\n");
657 fprintf(f, "# All byte equivalent offsets must be aligned.\n");
658 fprintf(f, "#\n");
660 uuid_to_string(&lp->d_stor_uuid, &str, NULL);
661 fprintf(f, "diskid: %s\n", str ? str : "<unknown>");
662 free(str);
664 fprintf(f, "label: %s\n", lp->d_packname);
665 fprintf(f, "boot2 data base: 0x%012jx\n", (intmax_t)lp->d_bbase);
666 fprintf(f, "partitions data base: 0x%012jx\n", (intmax_t)lp->d_pbase);
667 fprintf(f, "partitions data stop: 0x%012jx\n", (intmax_t)lp->d_pstop);
668 fprintf(f, "backup label: 0x%012jx\n", (intmax_t)lp->d_abase);
669 fprintf(f, "total size: 0x%012jx\t# %6.2f MB\n",
670 (intmax_t)lp->d_total_size,
671 (double)lp->d_total_size / 1024.0 / 1024.0);
672 fprintf(f, "alignment: %u\n", lp->d_align);
673 fprintf(f, "display block size: %u\t# for partition display and edit only\n",
674 blksize);
676 fprintf(f, "\n");
677 fprintf(f, "%u partitions:\n", lp->d_npartitions);
678 fprintf(f, "# size offset fstype fsuuid\n");
679 didany = 0;
680 for (part = 0; part < lp->d_npartitions; part++) {
681 pp = &lp->d_partitions[part];
682 const u_long onemeg = 1024 * 1024;
684 if (pp->p_bsize == 0)
685 continue;
686 didany = 1;
687 fprintf(f, " %c: ", 'a' + part);
689 if (pp->p_bsize % lp->d_align)
690 fprintf(f, "%10s ", "ILLEGAL");
691 else
692 fprintf(f, "%10ju ", (intmax_t)pp->p_bsize / blksize);
694 if ((pp->p_boffset - lp->d_pbase) % lp->d_align)
695 fprintf(f, "%10s ", "ILLEGAL");
696 else
697 fprintf(f, "%10ju ",
698 (intmax_t)(pp->p_boffset - lp->d_pbase) / blksize);
700 if (pp->p_fstype < FSMAXTYPES)
701 fprintf(f, "%8.8s", fstypenames[pp->p_fstype]);
702 else
703 fprintf(f, "%8d", pp->p_fstype);
704 fprintf(f, "\t# %11.3fMB", (double)pp->p_bsize / onemeg);
705 fprintf(f, "\n");
707 for (part = 0; part < lp->d_npartitions; part++) {
708 pp = &lp->d_partitions[part];
710 if (pp->p_bsize == 0)
711 continue;
713 if (uuid_is_nil(&lp->d_stor_uuid, NULL) == 0) {
714 fprintf(f, " %c-stor_uuid: ", 'a' + part);
715 str = NULL;
716 uuid_to_string(&pp->p_stor_uuid, &str, NULL);
717 if (str) {
718 fprintf(f, "%s", str);
719 free(str);
721 fprintf(f, "\n");
724 if (didany == 0) {
725 fprintf(f, "# EXAMPLE\n");
726 fprintf(f, "#a: 4g 0 4.2BSD\n");
727 fprintf(f, "#a: * * 4.2BSD\n");
730 fflush(f);
733 static int
734 edit(struct disklabel64 *lp, int f)
736 int c, fd;
737 struct disklabel64 label;
738 FILE *fp;
740 if ((fd = mkstemp(tmpfil)) == -1 ||
741 (fp = fdopen(fd, "w")) == NULL) {
742 warnx("can't create %s", tmpfil);
743 return (1);
745 display(fp, lp);
746 fclose(fp);
747 for (;;) {
748 if (!editit())
749 break;
750 fp = fopen(tmpfil, "r");
751 if (fp == NULL) {
752 warnx("can't reopen %s for reading", tmpfil);
753 break;
755 bzero((char *)&label, sizeof(label));
756 if (getasciilabel(fp, &label)) {
757 *lp = label;
758 if (writelabel(f, lp) == 0) {
759 fclose(fp);
760 unlink(tmpfil);
761 return (0);
764 fclose(fp);
765 printf("re-edit the label? [y]: "); fflush(stdout);
766 c = getchar();
767 if (c != EOF && c != (int)'\n')
768 while (getchar() != (int)'\n')
770 if (c == (int)'n')
771 break;
773 unlink(tmpfil);
774 return (1);
777 static int
778 editit(void)
780 int pid, xpid;
781 int status, omask;
782 const char *ed;
784 omask = sigblock(sigmask(SIGINT)|sigmask(SIGQUIT)|sigmask(SIGHUP));
785 while ((pid = fork()) < 0) {
786 if (errno == EPROCLIM) {
787 warnx("you have too many processes");
788 return(0);
790 if (errno != EAGAIN) {
791 warn("fork");
792 return(0);
794 sleep(1);
796 if (pid == 0) {
797 sigsetmask(omask);
798 setgid(getgid());
799 setuid(getuid());
800 if ((ed = getenv("EDITOR")) == NULL)
801 ed = DEFEDITOR;
802 execlp(ed, ed, tmpfil, NULL);
803 err(1, "%s", ed);
805 while ((xpid = wait(&status)) >= 0)
806 if (xpid == pid)
807 break;
808 sigsetmask(omask);
809 return(!status);
812 static char *
813 skip(char *cp)
816 while (*cp != '\0' && isspace(*cp))
817 cp++;
818 if (*cp == '\0' || *cp == '#')
819 return (NULL);
820 return (cp);
823 static char *
824 word(char *cp)
826 char c;
828 while (*cp != '\0' && !isspace(*cp) && *cp != '#')
829 cp++;
830 if ((c = *cp) != '\0') {
831 *cp++ = '\0';
832 if (c != '#')
833 return (skip(cp));
835 return (NULL);
839 * Read an ascii label in from fd f,
840 * in the same format as that put out by display(),
841 * and fill in lp.
843 static int
844 getasciilabel(FILE *f, struct disklabel64 *lp)
846 char *cp;
847 u_int part;
848 char *tp, line[BUFSIZ];
849 u_long v;
850 uint32_t blksize = 0;
851 uint64_t vv;
852 int lineno = 0, errors = 0;
853 char empty[] = "";
855 bzero(&part_set, sizeof(part_set));
856 bzero(&part_size_type, sizeof(part_size_type));
857 bzero(&part_offset_type, sizeof(part_offset_type));
858 while (fgets(line, sizeof(line) - 1, f)) {
859 lineno++;
860 if ((cp = strchr(line,'\n')) != NULL)
861 *cp = '\0';
862 cp = skip(line);
863 if (cp == NULL)
864 continue;
865 tp = strchr(cp, ':');
866 if (tp == NULL) {
867 fprintf(stderr, "line %d: syntax error\n", lineno);
868 errors++;
869 continue;
871 *tp++ = '\0', tp = skip(tp);
872 if (sscanf(cp, "%lu partitions", &v) == 1) {
873 if (v == 0 || v > MAXPARTITIONS64) {
874 fprintf(stderr,
875 "line %d: bad # of partitions\n", lineno);
876 lp->d_npartitions = MAXPARTITIONS64;
877 errors++;
878 } else
879 lp->d_npartitions = v;
880 continue;
882 if (tp == NULL)
883 tp = empty;
885 if (streq(cp, "diskid")) {
886 uint32_t status = 0;
887 uuid_from_string(tp, &lp->d_stor_uuid, &status);
888 if (status != uuid_s_ok) {
889 fprintf(stderr,
890 "line %d: %s: illegal UUID\n",
891 lineno, tp);
892 errors++;
894 continue;
896 if (streq(cp, "label")) {
897 strlcpy((char *)lp->d_packname, tp, sizeof(lp->d_packname));
898 continue;
901 if (streq(cp, "alignment")) {
902 v = strtoul(tp, NULL, 0);
903 if (v <= 0 || (v & DEV_BMASK) != 0 || v > 1024*1024) {
904 fprintf(stderr,
905 "line %d: %s: bad alignment\n",
906 lineno, tp);
907 errors++;
908 } else {
909 lp->d_align = v;
911 continue;
913 if (streq(cp, "total size")) {
914 vv = strtoull(tp, NULL, 0);
915 if (vv == 0 || vv == (uint64_t)-1) {
916 fprintf(stderr, "line %d: %s: bad %s\n",
917 lineno, tp, cp);
918 errors++;
919 } else {
920 lp->d_total_size = vv;
922 continue;
924 if (streq(cp, "boot2 data base")) {
925 vv = strtoull(tp, NULL, 0);
926 if (vv == 0 || vv == (uint64_t)-1) {
927 fprintf(stderr, "line %d: %s: bad %s\n",
928 lineno, tp, cp);
929 errors++;
930 } else {
931 lp->d_bbase = vv;
933 continue;
935 if (streq(cp, "partitions data base")) {
936 vv = strtoull(tp, NULL, 0);
937 if (vv == 0 || vv == (uint64_t)-1) {
938 fprintf(stderr, "line %d: %s: bad %s\n",
939 lineno, tp, cp);
940 errors++;
941 } else {
942 lp->d_pbase = vv;
944 continue;
946 if (streq(cp, "partitions data stop")) {
947 vv = strtoull(tp, NULL, 0);
948 if (vv == 0 || vv == (uint64_t)-1) {
949 fprintf(stderr, "line %d: %s: bad %s\n",
950 lineno, tp, cp);
951 errors++;
952 } else {
953 lp->d_pstop = vv;
955 continue;
957 if (streq(cp, "backup label")) {
958 vv = strtoull(tp, NULL, 0);
959 if (vv == 0 || vv == (uint64_t)-1) {
960 fprintf(stderr, "line %d: %s: bad %s\n",
961 lineno, tp, cp);
962 errors++;
963 } else {
964 lp->d_abase = vv;
966 continue;
968 if (streq(cp, "display block size")) {
969 v = strtoul(tp, NULL, 0);
970 if (v <= 0 || (v & DEV_BMASK) != 0 || v > 1024*1024) {
971 fprintf(stderr, "line %d: %s: bad %s\n",
972 lineno, tp, cp);
973 errors++;
974 } else {
975 blksize = v;
977 continue;
980 /* the ':' was removed above */
983 * Handle main partition data, e.g. a:, b:, etc.
985 if (*cp < 'a' || *cp > MAX_PART) {
986 fprintf(stderr,
987 "line %d: %s: Unknown disklabel field\n", lineno,
988 cp);
989 errors++;
990 continue;
993 /* Process a partition specification line. */
994 part = *cp - 'a';
995 if (part >= lp->d_npartitions) {
996 fprintf(stderr,
997 "line %d: partition name out of range a-%c: %s\n",
998 lineno, 'a' + lp->d_npartitions - 1, cp);
999 errors++;
1000 continue;
1003 if (blksize == 0) {
1004 fprintf(stderr, "block size to use for partition "
1005 "display was not specified!\n");
1006 errors++;
1007 continue;
1010 if (strcmp(cp + 1, "-stor_uuid") == 0) {
1011 if (getasciipartuuid(tp, lp, part, lineno, blksize)) {
1012 errors++;
1013 break;
1015 continue;
1016 } else if (cp[1] == 0) {
1017 part_set[part] = 1;
1018 if (getasciipartspec(tp, lp, part, lineno, blksize)) {
1019 errors++;
1020 break;
1022 continue;
1024 fprintf(stderr, "line %d: %s: Unknown disklabel field\n",
1025 lineno, cp);
1026 errors++;
1027 continue;
1029 errors += checklabel(lp);
1030 return (errors == 0);
1033 static int
1034 parse_field_val(char **tp, char **cp, u_int64_t *vv, int lineno)
1036 char *tmp;
1038 if (*tp == NULL || **tp == 0) {
1039 fprintf(stderr, "line %d: too few numeric fields\n", lineno);
1040 return(-1);
1042 *cp = *tp;
1043 *tp = word(*cp);
1044 *vv = strtoull(*cp, &tmp, 0);
1045 if (*vv == ULLONG_MAX) {
1046 fprintf(stderr, "line %d: illegal number\n", lineno);
1047 return(-1);
1049 if (tmp)
1050 return(*tmp);
1051 else
1052 return(0);
1056 * Read a partition line into partition `part' in the specified disklabel.
1057 * Return 0 on success, 1 on failure.
1059 static int
1060 getasciipartspec(char *tp, struct disklabel64 *lp, int part,
1061 int lineno, uint32_t blksize)
1063 struct partition64 *pp;
1064 char *cp;
1065 const char **cpp;
1066 int r;
1067 u_long v;
1068 uint64_t vv;
1069 uint64_t mpx;
1071 pp = &lp->d_partitions[part];
1072 cp = NULL;
1075 * size
1077 r = parse_field_val(&tp, &cp, &vv, lineno);
1078 if (r < 0)
1079 return (1);
1081 mpx = 1;
1082 switch(r) {
1083 case 0:
1084 mpx = blksize;
1085 break;
1086 case '%':
1087 /* mpx = 1; */
1088 break;
1089 case '*':
1090 mpx = 0;
1091 break;
1092 case 't':
1093 case 'T':
1094 mpx *= 1024ULL;
1095 /* fall through */
1096 case 'g':
1097 case 'G':
1098 mpx *= 1024ULL;
1099 /* fall through */
1100 case 'm':
1101 case 'M':
1102 mpx *= 1024ULL;
1103 /* fall through */
1104 case 'k':
1105 case 'K':
1106 mpx *= 1024ULL;
1107 r = 0; /* eat the suffix */
1108 break;
1109 default:
1110 Warning("unknown size specifier '%c' (*/%%/K/M/G/T are valid)",
1112 return(1);
1115 part_size_type[part] = r;
1116 if (vv == 0 && r != '*') {
1117 fprintf(stderr,
1118 "line %d: %s: bad partition size (0)\n", lineno, cp);
1119 return (1);
1121 pp->p_bsize = vv * mpx;
1124 * offset
1126 r = parse_field_val(&tp, &cp, &vv, lineno);
1127 if (r < 0)
1128 return (1);
1129 part_offset_type[part] = r;
1130 switch(r) {
1131 case '*':
1132 pp->p_boffset = 0;
1133 break;
1134 case 0:
1135 pp->p_boffset = vv * blksize + lp->d_pbase;
1136 break;
1137 default:
1138 fprintf(stderr,
1139 "line %d: %s: bad suffix on partition offset (%c)\n",
1140 lineno, cp, r);
1141 return (1);
1145 * fstype
1147 if (tp == NULL) {
1148 fprintf(stderr,
1149 "line %d: no filesystem type was specified\n", lineno);
1150 return(1);
1152 cp = tp;
1153 tp = word(cp);
1154 for (cpp = fstypenames; cpp < &fstypenames[FSMAXTYPES]; cpp++) {
1155 if (*cpp && strcasecmp(*cpp, cp) == 0)
1156 break;
1158 if (*cpp != NULL) {
1159 pp->p_fstype = cpp - fstypenames;
1160 } else {
1161 if (isdigit(*cp))
1162 v = strtoul(cp, NULL, 0);
1163 else
1164 v = FSMAXTYPES;
1165 if (v >= FSMAXTYPES) {
1166 fprintf(stderr,
1167 "line %d: Warning, unknown filesystem type %s\n",
1168 lineno, cp);
1169 v = FS_UNUSED;
1171 pp->p_fstype = v;
1174 cp = tp;
1175 if (tp) {
1176 fprintf(stderr, "line %d: Warning, extra data on line\n",
1177 lineno);
1179 return(0);
1182 static int
1183 getasciipartuuid(char *tp, struct disklabel64 *lp, int part,
1184 int lineno, uint32_t blksize __unused)
1186 struct partition64 *pp;
1187 uint32_t status;
1188 char *cp;
1190 pp = &lp->d_partitions[part];
1192 cp = tp;
1193 tp = word(cp);
1194 uuid_from_string(cp, &pp->p_stor_uuid, &status);
1195 if (status != uuid_s_ok) {
1196 fprintf(stderr, "line %d: Illegal storage uuid specification\n",
1197 lineno);
1198 return(1);
1200 return(0);
1204 * Check disklabel for errors and fill in
1205 * derived fields according to supplied values.
1207 static int
1208 checklabel(struct disklabel64 *lp)
1210 struct partition64 *pp;
1211 int errors = 0;
1212 char part;
1213 u_int64_t total_size;
1214 u_int64_t current_offset;
1215 u_long total_percent;
1216 int seen_default_offset;
1217 int hog_part;
1218 int i, j;
1219 struct partition64 *pp2;
1220 u_int64_t off;
1222 if (lp->d_align < 512 ||
1223 (lp->d_align ^ (lp->d_align - 1)) != lp->d_align * 2 - 1) {
1224 Warning("Illegal alignment specified: %u\n", lp->d_align);
1225 return (1);
1227 if (lp->d_npartitions > MAXPARTITIONS64) {
1228 Warning("number of partitions (%u) > MAXPARTITIONS (%d)",
1229 lp->d_npartitions, MAXPARTITIONS64);
1230 return (1);
1232 off = offsetof(struct disklabel64, d_partitions[lp->d_npartitions]);
1233 off = (off + lp->d_align - 1) & ~(int64_t)(lp->d_align - 1);
1235 if (lp->d_bbase < off || lp->d_bbase % lp->d_align) {
1236 Warning("illegal boot2 data base ");
1237 return (1);
1241 * pbase can be unaligned slice-relative but will be
1242 * aligned physically.
1244 if (lp->d_pbase < lp->d_bbase) {
1245 Warning("illegal partition data base");
1246 return (1);
1248 if (lp->d_pstop < lp->d_pbase) {
1249 Warning("illegal partition data stop");
1250 return (1);
1252 if (lp->d_pstop > lp->d_total_size) {
1253 printf("%012jx\n%012jx\n",
1254 (intmax_t)lp->d_pstop, (intmax_t)lp->d_total_size);
1255 Warning("disklabel control info is beyond the total size");
1256 return (1);
1258 if (lp->d_abase &&
1259 (lp->d_abase < lp->d_pstop ||
1260 lp->d_abase > lp->d_total_size - off)) {
1261 Warning("illegal backup label location");
1262 return (1);
1265 /* first allocate space to the partitions, then offsets */
1266 total_size = 0; /* in bytes */
1267 total_percent = 0; /* in percent */
1268 hog_part = -1;
1269 /* find all fixed partitions */
1270 for (i = 0; i < (int)lp->d_npartitions; i++) {
1271 pp = &lp->d_partitions[i];
1272 if (part_set[i]) {
1273 if (part_size_type[i] == '*') {
1274 if (part_offset_type[i] != '*') {
1275 if (total_size < pp->p_boffset)
1276 total_size = pp->p_boffset;
1278 if (hog_part != -1) {
1279 Warning("Too many '*' partitions (%c and %c)",
1280 hog_part + 'a',i + 'a');
1281 } else {
1282 hog_part = i;
1284 } else {
1285 off_t size;
1287 size = pp->p_bsize;
1288 if (part_size_type[i] == '%') {
1290 * don't count %'s yet
1292 total_percent += size;
1293 } else {
1295 * Value has already been converted
1296 * to bytes.
1298 if (size % lp->d_align != 0) {
1299 Warning("partition %c's size is not properly aligned",
1300 i + 'a');
1302 total_size += size;
1307 /* handle % partitions - note %'s don't need to add up to 100! */
1308 if (total_percent != 0) {
1309 int64_t free_space;
1310 int64_t space_left;
1312 free_space = (int64_t)(lp->d_pstop - lp->d_pbase - total_size);
1313 free_space &= ~(u_int64_t)(lp->d_align - 1);
1315 space_left = free_space;
1316 if (total_percent > 100) {
1317 fprintf(stderr,"total percentage %lu is greater than 100\n",
1318 total_percent);
1319 errors++;
1322 if (free_space > 0) {
1323 for (i = 0; i < (int)lp->d_npartitions; i++) {
1324 pp = &lp->d_partitions[i];
1325 if (part_set[i] && part_size_type[i] == '%') {
1326 /* careful of overflows! and integer roundoff */
1327 pp->p_bsize = ((double)pp->p_bsize/100) * free_space;
1328 pp->p_bsize = (pp->p_bsize + lp->d_align - 1) & ~(u_int64_t)(lp->d_align - 1);
1329 if ((int64_t)pp->p_bsize > space_left)
1330 pp->p_bsize = (u_int64_t)space_left;
1331 total_size += pp->p_bsize;
1332 space_left -= pp->p_bsize;
1335 } else {
1336 fprintf(stderr, "%jd bytes available to give to "
1337 "'*' and '%%' partitions\n",
1338 (intmax_t)free_space);
1339 errors++;
1340 /* fix? set all % partitions to size 0? */
1343 /* give anything remaining to the hog partition */
1344 if (hog_part != -1) {
1345 off = lp->d_pstop - lp->d_pbase - total_size;
1346 off &= ~(u_int64_t)(lp->d_align - 1);
1347 lp->d_partitions[hog_part].p_bsize = off;
1348 total_size = lp->d_pstop - lp->d_pbase;
1351 /* Now set the offsets for each partition */
1352 current_offset = lp->d_pbase;
1353 seen_default_offset = 0;
1354 for (i = 0; i < (int)lp->d_npartitions; i++) {
1355 part = 'a' + i;
1356 pp = &lp->d_partitions[i];
1357 if (pp->p_bsize == 0)
1358 continue;
1359 if (part_set[i]) {
1360 if (part_offset_type[i] == '*') {
1361 pp->p_boffset = current_offset;
1362 seen_default_offset = 1;
1363 } else {
1364 /* allow them to be out of order for old-style tables */
1365 if (pp->p_boffset < current_offset &&
1366 seen_default_offset &&
1367 pp->p_fstype != FS_VINUM) {
1368 fprintf(stderr,
1369 "Offset 0x%012jx for partition %c overlaps previous partition which ends at 0x%012jx\n",
1370 (intmax_t)pp->p_boffset,
1371 i + 'a',
1372 (intmax_t)current_offset);
1373 fprintf(stderr,
1374 "Labels with any *'s for offset must be in ascending order by sector\n");
1375 errors++;
1376 } else if (pp->p_boffset != current_offset &&
1377 seen_default_offset) {
1379 * this may give unneeded warnings if
1380 * partitions are out-of-order
1382 Warning(
1383 "Offset 0x%012jx for partition %c doesn't match expected value 0x%012jx",
1384 pp->p_boffset, i + 'a',
1385 (intmax_t)current_offset);
1388 current_offset = pp->p_boffset + pp->p_bsize;
1392 for (i = 0; i < (int)lp->d_npartitions; i++) {
1393 part = 'a' + i;
1394 pp = &lp->d_partitions[i];
1395 if (pp->p_bsize == 0 && pp->p_boffset != 0)
1396 Warning("partition %c: size 0, but offset 0x%012jx",
1397 part, (intmax_t)pp->p_boffset);
1398 if (pp->p_bsize == 0) {
1399 pp->p_boffset = 0;
1400 continue;
1402 if (uuid_is_nil(&pp->p_stor_uuid, NULL))
1403 uuid_create(&pp->p_stor_uuid, NULL);
1405 if (pp->p_boffset < lp->d_pbase) {
1406 fprintf(stderr,
1407 "partition %c: offset out of bounds (%jd)\n",
1408 part, (intmax_t)(pp->p_boffset - lp->d_pbase));
1409 errors++;
1411 if (pp->p_boffset > lp->d_pstop) {
1412 fprintf(stderr,
1413 "partition %c: offset out of bounds (%jd)\n",
1414 part, (intmax_t)(pp->p_boffset - lp->d_pbase));
1415 errors++;
1417 if (pp->p_boffset + pp->p_bsize > lp->d_pstop) {
1418 fprintf(stderr,
1419 "partition %c: size out of bounds (%jd)\n",
1420 part, (intmax_t)(pp->p_boffset - lp->d_pbase));
1421 errors++;
1424 /* check for overlaps */
1425 /* this will check for all possible overlaps once and only once */
1426 for (j = 0; j < i; j++) {
1427 pp2 = &lp->d_partitions[j];
1428 if (pp->p_fstype != FS_VINUM &&
1429 pp2->p_fstype != FS_VINUM &&
1430 part_set[i] && part_set[j]) {
1431 if (pp2->p_boffset < pp->p_boffset + pp->p_bsize &&
1432 (pp2->p_boffset + pp2->p_bsize > pp->p_boffset ||
1433 pp2->p_boffset >= pp->p_boffset)) {
1434 fprintf(stderr,"partitions %c and %c overlap!\n",
1435 j + 'a', i + 'a');
1436 errors++;
1441 for (; i < (int)lp->d_npartitions; i++) {
1442 part = 'a' + i;
1443 pp = &lp->d_partitions[i];
1444 if (pp->p_bsize || pp->p_boffset)
1445 Warning("unused partition %c: size 0x%012jx "
1446 "offset 0x%012jx",
1447 'a' + i, (intmax_t)pp->p_bsize,
1448 (intmax_t)pp->p_boffset);
1450 return (errors);
1454 * When operating on a "virgin" disk, try getting an initial label
1455 * from the associated device driver. This might work for all device
1456 * drivers that are able to fetch some initial device parameters
1457 * without even having access to a (BSD) disklabel, like SCSI disks,
1458 * most IDE drives, or vn devices.
1460 * The device name must be given in its "canonical" form.
1462 static struct disklabel64 dlab;
1464 static struct disklabel64 *
1465 getvirginlabel(void)
1467 struct disklabel64 *dl = &dlab;
1468 int f;
1470 if ((f = open(dkname, O_RDONLY)) == -1) {
1471 warn("cannot open %s", dkname);
1472 return (NULL);
1476 * Generate a virgin disklabel via ioctl
1478 if (ioctl(f, DIOCGDVIRGIN64, dl) < 0) {
1479 l_perror("ioctl DIOCGDVIRGIN64");
1480 close(f);
1481 return (NULL);
1483 close(f);
1484 return (dl);
1487 /*VARARGS1*/
1488 static void
1489 Warning(const char *fmt, ...)
1491 va_list ap;
1493 fprintf(stderr, "Warning, ");
1494 va_start(ap, fmt);
1495 vfprintf(stderr, fmt, ap);
1496 fprintf(stderr, "\n");
1497 va_end(ap);
1500 static void
1501 usage(void)
1503 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",
1504 "usage: disklabel64 [-r] disk",
1505 "\t\t(to read label)",
1506 " disklabel64 -w [-r] [-n] disk [type [packid]]",
1507 "\t\t(to write label with existing boot program)",
1508 " disklabel64 -e [-r] [-n] disk",
1509 "\t\t(to edit label)",
1510 " disklabel64 -R [-r] [-n] disk protofile",
1511 "\t\t(to restore label with existing boot program)",
1512 " disklabel64 -B [-n] [-b boot1 -s boot2] disk [type]",
1513 "\t\t(to install boot program with existing label)",
1514 " disklabel64 -w -B [-n] [-b boot1 -s boot2] disk [type [packid]]",
1515 "\t\t(to write label and boot program)",
1516 " disklabel64 -R -B [-n] [-b boot1 -s boot2] disk protofile [type]",
1517 "\t\t(to restore label and boot program)",
1518 " disklabel64 [-NW] disk",
1519 "\t\t(to write disable/enable label)");
1520 exit(1);