8354 sync regcomp(3C) with upstream (fix make catalog)
[unleashed/tickless.git] / usr / src / cmd / format / menu_command.c
blobaff9d39ce8a7e8721202b862cf2f291027958627
1 /*
2 * CDDL HEADER START
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
19 * CDDL HEADER END
22 * Copyright (c) 1993, 2010, Oracle and/or its affiliates. All rights reserved.
23 * Copyright 2012 Milan Jurik. All rights reserved.
24 * Copyright 2014 Toomas Soome <tsoome@me.com>
25 * Copyright 2015 Nexenta Systems, Inc. All rights reserved.
26 * Copyright 2016 Igor Kozhukhov <ikozhukhov@gmail.com>
27 * Copyright (c) 2016 by Delphix. All rights reserved.
31 * This file contains functions that implement the command menu commands.
34 #include "global.h"
35 #include <time.h>
36 #include <sys/time.h>
37 #include <sys/resource.h>
38 #include <sys/wait.h>
39 #include <strings.h>
40 #include <signal.h>
41 #include <stdlib.h>
42 #include <string.h>
44 #if defined(sparc)
45 #include <sys/hdio.h>
46 #endif /* defined(sparc) */
48 #include "main.h"
49 #include "analyze.h"
50 #include "menu.h"
51 #include "menu_command.h"
52 #include "menu_defect.h"
53 #include "menu_partition.h"
54 #include "param.h"
55 #include "misc.h"
56 #include "label.h"
57 #include "startup.h"
58 #include "partition.h"
59 #include "prompts.h"
60 #include "checkdev.h"
61 #include "io.h"
62 #include "ctlr_scsi.h"
63 #include "auto_sense.h"
64 #include "modify_partition.h"
67 extern struct menu_item menu_partition[];
68 extern struct menu_item menu_analyze[];
69 extern struct menu_item menu_defect[];
72 * Choices for the p_tag vtoc field
74 slist_t ptag_choices[] = {
75 { "unassigned", "", V_UNASSIGNED },
76 { "boot", "", V_BOOT },
77 { "root", "", V_ROOT },
78 { "swap", "", V_SWAP },
79 { "usr", "", V_USR },
80 { "backup", "", V_BACKUP },
81 { "stand", "", V_STAND },
82 { "var", "", V_VAR },
83 { "home", "", V_HOME },
84 { "alternates", "", V_ALTSCTR },
85 { "reserved", "", V_RESERVED },
86 { "system", "", V_SYSTEM },
87 { "BIOS_boot", "", V_BIOS_BOOT },
88 { NULL }
93 * Choices for the p_flag vtoc field
95 slist_t pflag_choices[] = {
96 { "wm", "read-write, mountable", 0 },
97 { "wu", "read-write, unmountable", V_UNMNT },
98 { "rm", "read-only, mountable", V_RONLY },
99 { "ru", "read-only, unmountable", V_RONLY|V_UNMNT },
100 { NULL }
105 * This routine implements the 'disk' command. It allows the user to
106 * select a disk to be current. The list of choices is the list of
107 * disks that were found at startup time.
110 c_disk()
112 struct disk_info *disk;
113 u_ioparam_t ioparam;
114 int i;
115 int ndisks = 0;
116 int blind_select = 0;
117 int deflt;
118 int index;
119 int *defltptr = NULL;
120 int more = 0;
121 int more_quit = 0;
122 int one_line = 0;
123 int tty_lines;
126 * This buffer holds the check() prompt that verifies we've got the right
127 * disk when performing a blind selection. The size should be sufficient
128 * to hold the prompt string, plus 256 characters for the disk name -
129 * way more than should ever be necessary. See the #define in misc.h.
131 char chk_buf[BLIND_SELECT_VER_PROMPT];
133 if (istokenpresent()) {
135 * disk number to be selected is already in the
136 * input stream .
138 TOKEN token, cleantoken;
141 * Get the disk number the user has given.
143 i = 0;
144 for (disk = disk_list; disk != NULL; disk = disk->disk_next) {
145 i++;
148 ioparam.io_bounds.lower = 0;
149 ioparam.io_bounds.upper = i - 1;
150 (void) gettoken(token);
151 clean_token(cleantoken, token);
154 * Convert the token into an integer.
156 if (geti(cleantoken, &index, (int *)NULL))
157 return (0);
160 * Check to be sure it is within the legal bounds.
162 if ((index < 0) || (index >= i)) {
163 err_print("`%d' is out of range.\n", index);
164 return (0);
166 goto checkdisk;
169 fmt_print("\n\nAVAILABLE DISK SELECTIONS:\n");
171 i = 0;
172 if ((option_f == (char *)NULL) && isatty(0) == 1 && isatty(1) == 1) {
174 * We have a real terminal for std input and output, enable
175 * more style of output for disk selection list.
177 more = 1;
178 tty_lines = get_tty_lines();
179 enter_critical();
180 echo_off();
181 charmode_on();
182 exit_critical();
186 * Loop through the list of found disks.
188 for (disk = disk_list; disk != NULL; disk = disk->disk_next) {
190 * If using more output, account 2 lines for each disk.
192 if (more && !more_quit && i && (one_line ||
193 ((2 * i + 1) % (tty_lines - 2) <= 1))) {
194 int c;
197 * Get the next character.
199 fmt_print("- hit space for more or s to select - ");
200 c = getchar();
201 fmt_print("\015");
202 one_line = 0;
204 * Handle display one line command
205 * (return key)
207 if (c == '\012') {
208 one_line++;
210 /* Handle Quit command */
211 if (c == 'q') {
212 fmt_print(
213 " \015");
214 more_quit++;
216 /* Handle ^D command */
217 if (c == '\004')
218 fullabort();
219 /* or get on with the show */
220 if (c == 's' || c == 'S') {
221 fmt_print("%80s\n", " ");
222 break;
226 * If this is the current disk, mark it as
227 * the default.
229 if (cur_disk == disk) {
230 deflt = i;
231 defltptr = &deflt;
233 if (!more || !more_quit)
234 pr_diskline(disk, i);
235 i++;
237 if (more) {
238 enter_critical();
239 charmode_off();
240 echo_on();
241 exit_critical();
245 * Determine total number of disks, and ask the user which disk they
246 * would like to make current.
249 for (disk = disk_list; disk != NULL; disk = disk->disk_next) {
250 ndisks++;
253 ioparam.io_bounds.lower = 0;
254 ioparam.io_bounds.upper = ndisks - 1;
255 index = input(FIO_INT, "Specify disk (enter its number)", ':',
256 &ioparam, defltptr, DATA_INPUT);
258 if (index >= i) {
259 blind_select = 1;
263 * Find the disk chosen. Search through controllers/disks
264 * in the same original order, so we match what the user
265 * chose.
267 checkdisk:
268 i = 0;
269 for (disk = disk_list; disk != NULL; disk = disk->disk_next) {
270 if (i == index)
271 goto found;
272 i++;
275 * Should never happen.
277 impossible("no disk found");
279 found:
280 if (blind_select) {
281 (void) snprintf(chk_buf, sizeof (chk_buf),
282 "Disk %s selected - is this the desired disk? ", disk->disk_name);
283 if (check(chk_buf)) {
284 return (-1);
289 * Update the state. We lock out interrupts so the state can't
290 * get half-updated.
293 enter_critical();
294 init_globals(disk);
295 exit_critical();
298 * If type unknown and interactive, ask user to specify type.
299 * Also, set partition table (best guess) too.
301 if (!option_f && ncyl == 0 && nhead == 0 && nsect == 0 &&
302 (disk->label_type != L_TYPE_EFI)) {
303 (void) c_type();
307 * Get the Solaris Fdisk Partition information
309 if (nhead != 0 && nsect != 0)
310 (void) copy_solaris_part(&cur_disk->fdisk_part);
312 if ((cur_disk->label_type == L_TYPE_EFI) &&
313 (cur_disk->disk_parts->etoc->efi_flags &
314 EFI_GPT_PRIMARY_CORRUPT)) {
315 err_print("Reading the primary EFI GPT label ");
316 err_print("failed. Using backup label.\n");
317 err_print("Use the 'backup' command to restore ");
318 err_print("the primary label.\n");
321 #if defined(_SUNOS_VTOC_16)
323 * If there is no fdisk solaris partition.
325 if (cur_disk->fdisk_part.numsect == 0) {
326 err_print("No Solaris fdisk partition found.\n");
327 goto exit;
329 #endif /* defined(_SUNOS_VTOC_16) */
332 * If the label of the disk is marked dirty,
333 * see if they'd like to label the disk now.
335 if (cur_disk->disk_flags & DSK_LABEL_DIRTY) {
336 if (check("Disk not labeled. Label it now") == 0) {
337 if (write_label()) {
338 err_print("Write label failed\n");
339 } else {
340 cur_disk->disk_flags &= ~DSK_LABEL_DIRTY;
344 exit:
345 return (0);
349 * This routine implements the 'type' command. It allows the user to
350 * specify the type of the current disk. It should be necessary only
351 * if the disk was not labelled or was somehow labelled incorrectly.
352 * The list of legal types for the disk comes from information that was
353 * in the data file.
356 c_type()
358 struct disk_type *type, *tptr, *oldtype;
359 u_ioparam_t ioparam;
360 int i, index, deflt, *defltptr = NULL;
361 struct disk_type disk_type;
362 struct disk_type *d = &disk_type;
363 int first_disk;
364 int auto_conf_choice;
365 int other_choice;
366 struct dk_label label;
367 struct efi_info efi_info;
368 uint64_t maxLBA;
369 char volname[LEN_DKL_VVOL];
370 int volinit = 0;
373 * There must be a current disk.
375 if (cur_disk == NULL) {
376 err_print("Current Disk is not set.\n");
377 return (-1);
379 oldtype = cur_disk->disk_type;
380 type = cur_ctype->ctype_dlist;
382 * Print out the list of choices.
384 fmt_print("\n\nAVAILABLE DRIVE TYPES:\n");
385 first_disk = 0;
386 if (cur_ctype->ctype_ctype == DKC_SCSI_CCS) {
387 auto_conf_choice = 0;
388 fmt_print(" %d. Auto configure\n", first_disk++);
389 } else {
390 auto_conf_choice = -1;
393 i = first_disk;
394 for (tptr = type; tptr != NULL; tptr = tptr->dtype_next) {
396 * If we pass the current type, mark it to be the default.
398 if (cur_dtype == tptr) {
399 deflt = i;
400 defltptr = &deflt;
402 if (cur_disk->label_type == L_TYPE_EFI) {
403 continue;
405 if (tptr->dtype_asciilabel)
406 fmt_print(" %d. %s\n", i++,
407 tptr->dtype_asciilabel);
409 other_choice = i;
410 fmt_print(" %d. other\n", i);
411 ioparam.io_bounds.lower = 0;
412 ioparam.io_bounds.upper = i;
414 * Ask the user which type the disk is.
416 index = input(FIO_INT, "Specify disk type (enter its number)", ':',
417 &ioparam, defltptr, DATA_INPUT);
419 * Find the type they chose.
421 if (index == auto_conf_choice) {
422 float scaled;
423 diskaddr_t nblks;
424 int nparts;
427 * User chose "auto configure".
429 (void) strcpy(x86_devname, cur_disk->disk_name);
430 switch (cur_disk->label_type) {
431 case L_TYPE_SOLARIS:
432 if ((tptr = auto_sense(cur_file, 1, &label)) == NULL) {
433 err_print("Auto configure failed\n");
434 return (-1);
436 fmt_print("%s: configured with capacity of ",
437 cur_disk->disk_name);
438 nblks = (diskaddr_t)tptr->dtype_ncyl *
439 tptr->dtype_nhead * tptr->dtype_nsect;
440 scaled = bn2mb(nblks);
441 if (scaled > 1024.0) {
442 fmt_print("%1.2fGB\n", scaled/1024.0);
443 } else {
444 fmt_print("%1.2fMB\n", scaled);
446 fmt_print("<%s cyl %d alt %d hd %d sec %d>\n",
447 tptr->dtype_asciilabel, tptr->dtype_ncyl,
448 tptr->dtype_acyl, tptr->dtype_nhead,
449 tptr->dtype_nsect);
450 break;
451 case L_TYPE_EFI:
452 if ((tptr = auto_efi_sense(cur_file, &efi_info))
453 == NULL) {
454 err_print("Auto configure failed\n");
455 return (-1);
457 fmt_print("%s: configured with capacity of ",
458 cur_disk->disk_name);
459 scaled = bn2mb(efi_info.capacity);
460 if (scaled > 1024.0) {
461 fmt_print("%1.2fGB\n", scaled/1024.0);
462 } else {
463 fmt_print("%1.2fMB\n", scaled);
465 cur_blksz = efi_info.e_parts->efi_lbasize;
466 print_efi_string(efi_info.vendor, efi_info.product,
467 efi_info.revision, efi_info.capacity);
468 fmt_print("\n");
469 for (nparts = 0; nparts < cur_parts->etoc->efi_nparts;
470 nparts++) {
471 if (cur_parts->etoc->efi_parts[nparts].p_tag ==
472 V_RESERVED) {
473 if (cur_parts->etoc->efi_parts[nparts].
474 p_name) {
475 (void) strcpy(volname,
476 cur_parts->etoc->efi_parts
477 [nparts].p_name);
478 volinit = 1;
480 break;
483 enter_critical();
484 if (delete_disk_type(cur_disk->disk_type) != 0) {
485 fmt_print("Autoconfiguration failed.\n");
486 return (-1);
488 cur_disk->disk_type = tptr;
489 cur_disk->disk_parts = tptr->dtype_plist;
490 init_globals(cur_disk);
491 exit_critical();
492 if (volinit) {
493 for (nparts = 0; nparts <
494 cur_parts->etoc->efi_nparts; nparts++) {
495 if (cur_parts->etoc->efi_parts[nparts].p_tag ==
496 V_RESERVED) {
497 (void) strcpy(
498 cur_parts->etoc->efi_parts[nparts].p_name,
499 volname);
500 (void) strlcpy(cur_disk->v_volume, volname,
501 LEN_DKL_VVOL);
502 break;
506 return (0);
507 default:
508 /* Should never happen */
509 return (-1);
511 } else if ((index == other_choice) && (cur_label == L_TYPE_SOLARIS)) {
513 * User chose "other".
514 * Get the standard information on the new type.
515 * Put all information in a tmp structure, in
516 * case user aborts.
518 bzero((char *)d, sizeof (struct disk_type));
520 d->dtype_ncyl = get_ncyl();
521 d->dtype_acyl = get_acyl(d->dtype_ncyl);
522 d->dtype_pcyl = get_pcyl(d->dtype_ncyl, d->dtype_acyl);
523 d->dtype_nhead = get_nhead();
524 d->dtype_phead = get_phead(d->dtype_nhead, &d->dtype_options);
525 d->dtype_nsect = get_nsect();
526 d->dtype_psect = get_psect(&d->dtype_options);
527 d->dtype_bpt = get_bpt(d->dtype_nsect, &d->dtype_options);
528 d->dtype_rpm = get_rpm();
529 d->dtype_fmt_time = get_fmt_time(&d->dtype_options);
530 d->dtype_cyl_skew = get_cyl_skew(&d->dtype_options);
531 d->dtype_trk_skew = get_trk_skew(&d->dtype_options);
532 d->dtype_trks_zone = get_trks_zone(&d->dtype_options);
533 d->dtype_atrks = get_atrks(&d->dtype_options);
534 d->dtype_asect = get_asect(&d->dtype_options);
535 d->dtype_cache = get_cache(&d->dtype_options);
536 d->dtype_threshold = get_threshold(&d->dtype_options);
537 d->dtype_prefetch_min = get_min_prefetch(&d->dtype_options);
538 d->dtype_prefetch_max = get_max_prefetch(d->dtype_prefetch_min,
539 &d->dtype_options);
540 d->dtype_bps = get_bps();
541 #if defined(sparc)
542 d->dtype_dr_type = 0;
543 #endif /* defined(sparc) */
545 d->dtype_asciilabel = get_asciilabel();
547 * Add the new type to the list of possible types for
548 * this controller. We lock out interrupts so the lists
549 * can't get munged. We put off actually allocating the
550 * structure till here in case the user wanted to
551 * interrupt while still inputting information.
553 enter_critical();
554 tptr = (struct disk_type *)zalloc(sizeof (struct disk_type));
555 if (type == NULL)
556 cur_ctype->ctype_dlist = tptr;
557 else {
558 while (type->dtype_next != NULL)
559 type = type->dtype_next;
560 type->dtype_next = tptr;
562 bcopy((char *)d, (char *)tptr, sizeof (disk_type));
563 tptr->dtype_next = NULL;
565 * the new disk type does not have any defined
566 * partition table . Hence copy the current partition
567 * table if possible else create a default
568 * paritition table.
570 new_partitiontable(tptr, oldtype);
571 } else if ((index == other_choice) && (cur_label == L_TYPE_EFI)) {
572 maxLBA = get_mlba();
573 cur_parts->etoc->efi_last_lba = maxLBA;
574 cur_parts->etoc->efi_last_u_lba = maxLBA - 34;
575 for (i = 0; i < cur_parts->etoc->efi_nparts; i++) {
576 cur_parts->etoc->efi_parts[i].p_start = 0;
577 cur_parts->etoc->efi_parts[i].p_size = 0;
578 cur_parts->etoc->efi_parts[i].p_tag = V_UNASSIGNED;
580 cur_parts->etoc->efi_parts[8].p_start =
581 maxLBA - 34 - (1024 * 16);
582 cur_parts->etoc->efi_parts[8].p_size = (1024 * 16);
583 cur_parts->etoc->efi_parts[8].p_tag = V_RESERVED;
584 if (write_label()) {
585 err_print("Write label failed\n");
586 } else {
587 cur_disk->disk_flags &= ~DSK_LABEL_DIRTY;
589 return (0);
590 } else {
592 * User picked an existing disk type.
594 i = first_disk;
595 tptr = type;
596 while (i < index) {
597 if (tptr->dtype_asciilabel) {
598 i++;
600 tptr = tptr->dtype_next;
602 if ((tptr->dtype_asciilabel == NULL) &&
603 (tptr->dtype_next != NULL)) {
604 while (tptr->dtype_asciilabel == NULL) {
605 tptr = tptr->dtype_next;
610 * Check for mounted file systems in the format zone.
611 * One potential problem with this would be that check()
612 * always returns 'yes' when running out of a file. However,
613 * it is actually ok because we don't let the program get
614 * started if there are mounted file systems and we are
615 * running from a file.
617 if ((tptr != oldtype) &&
618 checkmount((diskaddr_t)-1, (diskaddr_t)-1)) {
619 err_print(
620 "Cannot set disk type while it has mounted "
621 "partitions.\n\n");
622 return (-1);
625 * check for partitions being used for swapping in format zone
627 if ((tptr != oldtype) &&
628 checkswap((diskaddr_t)-1, (diskaddr_t)-1)) {
629 err_print("Cannot set disk type while its partition are "
630 "currently being used for swapping.\n");
631 return (-1);
635 * Check for partitions being used in SVM, VxVM or LU devices
638 if ((tptr != oldtype) &&
639 checkdevinuse(cur_disk->disk_name, (diskaddr_t)-1,
640 (diskaddr_t)-1, 0, 0)) {
641 err_print("Cannot set disk type while its "
642 "partitions are currently in use.\n");
643 return (-1);
646 * If the type selected is different from the previous type,
647 * mark the disk as not labelled and reload the current
648 * partition info. This is not essential but probably the
649 * right thing to do, since the size of the disk has probably
650 * changed.
652 enter_critical();
653 if (tptr != oldtype) {
654 cur_disk->disk_type = tptr;
655 cur_disk->disk_parts = NULL;
656 cur_disk->disk_flags &= ~DSK_LABEL;
659 * Initialize the state of the current disk.
661 init_globals(cur_disk);
662 (void) get_partition();
663 exit_critical();
666 * If the label of the disk is marked dirty,
667 * see if they'd like to label the disk now.
669 if (cur_disk->disk_flags & DSK_LABEL_DIRTY) {
670 if (check("Disk not labeled. Label it now") == 0) {
671 if (write_label()) {
672 err_print("Write label failed\n");
673 } else {
674 cur_disk->disk_flags &= ~DSK_LABEL_DIRTY;
679 return (0);
683 * This routine implements the 'partition' command. It simply runs
684 * the partition menu.
687 c_partition()
691 * There must be a current disk type and a current disk
693 if (cur_dtype == NULL) {
694 err_print("Current Disk Type is not set.\n");
695 return (-1);
698 * Check for a valid fdisk table entry for Solaris
700 if (!good_fdisk()) {
701 return (-1);
704 cur_menu++;
705 last_menu = cur_menu;
707 #ifdef not
709 * If there is no current partition table, make one. This is
710 * so the commands within the menu never have to check for
711 * a non-existent table.
713 if (cur_parts == NULL)
714 err_print("making partition.\n");
715 make_partition();
716 #endif /* not */
719 * Run the menu.
721 run_menu(menu_partition, "PARTITION", "partition", 0);
722 cur_menu--;
723 return (0);
727 * This routine implements the 'current' command. It describes the
728 * current disk.
731 c_current()
735 * If there is no current disk, say so. Note that this is
736 * not an error since it is a legitimate response to the inquiry.
738 if (cur_disk == NULL) {
739 fmt_print("No Current Disk.\n");
740 return (0);
743 * Print out the info we have on the current disk.
745 fmt_print("Current Disk = %s", cur_disk->disk_name);
746 if (chk_volname(cur_disk)) {
747 fmt_print(": ");
748 print_volname(cur_disk);
750 fmt_print("\n");
751 if (cur_disk->devfs_name != NULL) {
752 if (cur_dtype == NULL) {
753 fmt_print("<type unknown>\n");
754 } else if (cur_label == L_TYPE_SOLARIS) {
755 fmt_print("<%s cyl %d alt %d hd %d sec %d>\n",
756 cur_dtype->dtype_asciilabel, ncyl,
757 acyl, nhead, nsect);
758 } else if (cur_label == L_TYPE_EFI) {
759 print_efi_string(cur_dtype->vendor,
760 cur_dtype->product, cur_dtype->revision,
761 cur_dtype->capacity);
762 fmt_print("\n");
764 fmt_print("%s\n", cur_disk->devfs_name);
765 } else {
766 fmt_print("%s%d: <", cur_ctlr->ctlr_dname,
767 cur_disk->disk_dkinfo.dki_unit);
768 if (cur_dtype == NULL) {
769 fmt_print("type unknown");
770 } else if (cur_label == L_TYPE_SOLARIS) {
771 fmt_print("%s cyl %d alt %d hd %d sec %d",
772 cur_dtype->dtype_asciilabel, ncyl,
773 acyl, nhead, nsect);
774 } else if (cur_label == L_TYPE_EFI) {
775 print_efi_string(cur_dtype->vendor,
776 cur_dtype->product, cur_dtype->revision,
777 cur_dtype->capacity);
778 fmt_print("\n");
780 fmt_print(">\n");
782 fmt_print("\n");
783 return (0);
786 * This routine implements the 'format' command. It allows the user
787 * to format and verify any portion of the disk.
790 c_format()
792 diskaddr_t start, end;
793 time_t clock;
794 int format_time, format_tracks, format_cyls;
795 int format_interval;
796 diskaddr_t deflt;
797 int status;
798 u_ioparam_t ioparam;
799 struct scsi_inquiry *inq;
800 char rawbuf[MAX_MODE_SENSE_SIZE];
801 struct scsi_capacity_16 capacity;
802 struct vpd_hdr *vpdhdr;
803 uint8_t protect;
804 uint8_t pagecode;
805 uint8_t spt;
806 uint8_t p_type;
807 uint8_t prot_flag[NUM_PROT_TYPE] = {1, 0, 0, 0};
808 int i;
809 char *prot_descriptor[NUM_PROT_TYPE] = {
810 "Protection Information is disabled.",
811 "Protection Information is enabled.",
812 "Protection Information is enabled.",
813 "Protection Information is enabled.", };
816 * There must be a current disk type and a current disk
818 if (cur_dtype == NULL) {
819 err_print("Current Disk Type is not set.\n");
820 return (-1);
824 * There must be a format routine in cur_ops structure to have
825 * this routine work.
827 if (cur_ops->op_format == NULL) {
828 err_print(
829 "Cannot format this drive. Please use your Manufacturer supplied formatting "
830 "utility.\n");
831 return (-1);
835 * There must be a current defect list. Except for
836 * unformatted SCSI disks. For them the defect list
837 * can only be retrieved after formatting the disk.
839 if ((cur_ctype->ctype_flags & CF_SCSI) && !EMBEDDED_SCSI &&
840 (cur_ctype->ctype_flags & CF_DEFECTS) &&
841 ! (cur_flags & DISK_FORMATTED)) {
842 cur_list.flags |= LIST_RELOAD;
844 } else if (cur_list.list == NULL && !EMBEDDED_SCSI) {
845 err_print("Current Defect List must be initialized.\n");
846 return (-1);
849 * Ask for the bounds of the format. We always use the whole
850 * disk as the default, since that is the most likely case.
851 * Note, for disks which must be formatted accross the whole disk,
852 * don't bother the user.
854 ioparam.io_bounds.lower = start = 0;
855 if (cur_label == L_TYPE_SOLARIS) {
856 if (cur_ctype->ctype_flags & CF_SCSI) {
857 ioparam.io_bounds.upper = end = datasects() - 1;
858 } else {
859 ioparam.io_bounds.upper = end = physsects() - 1;
861 } else {
862 ioparam.io_bounds.upper = end = cur_parts->etoc->efi_last_lba;
865 if (! (cur_ctlr->ctlr_flags & DKI_FMTVOL)) {
866 deflt = ioparam.io_bounds.lower;
867 start = input(FIO_BN,
868 "Enter starting block number", ':',
869 &ioparam, (int *)&deflt, DATA_INPUT);
870 ioparam.io_bounds.lower = start;
871 deflt = ioparam.io_bounds.upper;
872 end = input(FIO_BN,
873 "Enter ending block number", ':',
874 &ioparam, (int *)&deflt, DATA_INPUT);
877 * Some disks can format tracks. Make sure the whole track is
878 * specified for them.
880 if (cur_ctlr->ctlr_flags & DKI_FMTTRK) {
881 if (bn2s(start) != 0 ||
882 bn2s(end) != sectors(bn2h(end)) - 1) {
883 err_print("Controller requires formatting of ");
884 err_print("entire tracks.\n");
885 return (-1);
889 * Check for mounted file systems in the format zone, and if we
890 * find any, make sure they are really serious. One potential
891 * problem with this would be that check() always returns 'yes'
892 * when running out of a file. However, it is actually ok
893 * because we don't let the program get started if there are
894 * mounted file systems and we are running from a file.
896 if (checkmount(start, end)) {
897 err_print(
898 "Cannot format disk while it has mounted partitions.\n\n");
899 return (-1);
902 * check for partitions being used for swapping in format zone
904 if (checkswap(start, end)) {
905 err_print("Cannot format disk while its partition are \
906 currently being used for swapping.\n");
907 return (-1);
910 * Check for partitions being used in SVM, VxVM or LU devices
911 * in this format zone
913 if (checkdevinuse(cur_disk->disk_name, start, end, 0, 0)) {
914 err_print("Cannot format disk while its partitions "
915 "are currently in use.\n");
916 return (-1);
919 if (cur_disk->disk_lbasize != DEV_BSIZE) {
920 fmt_print("Current disk sector size is %d Byte, format\n"
921 "will change the sector size to 512 Byte. ",
922 cur_disk->disk_lbasize);
923 if (check("Continue")) {
924 return (-1);
929 * set the default protection type
931 prot_type = PROT_TYPE_0;
934 * Check if the protect information of this disk is enabled
936 if (uscsi_inquiry(cur_file, rawbuf, sizeof (rawbuf))) {
937 err_print("Inquiry failed\n");
938 return (-1);
940 inq = (struct scsi_inquiry *)rawbuf;
941 protect = inq->inq_protect;
942 if (protect == 0) {
943 fmt_print("The protection information is not enabled\n");
944 fmt_print(
945 "The disk will be formatted with protection type 0\n");
946 } else {
947 (void) memset(rawbuf, 0, MAX_MODE_SENSE_SIZE);
948 if (uscsi_inquiry_page_86h(cur_file, rawbuf, sizeof (rawbuf))) {
949 err_print("Inquiry with page 86h failed\n");
950 return (-1);
952 vpdhdr = (struct vpd_hdr *)rawbuf;
953 pagecode = vpdhdr->page_code;
954 if (pagecode != 0x86) {
955 err_print("Inquiry with page 86h failed\n");
956 return (-1);
958 spt = (rawbuf[4] << 2) >> 5;
959 fmt_print("This disk can support protection types:\n");
961 switch (spt) {
962 case 0:
963 prot_flag[1] = 1;
964 break;
965 case 1:
966 prot_flag[1] = 1;
967 prot_flag[2] = 1;
968 break;
969 case 2:
970 prot_flag[2] = 1;
971 break;
972 case 3:
973 prot_flag[1] = 1;
974 prot_flag[3] = 1;
975 break;
976 case 4:
977 prot_flag[3] = 1;
978 break;
979 case 5:
980 prot_flag[2] = 1;
981 prot_flag[3] = 1;
982 break;
983 case 7:
984 prot_flag[1] = 1;
985 prot_flag[2] = 1;
986 prot_flag[3] = 1;
987 break;
988 default:
989 err_print(
990 "Invalid supported protection types\n");
991 return (-1);
993 for (i = 0; i < NUM_PROT_TYPE; i++) {
994 if (prot_flag[i] == 1) {
995 fmt_print("[%d] TYPE_%d : ", i, i);
996 fmt_print("%s\n", prot_descriptor[i]);
1001 * Get the current protection type
1003 if (uscsi_read_capacity_16(cur_file, &capacity)) {
1004 err_print("Read capacity_16 failed\n");
1005 return (-1);
1007 p_type = get_cur_protection_type(&capacity);
1008 fmt_print("\nThe disk is currently formatted with TYPE_%d.\n",
1009 p_type);
1012 * Ask user what protection type to use
1014 ioparam.io_bounds.lower = PROT_TYPE_0;
1015 ioparam.io_bounds.upper = PROT_TYPE_3;
1016 prot_type = input(FIO_INT, "Specify the New Protection Type",
1017 ':', &ioparam, NULL, DATA_INPUT);
1019 * if get a unsupported protection type, then use the
1020 * current type: p_type.
1022 if (prot_flag[prot_type] == 0) {
1023 fmt_print("Unsupported protection type.\n");
1024 prot_type = p_type;
1026 fmt_print("The disk will be formatted to type %d\n", prot_type);
1029 if (SCSI && (format_time = scsi_format_time()) > 0) {
1030 fmt_print(
1031 "\nReady to format. Formatting cannot be interrupted\n"
1032 "and takes %d minutes (estimated). ", format_time);
1034 } else if (cur_dtype->dtype_options & SUP_FMTTIME) {
1036 * Formatting time is (2 * time of 1 spin * number of
1037 * tracks) + (step rate * number of cylinders) rounded
1038 * up to the nearest minute. Note, a 10% fudge factor
1039 * is thrown in for insurance.
1041 if (cur_dtype->dtype_fmt_time == 0)
1042 cur_dtype->dtype_fmt_time = 2;
1044 format_tracks = ((end-start) / cur_dtype->dtype_nsect) + 1;
1045 format_cyls = format_tracks / cur_dtype->dtype_nhead;
1046 format_tracks = format_tracks * cur_dtype->dtype_fmt_time;
1049 * ms.
1051 format_time = ((60000 / cur_dtype->dtype_rpm) +1) *
1052 format_tracks + format_cyls * 7;
1054 * 20% done tick (sec)
1056 format_interval = format_time / 5000;
1058 * min.
1060 format_time = (format_time + 59999) / 60000;
1063 * Check format time values and make adjustments
1064 * to prevent sleeping too long (forever?) or
1065 * too short.
1067 if (format_time <= 1) {
1069 * Format time is less than 1 min..
1071 format_time = 1;
1074 if (format_interval < 11) {
1075 /* Format time is less than 1 minute. */
1076 if (format_interval < 2)
1077 format_interval = 2; /* failsafe */
1078 format_interval = 10;
1079 } else {
1080 /* Format time is greater than 1 minute. */
1081 format_interval -= 10;
1084 fmt_print(
1085 "Ready to format. Formatting cannot be interrupted\n"
1086 "and takes %d minutes (estimated). ", format_time);
1087 } else {
1088 fmt_print(
1089 "Ready to format. Formatting cannot be interrupted.\n");
1091 if (check("Continue")) {
1092 return (-1);
1096 * Print the time so that the user will know when format started.
1097 * Lock out interrupts. This could be a problem, since it could
1098 * cause the user to sit for quite awhile with no control, but we
1099 * don't have any other good way of keeping their gun from going off.
1101 clock = time((time_t *)0);
1102 fmt_print("Beginning format. The current time is %s\n",
1103 ctime(&clock));
1104 enter_critical();
1106 * Mark the defect list dirty so it will be rewritten when we are
1107 * done. It is possible to qualify this so it doesn't always
1108 * get rewritten, but it's not worth the trouble.
1109 * Note: no defect lists for embedded scsi drives.
1111 if (!EMBEDDED_SCSI) {
1112 cur_list.flags |= LIST_DIRTY;
1115 * If we are formatting over any of the labels, mark the label
1116 * dirty so it will be rewritten.
1118 if (cur_disk->label_type == L_TYPE_SOLARIS) {
1119 if (start < totalsects() && end >= datasects()) {
1120 if (cur_disk->disk_flags & DSK_LABEL)
1121 cur_flags |= LABEL_DIRTY;
1123 } else if (cur_disk->label_type == L_TYPE_EFI) {
1124 if (start < 34) {
1125 if (cur_disk->disk_flags & DSK_LABEL)
1126 cur_flags |= LABEL_DIRTY;
1129 if (start == 0) {
1130 cur_flags |= LABEL_DIRTY;
1133 * Do the format. bugid 1009138 removed the use of fork to
1134 * background the format and print a tick.
1137 status = (*cur_ops->op_format)(start, end, &cur_list);
1138 if (status) {
1139 exit_critical();
1140 err_print("failed\n");
1141 return (-1);
1143 fmt_print("done\n");
1144 if (option_msg && diag_msg) {
1145 clock = time((time_t *)0);
1146 fmt_print("The current time is %s\n", ctime(&clock));
1148 cur_flags |= DISK_FORMATTED;
1150 * If the defect list or label is dirty, write them out again.
1151 * Note, for SCSI we have to wait til now to load defect list
1152 * since we can't access it until after formatting a virgin disk.
1154 /* enter_critical(); */
1155 if (cur_list.flags & LIST_RELOAD) {
1156 assert(!EMBEDDED_SCSI);
1157 if (*cur_ops->op_ex_man == NULL ||
1158 (*cur_ops->op_ex_man)(&cur_list)) {
1159 err_print("Warning: unable to reload defect list\n");
1160 cur_list.flags &= ~LIST_DIRTY;
1161 return (-1);
1163 cur_list.flags |= LIST_DIRTY;
1166 if (cur_list.flags & LIST_DIRTY) {
1167 assert(!EMBEDDED_SCSI);
1168 write_deflist(&cur_list);
1169 cur_list.flags = 0;
1171 if (cur_flags & LABEL_DIRTY) {
1172 (void) write_label();
1173 cur_flags &= ~LABEL_DIRTY;
1176 * Come up for air, since the verify step does not need to
1177 * be atomic (it does it's own lockouts when necessary).
1179 exit_critical();
1181 * If we are supposed to verify, we do the 'write' test over
1182 * the format zone. The rest of the analysis parameters are
1183 * left the way they were.
1185 if (scan_auto) {
1186 scan_entire = 0;
1187 scan_lower = start;
1188 scan_upper = end;
1189 fmt_print("\nVerifying media...");
1190 status = do_scan(SCAN_PATTERN, F_SILENT);
1193 * If the defect list or label is dirty, write them out again.
1195 if (cur_list.flags & LIST_DIRTY) {
1196 assert(!EMBEDDED_SCSI);
1197 cur_list.flags = 0;
1198 write_deflist(&cur_list);
1200 if (cur_flags & LABEL_DIRTY) {
1201 cur_flags &= ~LABEL_DIRTY;
1202 (void) write_label();
1204 return (status);
1208 * This routine implements the 'repair' command. It allows the user
1209 * to reallocate sectors on the disk that have gone bad.
1212 c_repair()
1214 diskaddr_t bn;
1215 int status;
1216 u_ioparam_t ioparam;
1217 char *buf;
1218 int buf_is_good;
1219 int block_has_error;
1220 int i;
1223 * There must be a current disk type (and therefore a current disk).
1225 if (cur_dtype == NULL) {
1226 err_print("Current Disk Type is not set.\n");
1227 return (-1);
1230 * The current disk must be formatted for repair to work.
1232 if (!(cur_flags & DISK_FORMATTED)) {
1233 err_print("Current Disk is unformatted.\n");
1234 return (-1);
1237 * Check for a valid fdisk table entry for Solaris
1239 if (!good_fdisk()) {
1240 return (-1);
1243 * Repair is an optional command for controllers, so it may
1244 * not be supported.
1246 if (cur_ops->op_repair == NULL) {
1247 err_print("Controller does not support repairing.\n");
1248 err_print("or disk supports automatic defect management.\n");
1249 return (-1);
1252 * There must be a defect list for non-embedded scsi devices,
1253 * since we will add to it.
1255 if (!EMBEDDED_SCSI && cur_list.list == NULL) {
1256 err_print("Current Defect List must be initialized.\n");
1257 return (-1);
1260 * Ask the user which sector has gone bad.
1262 ioparam.io_bounds.lower = 0;
1263 if (cur_disk->label_type == L_TYPE_SOLARIS) {
1264 ioparam.io_bounds.upper = physsects() - 1;
1265 } else {
1266 ioparam.io_bounds.upper = cur_parts->etoc->efi_last_lba;
1268 bn = input(FIO_BN,
1269 "Enter absolute block number of defect", ':',
1270 &ioparam, (int *)NULL, DATA_INPUT);
1272 * Check to see if there is a mounted file system over the
1273 * specified sector. If there is, make sure the user is
1274 * really serious.
1276 if (checkmount(bn, bn)) {
1277 if (check("Repair is in a mounted partition, continue"))
1278 return (-1);
1281 * check for partitions being used for swapping in format zone
1283 if (checkswap(bn, bn)) {
1284 if (check("Repair is in a partition which is currently \
1285 being used for swapping.\ncontinue"))
1286 return (-1);
1289 if (checkdevinuse(cur_disk->disk_name, bn, bn, 0, 0)) {
1290 if (check("Repair is in a partition which is currently "
1291 "in use.\ncontinue"))
1292 return (-1);
1295 buf = zalloc((cur_disk->disk_lbasize == 0) ?
1296 SECSIZE : cur_disk->disk_lbasize);
1299 * Try to read the sector before repairing it. If we can
1300 * get good data out of it, we can write that data back
1301 * after the repair. If the sector looks ok, ask the
1302 * user to confirm the repair, since it doesn't appear
1303 * necessary. Try reading the block several times to
1304 * see if we can read it consistently.
1306 * First, let's see if the block appears to have problems...
1308 block_has_error = 1;
1309 for (i = 0; i < 5; i++) {
1310 status = (*cur_ops->op_rdwr)(DIR_READ, cur_file, bn,
1311 1, buf, (F_SILENT | F_ALLERRS), NULL);
1312 if (status)
1313 break; /* one of the tries failed */
1315 if (status == 0) {
1316 block_has_error = 0;
1317 if (check("\
1318 This block doesn't appear to be bad. Repair it anyway")) {
1319 free(buf);
1320 return (0);
1324 * Last chance...
1326 if (check("Ready to repair defect, continue")) {
1327 free(buf);
1328 return (-1);
1331 * We're committed to repairing it. Try to get any good
1332 * data out of the block if possible. Note that we do
1333 * not set the F_ALLERRS flag.
1335 buf_is_good = 0;
1336 for (i = 0; i < 5; i++) {
1337 status = (*cur_ops->op_rdwr)(DIR_READ, cur_file, bn,
1338 1, buf, F_SILENT, NULL);
1339 if (status == 0) {
1340 buf_is_good = 1;
1341 break;
1345 * Lock out interrupts so the disk can't get out of sync with
1346 * the defect list.
1348 enter_critical();
1350 fmt_print("Repairing ");
1351 if (block_has_error) {
1352 fmt_print("%s error on ", buf_is_good ? "soft" : "hard");
1354 fmt_print("block %llu (", bn);
1355 pr_dblock(fmt_print, bn);
1356 fmt_print(")...");
1358 * Do the repair.
1360 status = (*cur_ops->op_repair)(bn, F_NORMAL);
1361 if (status) {
1362 fmt_print("failed.\n\n");
1363 } else {
1365 * The repair worked. Write the old data to the new
1366 * block if we were able to read it, otherwise
1367 * zero out the new block. If it looks like the
1368 * new block is bad, let the user know that, too.
1369 * Should we attempt auto-repair in this case?
1371 fmt_print("ok.\n");
1372 if (!buf_is_good) {
1373 bzero(buf, cur_disk->disk_lbasize);
1375 status = (*cur_ops->op_rdwr)(DIR_WRITE, cur_file, bn,
1376 1, buf, (F_SILENT | F_ALLERRS), NULL);
1377 if (status == 0) {
1378 status = (*cur_ops->op_rdwr)(DIR_READ, cur_file,
1379 bn, 1, buf, (F_SILENT | F_ALLERRS), NULL);
1381 if (status) {
1382 fmt_print("The new block %llu (", bn);
1383 pr_dblock(fmt_print, bn);
1384 fmt_print(") also appears defective.\n");
1386 fmt_print("\n");
1388 * Add the bad sector to the defect list, write out
1389 * the defect list, and kill off the working list so
1390 * it will get synced up with the current defect list
1391 * next time we need it.
1393 * For embedded scsi, we don't require a defect list.
1394 * However, if we have one, add the defect if the
1395 * list includes the grown list. If not, kill it
1396 * to force a resync if we need the list later.
1398 if (EMBEDDED_SCSI) {
1399 if (cur_list.list != NULL) {
1400 if (cur_list.flags & LIST_PGLIST) {
1401 add_ldef(bn, &cur_list);
1402 } else {
1403 kill_deflist(&cur_list);
1406 } else if (cur_ctype->ctype_flags & CF_WLIST) {
1407 kill_deflist(&cur_list);
1408 if (*cur_ops->op_ex_cur != NULL) {
1409 (*cur_ops->op_ex_cur)(&cur_list);
1410 fmt_print("Current list updated\n");
1412 } else {
1413 add_ldef(bn, &cur_list);
1414 write_deflist(&cur_list);
1416 kill_deflist(&work_list);
1418 exit_critical();
1419 free(buf);
1422 * Return status.
1424 return (status);
1428 * This routine implements the 'show' command. It translates a disk
1429 * block given in any format into decimal, hexadecimal, and
1430 * cylinder/head/sector format.
1433 c_show()
1435 u_ioparam_t ioparam;
1436 diskaddr_t bn;
1439 * There must be a current disk type, so we will know the geometry.
1441 if (cur_dtype == NULL) {
1442 err_print("Current Disk Type is not set.\n");
1443 return (-1);
1446 * Ask the user for a disk block.
1448 ioparam.io_bounds.lower = 0;
1449 if (cur_disk->label_type == L_TYPE_SOLARIS) {
1450 ioparam.io_bounds.upper = physsects() - 1;
1451 } else {
1452 ioparam.io_bounds.upper = cur_parts->etoc->efi_last_lba;
1454 bn = input(FIO_BN, "Enter a disk block", ':',
1455 &ioparam, (int *)NULL, DATA_INPUT);
1457 * Echo it back.
1459 fmt_print("Disk block = %lld = 0x%llx = (", bn, bn);
1460 pr_dblock(fmt_print, bn);
1461 fmt_print(")\n\n");
1462 return (0);
1466 * This routine implements the 'label' command. It writes the
1467 * primary and backup labels onto the current disk.
1470 c_label()
1472 int status;
1473 int deflt, *defltptr = NULL;
1476 * There must be a current disk type (and therefore a current disk).
1478 if (cur_dtype == NULL) {
1479 err_print("Current Disk Type is not set.\n");
1480 return (-1);
1483 * The current disk must be formatted to label it.
1485 if (!(cur_flags & DISK_FORMATTED)) {
1486 err_print("Current Disk is unformatted.\n");
1487 return (-1);
1490 * Check for a valid fdisk table entry for Solaris
1492 if (!good_fdisk()) {
1493 return (-1);
1496 * Check to see if there are any mounted file systems anywhere
1497 * on the current disk. If so, refuse to label the disk, but
1498 * only if the partitions would change for the mounted partitions.
1501 if (checkmount((diskaddr_t)-1, (diskaddr_t)-1)) {
1502 /* Bleagh, too descriptive */
1503 if (check_label_with_mount()) {
1504 err_print("Cannot label disk while it has "
1505 "mounted partitions.\n\n");
1506 return (-1);
1511 * check to see if there any partitions being used for swapping
1512 * on the current disk. If so, refuse to label the disk, but
1513 * only if the partitions would change for the mounted partitions.
1515 if (checkswap((diskaddr_t)-1, (diskaddr_t)-1)) {
1516 if (check_label_with_swap()) {
1517 err_print("Cannot label disk while its "
1518 "partitions are currently being used for "
1519 "swapping.\n");
1520 return (-1);
1525 * Check to see if any partitions used for svm, vxvm or live upgrade
1526 * are on the disk. If so, refuse to label the disk, but only
1527 * if we are trying to shrink a partition in use.
1529 if (checkdevinuse(cur_disk->disk_name, (diskaddr_t)-1,
1530 (diskaddr_t)-1, 0, 1)) {
1531 err_print("Cannot label disk when "
1532 "partitions are in use as described.\n");
1533 return (-1);
1537 * If there is not a current partition map, warn the user we
1538 * are going to use the default. The default is the first
1539 * partition map we encountered in the data file. If there is
1540 * no default we give up.
1542 if (cur_parts == NULL) {
1543 fmt_print("Current Partition Table is not set, "
1544 "using default.\n");
1545 cur_disk->disk_parts = cur_parts = cur_dtype->dtype_plist;
1546 if (cur_parts == NULL) {
1547 err_print("No default available, cannot label.\n");
1548 return (-1);
1552 * If expert (-e) mode, then ask user if they wish
1553 * to change the current solaris label into an EFI one
1555 if (expert_mode) {
1556 #if defined(_SUNOS_VTOC_8)
1557 int i;
1558 #endif
1559 int choice;
1560 u_ioparam_t ioparam;
1561 struct extvtoc vtoc;
1562 struct dk_label label;
1563 struct dk_gpt *vtoc64;
1564 struct efi_info efinfo;
1565 struct disk_type *dptr;
1567 /* Ask user what label to use */
1568 fmt_print("[0] SMI Label\n");
1569 fmt_print("[1] EFI Label\n");
1570 ioparam.io_bounds.lower = 0;
1571 ioparam.io_bounds.upper = 1;
1572 if ((cur_label == L_TYPE_SOLARIS) &&
1573 (cur_disk->fdisk_part.systid != EFI_PMBR))
1574 deflt = L_TYPE_SOLARIS;
1575 else
1576 deflt = L_TYPE_EFI;
1577 defltptr = &deflt;
1578 choice = input(FIO_INT, "Specify Label type", ':',
1579 &ioparam, defltptr, DATA_INPUT);
1580 if ((choice == L_TYPE_SOLARIS) &&
1581 (cur_label == L_TYPE_SOLARIS) &&
1582 (cur_disk->fdisk_part.systid != EFI_PMBR)) {
1583 goto expert_end;
1584 } else if ((choice == L_TYPE_EFI) &&
1585 (cur_label == L_TYPE_EFI)) {
1586 goto expert_end;
1588 switch (choice) {
1589 case L_TYPE_SOLARIS:
1591 * EFI label to SMI label
1593 if (cur_dtype->capacity > INFINITY) {
1594 fmt_print("Warning: SMI labels only support up to "
1595 "2 TB.\n");
1598 if (cur_disk->fdisk_part.systid == EFI_PMBR) {
1599 fmt_print("Warning: This disk has an EFI label. "
1600 "Changing to SMI label will erase all\n"
1601 "current partitions.\n");
1602 if (check("Continue"))
1603 return (-1);
1604 #if defined(_FIRMWARE_NEEDS_FDISK)
1605 fmt_print("You must use fdisk to delete the current "
1606 "EFI partition and create a new\n"
1607 "Solaris partition before you can convert the "
1608 "label.\n");
1609 return (-1);
1610 #endif
1613 #if defined(_FIRMWARE_NEEDS_FDISK)
1614 if (!(((cur_disk->fdisk_part.systid != SUNIXOS) ||
1615 (cur_disk->fdisk_part.systid != SUNIXOS2)) &&
1616 (cur_disk->fdisk_part.numsect > 0))) {
1617 fmt_print("You must use fdisk to create a Solaris "
1618 "partition before you can convert the label.\n");
1619 return (-1);
1621 #endif
1623 (void) memset((char *)&label, 0, sizeof (struct dk_label));
1625 (void) strcpy(x86_devname, cur_disk->disk_name);
1626 if (cur_ctype->ctype_ctype == DKC_DIRECT ||
1627 cur_ctype->ctype_ctype == DKC_BLKDEV)
1628 dptr = auto_direct_get_geom_label(cur_file, &label);
1629 else
1630 dptr = auto_sense(cur_file, 1, &label);
1631 if (dptr == NULL) {
1632 fmt_print("Autoconfiguration failed.\n");
1633 return (-1);
1636 pcyl = label.dkl_pcyl;
1637 ncyl = label.dkl_ncyl;
1638 acyl = label.dkl_acyl;
1639 nhead = label.dkl_nhead;
1640 nsect = label.dkl_nsect;
1642 if (delete_disk_type(cur_disk->disk_type) == 0) {
1643 cur_label = L_TYPE_SOLARIS;
1644 cur_disk->label_type = L_TYPE_SOLARIS;
1645 cur_disk->disk_type = dptr;
1646 cur_disk->disk_parts = dptr->dtype_plist;
1647 cur_dtype = dptr;
1648 cur_parts = dptr->dtype_plist;
1650 if (status = write_label())
1651 err_print("Label failed.\n");
1652 else
1653 cur_disk->disk_flags &= ~DSK_LABEL_DIRTY;
1655 return (status);
1656 } else {
1657 err_print("Label failed.\n");
1658 return (-1);
1662 case L_TYPE_EFI:
1664 * SMI label to EFI label
1667 if ((cur_disk->fdisk_part.systid == SUNIXOS) ||
1668 (cur_disk->fdisk_part.systid == SUNIXOS2)) {
1669 fmt_print("Warning: This disk has an SMI label. "
1670 "Changing to EFI label will erase all\ncurrent "
1671 "partitions.\n");
1672 if (check("Continue")) {
1673 return (-1);
1677 if (get_disk_info(cur_file, &efinfo, cur_disk) != 0) {
1678 return (-1);
1680 (void) memset((char *)&label, 0, sizeof (struct dk_label));
1681 label.dkl_pcyl = pcyl;
1682 label.dkl_ncyl = ncyl;
1683 label.dkl_acyl = acyl;
1684 #if defined(_SUNOS_VTOC_16)
1685 label.dkl_bcyl = bcyl;
1686 #endif /* defined(_SUNOC_VTOC_16) */
1687 label.dkl_nhead = nhead;
1688 label.dkl_nsect = nsect;
1689 #if defined(_SUNOS_VTOC_8)
1690 for (i = 0; i < NDKMAP; i++) {
1691 label.dkl_map[i] = cur_parts->pinfo_map[i];
1693 #endif /* defined(_SUNOS_VTOC_8) */
1694 label.dkl_magic = DKL_MAGIC;
1695 label.dkl_vtoc = cur_parts->vtoc;
1696 if (label_to_vtoc(&vtoc, &label) == -1) {
1697 return (-1);
1699 if (SMI_vtoc_to_EFI(cur_file, &vtoc64) == -1) {
1700 return (-1);
1702 if (efi_write(cur_file, vtoc64) != 0) {
1703 err_check(vtoc64);
1704 err_print("Warning: error writing EFI.\n");
1705 return (-1);
1706 } else {
1707 cur_disk->disk_flags &= ~DSK_LABEL_DIRTY;
1710 * copy over the EFI vtoc onto the SMI vtoc and return
1711 * okay.
1713 dptr = auto_efi_sense(cur_file, &efinfo);
1714 if (dptr == NULL) {
1715 fmt_print("Autoconfiguration failed.\n");
1716 return (-1);
1719 cur_label = L_TYPE_EFI;
1720 cur_disk->label_type = L_TYPE_EFI;
1721 cur_disk->disk_type = dptr;
1722 cur_disk->disk_parts = dptr->dtype_plist;
1723 cur_dtype = dptr;
1724 cur_parts = dptr->dtype_plist;
1725 cur_parts->etoc = vtoc64;
1727 ncyl = pcyl = nsect = psect = acyl = phead = 0;
1730 * Get the Solais Fdisk Partition information.
1732 (void) copy_solaris_part(&cur_disk->fdisk_part);
1734 return (0);
1738 expert_end:
1740 * Make sure the user is serious.
1742 if (check("Ready to label disk, continue")) {
1743 return (-1);
1746 * Write the labels out (this will also notify unix) and
1747 * return status.
1749 fmt_print("\n");
1750 if (status = write_label())
1751 err_print("Label failed.\n");
1752 return (status);
1756 * This routine implements the 'analyze' command. It simply runs
1757 * the analyze menu.
1760 c_analyze()
1764 * There must be a current disk type (and therefor a current disk).
1766 if (cur_dtype == NULL) {
1767 err_print("Current Disk Type is not set.\n");
1768 return (-1);
1770 cur_menu++;
1771 last_menu = cur_menu;
1774 * Run the menu.
1776 run_menu(menu_analyze, "ANALYZE", "analyze", 0);
1777 cur_menu--;
1778 return (0);
1782 * This routine implements the 'defect' command. It simply runs
1783 * the defect menu.
1786 c_defect()
1788 int i;
1791 * There must be a current disk type (and therefor a current disk).
1793 if (cur_dtype == NULL) {
1794 err_print("Current Disk Type is not set.\n");
1795 return (-1);
1799 * Check for the defect management and list management ops and
1800 * display appropriate message.
1802 if ((cur_ops->op_ex_man == NULL) && (cur_ops->op_ex_cur == NULL) &&
1803 (cur_ops->op_create == NULL) && (cur_ops->op_wr_cur == NULL)) {
1804 err_print("Controller does not support defect management\n");
1805 err_print("or disk supports automatic defect management.\n");
1806 return (-1);
1808 cur_menu++;
1809 last_menu = cur_menu;
1812 * Lock out interrupt while we manipulate the defect lists.
1814 enter_critical();
1816 * If the working list is null but there is a current list,
1817 * update the working list to be a copy of the current list.
1819 if ((work_list.list == NULL) && (cur_list.list != NULL)) {
1820 work_list.header = cur_list.header;
1821 work_list.list = (struct defect_entry *)zalloc(
1822 deflist_size(cur_blksz, work_list.header.count) *
1823 cur_blksz);
1824 for (i = 0; i < work_list.header.count; i++)
1825 *(work_list.list + i) = *(cur_list.list + i);
1826 work_list.flags = cur_list.flags & LIST_PGLIST;
1828 exit_critical();
1830 * Run the menu.
1832 run_menu(menu_defect, "DEFECT", "defect", 0);
1833 cur_menu--;
1836 * If the user has modified the working list but not committed
1837 * it, warn them that they are probably making a mistake.
1839 if (work_list.flags & LIST_DIRTY) {
1840 if (!EMBEDDED_SCSI) {
1841 err_print(
1842 "Warning: working defect list modified; but not committed.\n");
1843 if (!check(
1844 "Do you wish to commit changes to current defect list"))
1845 (void) do_commit();
1848 return (0);
1852 * This routine implements the 'backup' command. It allows the user
1853 * to search for backup labels on the current disk. This is useful
1854 * if the primary label was lost and the user wishes to recover the
1855 * partition information for the disk. The disk is relabeled and
1856 * the current defect list is written out if a backup label is found.
1859 c_backup()
1861 struct dk_label label;
1862 struct disk_type *dtype;
1863 struct partition_info *parts, *plist;
1864 diskaddr_t bn;
1865 int sec, head, i;
1866 char *buf;
1869 * There must be a current disk type (and therefore a current disk).
1871 if (cur_dtype == NULL) {
1872 err_print("Current Disk Type is not set.\n");
1873 return (-1);
1876 * The disk must be formatted to read backup labels.
1878 if (!(cur_flags & DISK_FORMATTED)) {
1879 err_print("Current Disk is unformatted.\n");
1880 return (-1);
1883 * Check for a valid fdisk table entry for Solaris
1885 if (!good_fdisk()) {
1886 return (-1);
1889 * If we found a primary label on this disk, make sure
1890 * the user is serious.
1892 if (cur_disk->label_type == L_TYPE_EFI) {
1893 if (((cur_disk->disk_parts->etoc->efi_flags &
1894 EFI_GPT_PRIMARY_CORRUPT) == 0) &&
1895 check("Disk has a primary label, still continue"))
1896 return (-1);
1897 fmt_print("Restoring primary label.\n");
1898 if (write_label()) {
1899 err_print("Failed\n");
1900 return (-1);
1902 return (0);
1903 } else if (((cur_disk->disk_flags & (DSK_LABEL | DSK_LABEL_DIRTY)) ==
1904 DSK_LABEL) &&
1905 (check("Disk has a primary label, still continue"))) {
1906 return (-1);
1909 buf = zalloc(cur_blksz);
1910 fmt_print("Searching for backup labels...");
1911 (void) fflush(stdout);
1914 * Some disks have the backup labels in a strange place.
1916 if (cur_ctype->ctype_flags & CF_BLABEL)
1917 head = 2;
1918 else
1919 head = nhead - 1;
1921 * Loop through each copy of the backup label.
1923 for (sec = 1; ((sec < BAD_LISTCNT * 2 + 1) && (sec < nsect));
1924 sec += 2) {
1925 bn = chs2bn(ncyl + acyl - 1, head, sec) + solaris_offset;
1927 * Attempt to read it.
1929 if ((*cur_ops->op_rdwr)(DIR_READ, cur_file, bn,
1930 1, buf, F_NORMAL, NULL)) {
1931 continue;
1934 (void *) memcpy((char *)&label, buf, sizeof (struct dk_label));
1937 * Verify that it is a reasonable label.
1939 if (!checklabel(&label))
1940 continue;
1941 if (trim_id(label.dkl_asciilabel))
1942 continue;
1944 * Lock out interrupts while we manipulate lists.
1946 enter_critical();
1947 fmt_print("found.\n");
1949 * Find out which disk type the backup label claims.
1951 for (dtype = cur_ctype->ctype_dlist; dtype != NULL;
1952 dtype = dtype->dtype_next)
1953 if (dtype_match(&label, dtype))
1954 break;
1956 * If it disagrees with our current type, something
1957 * real bad is happening.
1959 if (dtype != cur_dtype) {
1960 if (dtype == NULL) {
1961 fmt_print("\
1962 Unknown disk type in backup label\n");
1963 exit_critical();
1964 free(buf);
1965 return (-1);
1967 fmt_print("Backup label claims different type:\n");
1968 fmt_print(" <%s cyl %d alt %d hd %d sec %d>\n",
1969 label.dkl_asciilabel, label.dkl_ncyl,
1970 label.dkl_acyl, label.dkl_nhead,
1971 label.dkl_nsect);
1972 if (check("Continue")) {
1973 exit_critical();
1974 free(buf);
1975 return (-1);
1977 cur_dtype = dtype;
1980 * Try to match the partition map with a known map.
1982 for (parts = dtype->dtype_plist; parts != NULL;
1983 parts = parts->pinfo_next)
1984 if (parts_match(&label, parts))
1985 break;
1987 * If we couldn't match it, allocate space for a new one,
1988 * fill in the info, and add it to the list. The name
1989 * for the new map is derived from the disk name.
1991 if (parts == NULL) {
1992 parts = (struct partition_info *)
1993 zalloc(sizeof (struct partition_info));
1994 plist = dtype->dtype_plist;
1995 if (plist == NULL)
1996 dtype->dtype_plist = parts;
1997 else {
1998 while (plist->pinfo_next != NULL)
1999 plist = plist->pinfo_next;
2000 plist->pinfo_next = parts;
2002 parts->pinfo_name = alloc_string("original");
2003 for (i = 0; i < NDKMAP; i++) {
2005 #if defined(_SUNOS_VTOC_8)
2006 parts->pinfo_map[i] = label.dkl_map[i];
2008 #elif defined(_SUNOS_VTOC_16)
2009 parts->pinfo_map[i].dkl_cylno =
2010 label.dkl_vtoc.v_part[i].p_start / spc();
2011 parts->pinfo_map[i].dkl_nblk =
2012 label.dkl_vtoc.v_part[i].p_size;
2013 #else
2014 #error No VTOC layout defined.
2015 #endif /* defined(_SUNOS_VTOC_8) */
2017 parts->vtoc = label.dkl_vtoc;
2020 * We now have a partition map. Make it the current map.
2022 cur_disk->disk_parts = cur_parts = parts;
2023 exit_critical();
2025 * Rewrite the labels and defect lists, as appropriate.
2027 if (EMBEDDED_SCSI) {
2028 fmt_print("Restoring primary label.\n");
2029 if (write_label()) {
2030 free(buf);
2031 return (-1);
2033 } else {
2034 fmt_print("Restoring primary label and defect list.\n");
2035 if (write_label()) {
2036 free(buf);
2037 return (-1);
2039 if (cur_list.list != NULL)
2040 write_deflist(&cur_list);
2042 fmt_print("\n");
2043 free(buf);
2044 return (0);
2047 * If we didn't find any backup labels, say so.
2049 fmt_print("not found.\n\n");
2050 free(buf);
2051 return (0);
2055 * This routine is called by c_verify() for an EFI labeled disk
2057 static int
2058 c_verify_efi()
2060 struct efi_info efi_info;
2061 struct partition_info tmp_pinfo;
2062 int status;
2064 status = read_efi_label(cur_file, &efi_info, cur_disk);
2065 if (status != 0) {
2066 err_print("Warning: Could not read label.\n");
2067 return (-1);
2069 if (cur_parts->etoc->efi_flags & EFI_GPT_PRIMARY_CORRUPT) {
2070 err_print("Reading the primary EFI GPT label ");
2071 err_print("failed. Using backup label.\n");
2072 err_print("Use the 'backup' command to restore ");
2073 err_print("the primary label.\n");
2075 tmp_pinfo.etoc = efi_info.e_parts;
2076 fmt_print("\n");
2077 if (cur_parts->etoc->efi_parts[8].p_name) {
2078 fmt_print("Volume name = <%8s>\n",
2079 cur_parts->etoc->efi_parts[8].p_name);
2080 } else {
2081 fmt_print("Volume name = < >\n");
2083 fmt_print("ascii name = ");
2084 print_efi_string(efi_info.vendor, efi_info.product,
2085 efi_info.revision, efi_info.capacity);
2086 fmt_print("\n");
2088 fmt_print("bytes/sector = %d\n", cur_blksz);
2089 fmt_print("sectors = %llu\n", cur_parts->etoc->efi_last_lba);
2090 fmt_print("accessible sectors = %llu\n",
2091 cur_parts->etoc->efi_last_u_lba);
2093 print_map(&tmp_pinfo);
2095 free(efi_info.vendor);
2096 free(efi_info.product);
2097 free(efi_info.revision);
2098 return (0);
2102 * This routine implements the 'verify' command. It allows the user
2103 * to read the labels on the current disk.
2106 c_verify()
2108 struct dk_label p_label, b_label, *label;
2109 struct partition_info tmp_pinfo;
2110 diskaddr_t bn;
2111 int sec, head, i, status;
2112 int p_label_bad = 0;
2113 int b_label_bad = 0;
2114 int p_label_found = 0;
2115 int b_label_found = 0;
2116 char id_str[128];
2117 char *buf;
2120 * There must be a current disk type (and therefore a current disk).
2122 if (cur_dtype == NULL) {
2123 err_print("Current Disk Type is not set.\n");
2124 return (-1);
2127 * The disk must be formatted to read labels.
2129 if (!(cur_flags & DISK_FORMATTED)) {
2130 err_print("Current Disk is unformatted.\n");
2131 return (-1);
2134 * Check for a valid fdisk table entry for Solaris
2136 if (!good_fdisk()) {
2137 return (-1);
2140 * Branch off here if the disk is EFI labelled.
2142 if (cur_label == L_TYPE_EFI) {
2143 return (c_verify_efi());
2146 * Attempt to read the primary label.
2148 status = read_label(cur_file, &p_label);
2149 if (status == -1) {
2150 err_print("Warning: Could not read primary label.\n");
2151 p_label_bad = 1;
2152 } else {
2154 * Verify that it is a reasonable label.
2157 * Save complete ascii string for printing later.
2159 (void) strncpy(id_str, p_label.dkl_asciilabel, 128);
2161 if ((!checklabel((struct dk_label *)&p_label)) ||
2162 (trim_id(p_label.dkl_asciilabel))) {
2163 err_print("\
2164 Warning: Primary label appears to be corrupt.\n");
2165 p_label_bad = 1;
2166 } else {
2167 p_label_found = 1;
2169 * Make sure it matches current label
2171 if ((!dtype_match(&p_label, cur_dtype)) ||
2172 (!parts_match(&p_label, cur_parts))) {
2173 err_print("\
2174 Warning: Primary label on disk appears to be different from\ncurrent label.\n");
2175 p_label_bad = 1;
2181 * Read backup labels.
2182 * Some disks have the backup labels in a strange place.
2184 if (cur_ctype->ctype_flags & CF_BLABEL)
2185 head = 2;
2186 else
2187 head = nhead - 1;
2189 buf = zalloc(cur_blksz);
2191 * Loop through each copy of the backup label.
2193 for (sec = 1; ((sec < BAD_LISTCNT * 2 + 1) && (sec < nsect));
2194 sec += 2) {
2195 bn = chs2bn(ncyl + acyl - 1, head, sec) + solaris_offset;
2197 * Attempt to read it.
2199 if ((*cur_ops->op_rdwr)(DIR_READ, cur_file, bn,
2200 1, buf, F_NORMAL, NULL))
2201 continue;
2203 (void *) memcpy((char *)&b_label, buf,
2204 sizeof (struct dk_label));
2207 * Verify that it is a reasonable label.
2209 if (!checklabel(&b_label))
2210 continue;
2213 * Save complete label only if no primary label exists
2215 if (!p_label_found)
2216 (void) strncpy(id_str, b_label.dkl_asciilabel, 128);
2218 if (trim_id(b_label.dkl_asciilabel))
2219 continue;
2220 b_label_found = 1;
2222 * Compare against primary label
2224 if (p_label_found) {
2225 if ((strcmp(b_label.dkl_asciilabel,
2226 p_label.dkl_asciilabel) != 0) ||
2227 (b_label.dkl_ncyl != p_label.dkl_ncyl) ||
2228 (b_label.dkl_acyl != p_label.dkl_acyl) ||
2229 (b_label.dkl_nhead != p_label.dkl_nhead) ||
2230 (b_label.dkl_nsect != p_label.dkl_nsect)) {
2231 b_label_bad = 1;
2232 } else {
2233 for (i = 0; i < NDKMAP; i++) {
2234 #if defined(_SUNOS_VTOC_8)
2235 if ((b_label.dkl_map[i].dkl_cylno !=
2236 p_label.dkl_map[i].dkl_cylno) ||
2237 (b_label.dkl_map[i].dkl_nblk !=
2238 p_label.dkl_map[i].dkl_nblk)) {
2239 b_label_bad = 1;
2240 break;
2243 #elif defined(_SUNOS_VTOC_16)
2244 if ((b_label.dkl_vtoc.v_part[i].p_tag !=
2245 p_label.dkl_vtoc.v_part[i].p_tag) ||
2246 (b_label.dkl_vtoc.v_part[i].p_flag
2247 != p_label.dkl_vtoc.v_part[i].
2248 p_flag) ||
2249 (b_label.dkl_vtoc.v_part[i].p_start
2250 != p_label.dkl_vtoc.v_part[i].
2251 p_start) ||
2252 (b_label.dkl_vtoc.v_part[i].p_size
2253 != p_label.dkl_vtoc.v_part[i].
2254 p_size)) {
2255 b_label_bad = 1;
2256 break;
2258 #else
2259 #error No VTOC layout defined.
2260 #endif /* defined(_SUNOS_VTOC_8) */
2264 if (b_label_bad)
2265 err_print(
2266 "Warning: Primary and backup labels do not match.\n");
2267 break;
2270 * If we didn't find any backup labels, say so.
2272 if (!b_label_found)
2273 err_print("Warning: Could not read backup labels.\n");
2275 if ((!b_label_found) || (p_label_bad) || (b_label_bad))
2276 err_print("\n\
2277 Warning: Check the current partitioning and 'label' the disk or use the\n\
2278 \t 'backup' command.\n");
2281 * Print label information.
2283 if (p_label_found) {
2284 fmt_print("\nPrimary label contents:\n");
2285 label = &p_label;
2286 } else if (b_label_found) {
2287 fmt_print("\nBackup label contents:\n");
2288 label = &b_label;
2289 } else {
2290 free(buf);
2291 return (0);
2295 * Must put info into partition_info struct for
2296 * for print routine.
2298 bzero(&tmp_pinfo, sizeof (struct partition_info));
2299 for (i = 0; i < NDKMAP; i++) {
2301 #if defined(_SUNOS_VTOC_8)
2302 tmp_pinfo.pinfo_map[i] = label->dkl_map[i];
2304 #elif defined(_SUNOS_VTOC_16)
2305 tmp_pinfo.pinfo_map[i].dkl_cylno =
2306 label->dkl_vtoc.v_part[i].p_start / spc();
2307 tmp_pinfo.pinfo_map[i].dkl_nblk =
2308 label->dkl_vtoc.v_part[i].p_size;
2309 #else
2310 #error No VTOC layout defined.
2311 #endif /* defined(_SUNOS_VTOC_8) */
2313 tmp_pinfo.vtoc = label->dkl_vtoc;
2315 fmt_print("\n");
2316 fmt_print("Volume name = <%8s>\n", label->dkl_vtoc.v_volume);
2317 fmt_print("ascii name = <%s>\n", id_str);
2318 fmt_print("pcyl = %4d\n", label->dkl_pcyl);
2319 fmt_print("ncyl = %4d\n", label->dkl_ncyl);
2320 fmt_print("acyl = %4d\n", label->dkl_acyl);
2322 #if defined(_SUNOS_VTOC_16)
2323 fmt_print("bcyl = %4d\n", label->dkl_bcyl);
2324 #endif /* defined(_SUNOS_VTOC_16) */
2326 fmt_print("nhead = %4d\n", label->dkl_nhead);
2327 fmt_print("nsect = %4d\n", label->dkl_nsect);
2329 print_map(&tmp_pinfo);
2330 free(buf);
2331 return (0);
2336 * This command implements the inquiry command, for embedded SCSI
2337 * disks only, which issues a SCSI inquiry command, and
2338 * displays the resulting vendor, product id and revision level.
2341 c_inquiry()
2343 char inqbuf[255];
2344 struct scsi_inquiry *inq;
2346 assert(SCSI);
2348 inq = (struct scsi_inquiry *)inqbuf;
2350 if (uscsi_inquiry(cur_file, inqbuf, sizeof (inqbuf))) {
2351 err_print("Failed\n");
2352 return (-1);
2353 } else {
2354 fmt_print("Vendor: ");
2355 print_buf(inq->inq_vid, sizeof (inq->inq_vid));
2356 fmt_print("\nProduct: ");
2357 print_buf(inq->inq_pid, sizeof (inq->inq_pid));
2358 fmt_print("\nRevision: ");
2359 print_buf(inq->inq_revision, sizeof (inq->inq_revision));
2360 fmt_print("\n");
2363 return (0);
2368 * This routine allows the user to set the 8-character
2369 * volume name in the vtoc. It then writes both the
2370 * primary and backup labels onto the current disk.
2373 c_volname()
2375 int status;
2376 char *prompt;
2377 union {
2378 int xfoo;
2379 char defvolname[LEN_DKL_VVOL+1];
2380 } x;
2381 char s1[MAXPATHLEN], nclean[MAXPATHLEN];
2382 char *volname;
2386 * There must be a current disk type (and therefore a current disk).
2388 if (cur_dtype == NULL) {
2389 err_print("Current Disk Type is not set.\n");
2390 return (-1);
2393 * The current disk must be formatted to label it.
2395 if (!(cur_flags & DISK_FORMATTED)) {
2396 err_print("Current Disk is unformatted.\n");
2397 return (-1);
2400 * Check for a valid fdisk table entry for Solaris
2402 if (!good_fdisk()) {
2403 return (-1);
2406 * The current disk must be formatted to label it.
2408 if (cur_parts == NULL) {
2409 err_print(
2410 "Please select a partition map for the disk first.\n");
2411 return (-1);
2415 * Check to see if there are any mounted file systems anywhere
2416 * on the current disk. If so, refuse to label the disk, but
2417 * only if the partitions would change for the mounted partitions.
2420 if (checkmount((diskaddr_t)-1, (diskaddr_t)-1)) {
2421 /* Bleagh, too descriptive */
2422 if (check_label_with_mount()) {
2423 err_print(
2424 "Cannot label disk while it has mounted partitions.\n\n");
2425 return (-1);
2430 * Check to see if there are partitions being used for swapping
2431 * on the current disk. If so, refuse to label the disk, but
2432 * only if the partitions would change for the swap partitions.
2435 if (checkswap((diskaddr_t)-1, (diskaddr_t)-1)) {
2436 /* Bleagh, too descriptive */
2437 if (check_label_with_swap()) {
2438 err_print(
2439 "Cannot label disk while its partitions are currently \
2440 being used for swapping.\n\n");
2441 return (-1);
2446 * Check to see if any partitions used for svm, vxvm, ZFS zpool
2447 * or live upgrade are on the disk. If so, refuse to label the
2448 * disk, but only if we are trying to shrink a partition in
2449 * use.
2451 if (checkdevinuse(cur_disk->disk_name, (diskaddr_t)-1,
2452 (diskaddr_t)-1, 0, 1)) {
2453 err_print("Cannot label disk while its partitions "
2454 "are in use as described.\n");
2455 return (-1);
2459 * Prompt for the disk volume name.
2461 prompt = "Enter 8-character volume name (remember quotes)";
2462 bzero(x.defvolname, LEN_DKL_VVOL+1);
2463 bcopy(cur_disk->v_volume, x.defvolname, LEN_DKL_VVOL);
2465 * Get the input using "get_inputline" since
2466 * input would never return null string.
2468 fmt_print("%s[\"%s\"]:", prompt, x.defvolname);
2471 * Get input from the user.
2473 get_inputline(nclean, MAXPATHLEN);
2474 clean_token(s1, nclean);
2476 * check for return.
2478 if (s1[0] == 0) {
2479 volname = x.defvolname;
2480 } else {
2482 * remove the " mark from volname.
2484 if (s1[0] == '"') {
2485 int i = 1;
2486 volname = &s1[1];
2487 while (s1[i] != '"' && s1[i] != '\0')
2488 i++;
2489 s1[i] = '\0';
2490 clean_token(nclean, volname);
2491 volname = nclean;
2492 } else {
2493 (void) sscanf(&s1[0], "%1024s", nclean);
2494 volname = nclean;
2498 * Make sure the user is serious.
2500 if (check("Ready to label disk, continue")) {
2501 fmt_print("\n");
2502 return (-1);
2505 * Use the volume name chosen above
2507 bzero(cur_disk->v_volume, LEN_DKL_VVOL);
2508 bcopy(volname, cur_disk->v_volume, min((int)strlen(volname),
2509 LEN_DKL_VVOL));
2510 if (cur_label == L_TYPE_EFI) {
2511 bzero(cur_parts->etoc->efi_parts[8].p_name, LEN_DKL_VVOL);
2512 bcopy(volname, cur_parts->etoc->efi_parts[8].p_name,
2513 LEN_DKL_VVOL);
2516 * Write the labels out (this will also notify unix) and
2517 * return status.
2519 fmt_print("\n");
2520 if (status = write_label())
2521 err_print("Label failed.\n");
2522 return (status);