xref: /titanic_44/usr/src/cmd/format/menu_command.c (revision f169c0eae91b2ee787cf8d6dcf8edd9159d4c9e2)
1 /*
2  * CDDL HEADER START
3  *
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.
7  *
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.
12  *
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]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright (c) 1993, 2010, Oracle and/or its affiliates. All rights reserved.
23  */
24 
25 /*
26  * This file contains functions that implement the command menu commands.
27  */
28 
29 #include "global.h"
30 #include <time.h>
31 #include <sys/time.h>
32 #include <sys/resource.h>
33 #include <sys/wait.h>
34 #include <strings.h>
35 #include <signal.h>
36 #include <stdlib.h>
37 #include <string.h>
38 
39 #if defined(sparc)
40 #include <sys/hdio.h>
41 #endif /* defined(sparc) */
42 
43 #include "main.h"
44 #include "analyze.h"
45 #include "menu.h"
46 #include "menu_command.h"
47 #include "menu_defect.h"
48 #include "menu_partition.h"
49 #include "param.h"
50 #include "misc.h"
51 #include "label.h"
52 #include "startup.h"
53 #include "partition.h"
54 #include "prompts.h"
55 #include "checkdev.h"
56 #include "io.h"
57 #include "ctlr_scsi.h"
58 #include "auto_sense.h"
59 #include "modify_partition.h"
60 
61 
62 extern	struct menu_item menu_partition[];
63 extern	struct menu_item menu_analyze[];
64 extern	struct menu_item menu_defect[];
65 
66 /*
67  * Choices for the p_tag vtoc field
68  */
69 slist_t	ptag_choices[] = {
70 	{ "unassigned",	"",	V_UNASSIGNED	},
71 	{ "boot",	"",	V_BOOT		},
72 	{ "root",	"",	V_ROOT		},
73 	{ "swap",	"",	V_SWAP		},
74 	{ "usr",	"",	V_USR		},
75 	{ "backup",	"",	V_BACKUP	},
76 	{ "stand",	"",	V_STAND		},
77 	{ "var",	"",	V_VAR		},
78 	{ "home",	"",	V_HOME		},
79 	{ "alternates",	"",	V_ALTSCTR	},
80 	{ "reserved",	"",	V_RESERVED	},
81 	{ NULL }
82 };
83 
84 
85 /*
86  * Choices for the p_flag vtoc field
87  */
88 slist_t	pflag_choices[] = {
89 	{ "wm",	"read-write, mountable",	0		},
90 	{ "wu",	"read-write, unmountable",	V_UNMNT		},
91 	{ "rm",	"read-only, mountable",		V_RONLY		},
92 	{ "ru",	"read-only, unmountable",	V_RONLY|V_UNMNT	},
93 	{ NULL }
94 };
95 
96 
97 /*
98  * This routine implements the 'disk' command.  It allows the user to
99  * select a disk to be current.  The list of choices is the list of
100  * disks that were found at startup time.
101  */
102 int
103 c_disk()
104 {
105 	struct disk_info	*disk;
106 	u_ioparam_t		ioparam;
107 	int			i;
108 	int			ndisks = 0;
109 	int			blind_select = 0;
110 	int			deflt;
111 	int			index;
112 	int			*defltptr = NULL;
113 	int			more = 0;
114 	int			more_quit = 0;
115 	int			one_line = 0;
116 	int			tty_lines;
117 
118 /*
119  * This buffer holds the check() prompt that verifies we've got the right
120  * disk when performing a blind selection.  The size should be sufficient
121  * to hold the prompt string, plus 256 characters for the disk name -
122  * way more than should ever be necessary.  See the #define in misc.h.
123  */
124 	char			chk_buf[BLIND_SELECT_VER_PROMPT];
125 
126 	if (istokenpresent()) {
127 		/*
128 		 * disk number to be selected is already in the
129 		 * input stream .
130 		 */
131 		TOKEN token, cleantoken;
132 
133 		/*
134 		 * Get the disk number the user has given.
135 		 */
136 		i = 0;
137 		for (disk = disk_list; disk != NULL; disk = disk->disk_next) {
138 			i++;
139 		}
140 
141 		ioparam.io_bounds.lower = 0;
142 		ioparam.io_bounds.upper = i - 1;
143 		(void) gettoken(token);
144 		clean_token(cleantoken, token);
145 
146 		/*
147 		 * Convert the token into an integer.
148 		 */
149 		if (geti(cleantoken, &index, (int *)NULL))
150 			return (0);
151 
152 		/*
153 		 * Check to be sure it is within the legal bounds.
154 		 */
155 		if ((index < 0) || (index >= i)) {
156 			err_print("`%d' is out of range.\n", index);
157 			return (0);
158 		}
159 		goto checkdisk;
160 	}
161 
162 	fmt_print("\n\nAVAILABLE DISK SELECTIONS:\n");
163 
164 	i = 0;
165 	if ((option_f == (char *)NULL) && isatty(0) == 1 && isatty(1) == 1) {
166 		/*
167 		 * We have a real terminal for std input and output, enable
168 		 * more style of output for disk selection list.
169 		 */
170 		more = 1;
171 		tty_lines = get_tty_lines();
172 		enter_critical();
173 		echo_off();
174 		charmode_on();
175 		exit_critical();
176 	}
177 
178 	/*
179 	 * Loop through the list of found disks.
180 	 */
181 	for (disk = disk_list; disk != NULL; disk = disk->disk_next) {
182 		/*
183 		 * If using more output, account 2 lines for each disk.
184 		 */
185 		if (more && !more_quit && i && (one_line ||
186 		    ((2 * i + 1) % (tty_lines - 2) <= 1))) {
187 			int	c;
188 
189 			/*
190 			 * Get the next character.
191 			 */
192 			fmt_print("- hit space for more or s to select - ");
193 			c = getchar();
194 			fmt_print("\015");
195 			one_line = 0;
196 			/*
197 			 * Handle display one line command
198 			 * (return key)
199 			 */
200 			if (c == '\012') {
201 				one_line++;
202 			}
203 			/* Handle Quit command */
204 			if (c == 'q') {
205 				fmt_print(
206 				"                       \015");
207 				more_quit++;
208 			}
209 			/* Handle ^D command */
210 			if (c == '\004')
211 				fullabort();
212 			/* or get on with the show */
213 			if (c == 's' || c == 'S') {
214 				fmt_print("%80s\n", " ");
215 				break;
216 			}
217 		}
218 		/*
219 		 * If this is the current disk, mark it as
220 		 * the default.
221 		 */
222 		if (cur_disk == disk) {
223 			deflt = i;
224 			defltptr = &deflt;
225 		}
226 		if (!more || !more_quit)
227 			pr_diskline(disk, i);
228 		i++;
229 	}
230 	if (more) {
231 		enter_critical();
232 		charmode_off();
233 		echo_on();
234 		exit_critical();
235 	}
236 
237 	/*
238 	 * Determine total number of disks, and ask the user which disk he
239 	 * would like to make current.
240 	 */
241 
242 	for (disk = disk_list; disk != NULL; disk = disk->disk_next) {
243 		ndisks++;
244 	}
245 
246 	ioparam.io_bounds.lower = 0;
247 	ioparam.io_bounds.upper = ndisks - 1;
248 	index = input(FIO_INT, "Specify disk (enter its number)", ':',
249 	    &ioparam, defltptr, DATA_INPUT);
250 
251 	if (index >= i) {
252 		blind_select = 1;
253 	}
254 
255 	/*
256 	 * Find the disk chosen.  Search through controllers/disks
257 	 * in the same original order, so we match what the user
258 	 * chose.
259 	 */
260 checkdisk:
261 	i = 0;
262 	for (disk = disk_list; disk != NULL; disk = disk->disk_next) {
263 		if (i == index)
264 			goto found;
265 		i++;
266 	}
267 	/*
268 	 * Should never happen.
269 	 */
270 	impossible("no disk found");
271 
272 found:
273 	if (blind_select) {
274 		(void) snprintf(chk_buf, sizeof (chk_buf),
275 "Disk %s selected - is this the desired disk? ", disk->disk_name);
276 		if (check(chk_buf)) {
277 			return (-1);
278 		}
279 	}
280 
281 	/*
282 	 * Update the state.  We lock out interrupts so the state can't
283 	 * get half-updated.
284 	 */
285 
286 	enter_critical();
287 	init_globals(disk);
288 	exit_critical();
289 
290 	/*
291 	 * If type unknown and interactive, ask user to specify type.
292 	 * Also, set partition table (best guess) too.
293 	 */
294 	if (!option_f && ncyl == 0 && nhead == 0 && nsect == 0 &&
295 	    (disk->label_type != L_TYPE_EFI)) {
296 		(void) c_type();
297 	}
298 
299 	/*
300 	 * Get the Solaris Fdisk Partition information
301 	 */
302 	if (nhead != 0 && nsect != 0)
303 		(void) copy_solaris_part(&cur_disk->fdisk_part);
304 
305 	if ((cur_disk->label_type == L_TYPE_EFI) &&
306 	    (cur_disk->disk_parts->etoc->efi_flags &
307 	    EFI_GPT_PRIMARY_CORRUPT)) {
308 		err_print("Reading the primary EFI GPT label ");
309 		err_print("failed.  Using backup label.\n");
310 		err_print("Use the 'backup' command to restore ");
311 		err_print("the primary label.\n");
312 	}
313 
314 #if defined(_SUNOS_VTOC_16)
315 	/*
316 	 * If there is no fdisk solaris partition.
317 	 */
318 	if (cur_disk->fdisk_part.numsect == 0) {
319 		err_print("No Solaris fdisk partition found.\n");
320 		goto exit;
321 	}
322 #endif /* defined(_SUNOS_VTOC_16) */
323 
324 	/*
325 	 * If the label of the disk is marked dirty,
326 	 * see if they'd like to label the disk now.
327 	 */
328 	if (cur_disk->disk_flags & DSK_LABEL_DIRTY) {
329 		if (check("Disk not labeled.  Label it now") == 0) {
330 			if (write_label()) {
331 				err_print("Write label failed\n");
332 			} else {
333 				cur_disk->disk_flags &= ~DSK_LABEL_DIRTY;
334 			}
335 		}
336 	}
337 exit:
338 	return (0);
339 }
340 
341 /*
342  * This routine implements the 'type' command.  It allows the user to
343  * specify the type of the current disk.  It should be necessary only
344  * if the disk was not labelled or was somehow labelled incorrectly.
345  * The list of legal types for the disk comes from information that was
346  * in the data file.
347  */
348 int
349 c_type()
350 {
351 	struct disk_type	*type, *tptr, *oldtype;
352 	u_ioparam_t		ioparam;
353 	int			i, index, deflt, *defltptr = NULL;
354 	struct disk_type	disk_type;
355 	struct disk_type	*d = &disk_type;
356 	int			first_disk;
357 	int			auto_conf_choice;
358 	int			other_choice;
359 	struct dk_label		label;
360 	struct efi_info		efi_info;
361 	uint64_t		maxLBA;
362 	char			volname[LEN_DKL_VVOL];
363 	int			volinit = 0;
364 
365 	/*
366 	 * There must be a current disk.
367 	 */
368 	if (cur_disk == NULL) {
369 		err_print("Current Disk is not set.\n");
370 		return (-1);
371 	}
372 	oldtype = cur_disk->disk_type;
373 	type = cur_ctype->ctype_dlist;
374 	/*
375 	 * Print out the list of choices.
376 	 */
377 	fmt_print("\n\nAVAILABLE DRIVE TYPES:\n");
378 	first_disk = 0;
379 	if (cur_ctype->ctype_ctype == DKC_SCSI_CCS) {
380 		auto_conf_choice = 0;
381 		fmt_print("        %d. Auto configure\n", first_disk++);
382 	} else {
383 		auto_conf_choice = -1;
384 	}
385 
386 	i = first_disk;
387 	for (tptr = type; tptr != NULL; tptr = tptr->dtype_next) {
388 		/*
389 		 * If we pass the current type, mark it to be the default.
390 		 */
391 		if (cur_dtype == tptr) {
392 			deflt = i;
393 			defltptr = &deflt;
394 		}
395 		if (cur_disk->label_type == L_TYPE_EFI) {
396 			continue;
397 		}
398 		if (tptr->dtype_asciilabel)
399 			fmt_print("        %d. %s\n", i++,
400 			    tptr->dtype_asciilabel);
401 	}
402 	other_choice = i;
403 	fmt_print("        %d. other\n", i);
404 	ioparam.io_bounds.lower = 0;
405 	ioparam.io_bounds.upper = i;
406 	/*
407 	 * Ask the user which type the disk is.
408 	 */
409 	index = input(FIO_INT, "Specify disk type (enter its number)", ':',
410 	    &ioparam, defltptr, DATA_INPUT);
411 	/*
412 	 * Find the type s/he chose.
413 	 */
414 	if (index == auto_conf_choice) {
415 		float			scaled;
416 		diskaddr_t		nblks;
417 		int			nparts;
418 
419 		/*
420 		 * User chose "auto configure".
421 		 */
422 		(void) strcpy(x86_devname, cur_disk->disk_name);
423 		switch (cur_disk->label_type) {
424 		case L_TYPE_SOLARIS:
425 			if ((tptr = auto_sense(cur_file, 1, &label)) == NULL) {
426 				err_print("Auto configure failed\n");
427 				return (-1);
428 			}
429 			fmt_print("%s: configured with capacity of ",
430 			    cur_disk->disk_name);
431 			nblks = (diskaddr_t)tptr->dtype_ncyl *
432 			    tptr->dtype_nhead * tptr->dtype_nsect;
433 			scaled = bn2mb(nblks);
434 			if (scaled > 1024.0) {
435 				fmt_print("%1.2fGB\n", scaled/1024.0);
436 			} else {
437 				fmt_print("%1.2fMB\n", scaled);
438 			}
439 			fmt_print("<%s cyl %d alt %d hd %d sec %d>\n",
440 			    tptr->dtype_asciilabel, tptr->dtype_ncyl,
441 			    tptr->dtype_acyl, tptr->dtype_nhead,
442 			    tptr->dtype_nsect);
443 			break;
444 		case L_TYPE_EFI:
445 			if ((tptr = auto_efi_sense(cur_file, &efi_info))
446 			    == NULL) {
447 				err_print("Auto configure failed\n");
448 				return (-1);
449 			}
450 			fmt_print("%s: configured with capacity of ",
451 			    cur_disk->disk_name);
452 			scaled = bn2mb(efi_info.capacity);
453 			if (scaled > 1024.0) {
454 				fmt_print("%1.2fGB\n", scaled/1024.0);
455 			} else {
456 				fmt_print("%1.2fMB\n", scaled);
457 			}
458 			cur_blksz = efi_info.e_parts->efi_lbasize;
459 			print_efi_string(efi_info.vendor, efi_info.product,
460 			    efi_info.revision, efi_info.capacity);
461 			fmt_print("\n");
462 			for (nparts = 0; nparts < cur_parts->etoc->efi_nparts;
463 			    nparts++) {
464 				if (cur_parts->etoc->efi_parts[nparts].p_tag ==
465 				    V_RESERVED) {
466 					if (cur_parts->etoc->efi_parts[nparts].
467 					    p_name) {
468 						(void) strcpy(volname,
469 						    cur_parts->etoc->efi_parts
470 						    [nparts].p_name);
471 						volinit = 1;
472 					}
473 					break;
474 				}
475 			}
476 			enter_critical();
477 			if (delete_disk_type(cur_disk->disk_type) != 0) {
478 				fmt_print("Autoconfiguration failed.\n");
479 				return (-1);
480 			}
481 			cur_disk->disk_type = tptr;
482 			cur_disk->disk_parts = tptr->dtype_plist;
483 			init_globals(cur_disk);
484 			exit_critical();
485 			if (volinit) {
486 				for (nparts = 0; nparts <
487 				    cur_parts->etoc->efi_nparts; nparts++) {
488 				if (cur_parts->etoc->efi_parts[nparts].p_tag ==
489 				    V_RESERVED) {
490 				(void) strcpy(
491 				    cur_parts->etoc->efi_parts[nparts].p_name,
492 				    volname);
493 				(void) strlcpy(cur_disk->v_volume, volname,
494 				    LEN_DKL_VVOL);
495 				break;
496 				}
497 				}
498 			}
499 			return (0);
500 			break;
501 		default:
502 		/* Should never happen */
503 		return (-1);
504 		}
505 	} else if ((index == other_choice) && (cur_label == L_TYPE_SOLARIS)) {
506 		/*
507 		 * User chose "other".
508 		 * Get the standard information on the new type.
509 		 * Put all information in a tmp structure, in
510 		 * case user aborts.
511 		 */
512 		bzero((char *)d, sizeof (struct disk_type));
513 
514 		d->dtype_ncyl = get_ncyl();
515 		d->dtype_acyl = get_acyl(d->dtype_ncyl);
516 		d->dtype_pcyl = get_pcyl(d->dtype_ncyl, d->dtype_acyl);
517 		d->dtype_nhead = get_nhead();
518 		d->dtype_phead = get_phead(d->dtype_nhead, &d->dtype_options);
519 		d->dtype_nsect = get_nsect();
520 		d->dtype_psect = get_psect(&d->dtype_options);
521 		d->dtype_bpt = get_bpt(d->dtype_nsect, &d->dtype_options);
522 		d->dtype_rpm = get_rpm();
523 		d->dtype_fmt_time = get_fmt_time(&d->dtype_options);
524 		d->dtype_cyl_skew = get_cyl_skew(&d->dtype_options);
525 		d->dtype_trk_skew = get_trk_skew(&d->dtype_options);
526 		d->dtype_trks_zone = get_trks_zone(&d->dtype_options);
527 		d->dtype_atrks = get_atrks(&d->dtype_options);
528 		d->dtype_asect = get_asect(&d->dtype_options);
529 		d->dtype_cache = get_cache(&d->dtype_options);
530 		d->dtype_threshold = get_threshold(&d->dtype_options);
531 		d->dtype_prefetch_min = get_min_prefetch(&d->dtype_options);
532 		d->dtype_prefetch_max = get_max_prefetch(d->dtype_prefetch_min,
533 		    &d->dtype_options);
534 		d->dtype_bps = get_bps();
535 #if defined(sparc)
536 		d->dtype_dr_type = 0;
537 #endif /* defined(sparc) */
538 
539 		d->dtype_asciilabel = get_asciilabel();
540 		/*
541 		 * Add the new type to the list of possible types for
542 		 * this controller.  We lock out interrupts so the lists
543 		 * can't get munged.  We put off actually allocating the
544 		 * structure till here in case the user wanted to
545 		 * interrupt while still inputting information.
546 		 */
547 		enter_critical();
548 		tptr = (struct disk_type *)zalloc(sizeof (struct disk_type));
549 		if (type == NULL)
550 			cur_ctype->ctype_dlist = tptr;
551 		else {
552 			while (type->dtype_next != NULL)
553 				type = type->dtype_next;
554 			type->dtype_next = tptr;
555 		}
556 		bcopy((char *)d, (char *)tptr, sizeof (disk_type));
557 		tptr->dtype_next = NULL;
558 		/*
559 		 * the new disk type does not have any defined
560 		 * partition table . Hence copy the current partition
561 		 * table if possible else create a default
562 		 * paritition table.
563 		 */
564 		new_partitiontable(tptr, oldtype);
565 	} else if ((index == other_choice) && (cur_label == L_TYPE_EFI)) {
566 		maxLBA = get_mlba();
567 		cur_parts->etoc->efi_last_lba = maxLBA;
568 		cur_parts->etoc->efi_last_u_lba = maxLBA - 34;
569 		for (i = 0; i < cur_parts->etoc->efi_nparts; i++) {
570 			cur_parts->etoc->efi_parts[i].p_start = 0;
571 			cur_parts->etoc->efi_parts[i].p_size = 0;
572 			cur_parts->etoc->efi_parts[i].p_tag = V_UNASSIGNED;
573 		}
574 		cur_parts->etoc->efi_parts[8].p_start =
575 		    maxLBA - 34 - (1024 * 16);
576 		cur_parts->etoc->efi_parts[8].p_size = (1024 * 16);
577 		cur_parts->etoc->efi_parts[8].p_tag = V_RESERVED;
578 		if (write_label()) {
579 			err_print("Write label failed\n");
580 		} else {
581 			cur_disk->disk_flags &= ~DSK_LABEL_DIRTY;
582 		}
583 		return (0);
584 	} else {
585 		/*
586 		 * User picked an existing disk type.
587 		 */
588 		i = first_disk;
589 		tptr = type;
590 		while (i < index) {
591 			if (tptr->dtype_asciilabel) {
592 				i++;
593 			}
594 			tptr = tptr->dtype_next;
595 		}
596 		if ((tptr->dtype_asciilabel == NULL) &&
597 		    (tptr->dtype_next != NULL)) {
598 			while (tptr->dtype_asciilabel == NULL) {
599 				tptr = tptr->dtype_next;
600 			}
601 		}
602 	}
603 	/*
604 	 * Check for mounted file systems in the format zone.
605 	 * One potential problem with this would be that check()
606 	 * always returns 'yes' when running out of a file.  However,
607 	 * it is actually ok because we don't let the program get
608 	 * started if there are mounted file systems and we are
609 	 * running from a file.
610 	 */
611 	if ((tptr != oldtype) &&
612 	    checkmount((diskaddr_t)-1, (diskaddr_t)-1)) {
613 		err_print(
614 		    "Cannot set disk type while it has mounted "
615 		    "partitions.\n\n");
616 		return (-1);
617 	}
618 	/*
619 	 * check for partitions being used for swapping in format zone
620 	 */
621 	if ((tptr != oldtype) &&
622 	    checkswap((diskaddr_t)-1, (diskaddr_t)-1)) {
623 		err_print("Cannot set disk type while its partition are "
624 		"currently being used for swapping.\n");
625 		return (-1);
626 	}
627 
628 	/*
629 	 * Check for partitions being used in SVM, VxVM or LU devices
630 	 */
631 
632 	if ((tptr != oldtype) &&
633 	    checkdevinuse(cur_disk->disk_name, (diskaddr_t)-1,
634 	    (diskaddr_t)-1, 0, 0)) {
635 		err_print("Cannot set disk type while its "
636 		    "partitions are currently in use.\n");
637 		return (-1);
638 	}
639 	/*
640 	 * If the type selected is different from the previous type,
641 	 * mark the disk as not labelled and reload the current
642 	 * partition info.  This is not essential but probably the
643 	 * right thing to do, since the size of the disk has probably
644 	 * changed.
645 	 */
646 	enter_critical();
647 	if (tptr != oldtype) {
648 		cur_disk->disk_type = tptr;
649 		cur_disk->disk_parts = NULL;
650 		cur_disk->disk_flags &= ~DSK_LABEL;
651 	}
652 	/*
653 	 * Initialize the state of the current disk.
654 	 */
655 	init_globals(cur_disk);
656 	(void) get_partition();
657 	exit_critical();
658 
659 	/*
660 	 * If the label of the disk is marked dirty,
661 	 * see if they'd like to label the disk now.
662 	 */
663 	if (cur_disk->disk_flags & DSK_LABEL_DIRTY) {
664 		if (check("Disk not labeled.  Label it now") == 0) {
665 			if (write_label()) {
666 				err_print("Write label failed\n");
667 			} else {
668 				cur_disk->disk_flags &= ~DSK_LABEL_DIRTY;
669 			}
670 		}
671 	}
672 
673 	return (0);
674 }
675 
676 /*
677  * This routine implements the 'partition' command.  It simply runs
678  * the partition menu.
679  */
680 int
681 c_partition()
682 {
683 
684 	/*
685 	 * There must be a current disk type and a current disk
686 	 */
687 	if (cur_dtype == NULL) {
688 		err_print("Current Disk Type is not set.\n");
689 		return (-1);
690 	}
691 	/*
692 	 * Check for a valid fdisk table entry for Solaris
693 	 */
694 	if (!good_fdisk()) {
695 		return (-1);
696 	}
697 
698 	cur_menu++;
699 	last_menu = cur_menu;
700 
701 #ifdef	not
702 	/*
703 	 * If there is no current partition table, make one.  This is
704 	 * so the commands within the menu never have to check for
705 	 * a non-existent table.
706 	 */
707 	if (cur_parts == NULL)
708 		err_print("making partition.\n");
709 		make_partition();
710 #endif	/* not */
711 
712 	/*
713 	 * Run the menu.
714 	 */
715 	run_menu(menu_partition, "PARTITION", "partition", 0);
716 	cur_menu--;
717 	return (0);
718 }
719 
720 /*
721  * This routine implements the 'current' command.  It describes the
722  * current disk.
723  */
724 int
725 c_current()
726 {
727 
728 	/*
729 	 * If there is no current disk, say so.  Note that this is
730 	 * not an error since it is a legitimate response to the inquiry.
731 	 */
732 	if (cur_disk == NULL) {
733 		fmt_print("No Current Disk.\n");
734 		return (0);
735 	}
736 	/*
737 	 * Print out the info we have on the current disk.
738 	 */
739 	fmt_print("Current Disk = %s", cur_disk->disk_name);
740 	if (chk_volname(cur_disk)) {
741 		fmt_print(": ");
742 		print_volname(cur_disk);
743 	}
744 	fmt_print("\n");
745 	if (cur_disk->devfs_name != NULL) {
746 		if (cur_dtype == NULL) {
747 			fmt_print("<type unknown>\n");
748 		} else if (cur_label == L_TYPE_SOLARIS) {
749 			fmt_print("<%s cyl %d alt %d hd %d sec %d>\n",
750 			    cur_dtype->dtype_asciilabel, ncyl,
751 			    acyl, nhead, nsect);
752 		} else if (cur_label == L_TYPE_EFI) {
753 			print_efi_string(cur_dtype->vendor,
754 			    cur_dtype->product, cur_dtype->revision,
755 			    cur_dtype->capacity);
756 			fmt_print("\n");
757 		}
758 		fmt_print("%s\n", cur_disk->devfs_name);
759 	} else {
760 		fmt_print("%s%d: <", cur_ctlr->ctlr_dname,
761 		    cur_disk->disk_dkinfo.dki_unit);
762 		if (cur_dtype == NULL) {
763 			fmt_print("type unknown");
764 		} else if (cur_label == L_TYPE_SOLARIS) {
765 			fmt_print("%s cyl %d alt %d hd %d sec %d",
766 			    cur_dtype->dtype_asciilabel, ncyl,
767 			    acyl, nhead, nsect);
768 		} else if (cur_label == L_TYPE_EFI) {
769 			print_efi_string(cur_dtype->vendor,
770 			    cur_dtype->product, cur_dtype->revision,
771 			    cur_dtype->capacity);
772 			fmt_print("\n");
773 		}
774 		fmt_print(">\n");
775 	}
776 	fmt_print("\n");
777 	return (0);
778 }
779 /*
780  * This routine implements the 'format' command.  It allows the user
781  * to format and verify any portion of the disk.
782  */
783 int
784 c_format()
785 {
786 	diskaddr_t		start, end;
787 	time_t			clock;
788 	int			format_time, format_tracks, format_cyls;
789 	int			format_interval;
790 	diskaddr_t		deflt;
791 	int			status;
792 	u_ioparam_t		ioparam;
793 	struct scsi_inquiry	*inq;
794 	char	rawbuf[MAX_MODE_SENSE_SIZE];
795 	struct scsi_capacity_16	capacity;
796 	struct vpd_hdr	*vpdhdr;
797 	uint8_t	protect;
798 	uint8_t	pagecode;
799 	uint8_t	spt;
800 	uint8_t	p_type;
801 	uint8_t	prot_flag[NUM_PROT_TYPE] = {1, 0, 0, 0};
802 	int	i;
803 	char	*prot_descriptor[NUM_PROT_TYPE] = {
804 	    "Protection Information is disabled.",
805 	    "Protection Information is enabled.",
806 	    "Protection Information is enabled.",
807 	    "Protection Information is enabled.", };
808 
809 	/*
810 	 * There must be a current disk type and a current disk
811 	 */
812 	if (cur_dtype == NULL) {
813 		err_print("Current Disk Type is not set.\n");
814 		return (-1);
815 	}
816 
817 	/*
818 	 * There must be a format routine in cur_ops structure to have
819 	 *  this routine work.
820 	 */
821 	if (cur_ops->op_format == NULL) {
822 		err_print(
823 "Cannot format this drive. Please use your Manufacturer supplied formatting "
824 "utility.\n");
825 		return (-1);
826 	}
827 
828 	/*
829 	 * There must be a current defect list.  Except for
830 	 * unformatted SCSI disks.  For them the defect list
831 	 * can only be retrieved after formatting the disk.
832 	 */
833 	if ((cur_ctype->ctype_flags & CF_SCSI) && !EMBEDDED_SCSI &&
834 	    (cur_ctype->ctype_flags & CF_DEFECTS) &&
835 	    ! (cur_flags & DISK_FORMATTED)) {
836 		cur_list.flags |= LIST_RELOAD;
837 
838 	} else if (cur_list.list == NULL && !EMBEDDED_SCSI) {
839 		err_print("Current Defect List must be initialized.\n");
840 		return (-1);
841 	}
842 	/*
843 	 * Ask for the bounds of the format.  We always use the whole
844 	 * disk as the default, since that is the most likely case.
845 	 * Note, for disks which must be formatted accross the whole disk,
846 	 * don't bother the user.
847 	 */
848 	ioparam.io_bounds.lower = start = 0;
849 	if (cur_label == L_TYPE_SOLARIS) {
850 		if (cur_ctype->ctype_flags & CF_SCSI) {
851 			ioparam.io_bounds.upper = end = datasects() - 1;
852 		} else {
853 			ioparam.io_bounds.upper = end = physsects() - 1;
854 		}
855 	} else {
856 		ioparam.io_bounds.upper = end = cur_parts->etoc->efi_last_lba;
857 	}
858 
859 	if (! (cur_ctlr->ctlr_flags & DKI_FMTVOL)) {
860 		deflt = ioparam.io_bounds.lower;
861 		start = input(FIO_BN,
862 		    "Enter starting block number", ':',
863 		    &ioparam, (int *)&deflt, DATA_INPUT);
864 		ioparam.io_bounds.lower = start;
865 		deflt = ioparam.io_bounds.upper;
866 		end = input(FIO_BN,
867 		    "Enter ending block number", ':',
868 		    &ioparam, (int *)&deflt, DATA_INPUT);
869 	}
870 	/*
871 	 * Some disks can format tracks.  Make sure the whole track is
872 	 * specified for them.
873 	 */
874 	if (cur_ctlr->ctlr_flags & DKI_FMTTRK) {
875 		if (bn2s(start) != 0 ||
876 		    bn2s(end) != sectors(bn2h(end)) - 1) {
877 			err_print("Controller requires formatting of ");
878 			err_print("entire tracks.\n");
879 			return (-1);
880 		}
881 	}
882 	/*
883 	 * Check for mounted file systems in the format zone, and if we
884 	 * find any, make sure they are really serious.  One potential
885 	 * problem with this would be that check() always returns 'yes'
886 	 * when running out of a file.  However, it is actually ok
887 	 * because we don't let the program get started if there are
888 	 * mounted file systems and we are running from a file.
889 	 */
890 	if (checkmount(start, end)) {
891 		err_print(
892 		"Cannot format disk while it has mounted partitions.\n\n");
893 		return (-1);
894 	}
895 	/*
896 	 * check for partitions being used for swapping in format zone
897 	 */
898 	if (checkswap(start, end)) {
899 		err_print("Cannot format disk while its partition are \
900 currently being used for swapping.\n");
901 		return (-1);
902 	}
903 	/*
904 	 * Check for partitions being used in SVM, VxVM or LU devices
905 	 * in this format zone
906 	 */
907 	if (checkdevinuse(cur_disk->disk_name, start, end, 0, 0)) {
908 		err_print("Cannot format disk while its partitions "
909 		    "are currently in use.\n");
910 		return (-1);
911 	}
912 
913 	if (cur_disk->disk_lbasize != DEV_BSIZE) {
914 		fmt_print("Current disk sector size is %d Byte, format\n"
915 		    "will change the sector size to 512 Byte. ",
916 		    cur_disk->disk_lbasize);
917 		if (check("Continue")) {
918 			return (-1);
919 		}
920 	}
921 
922 	/*
923 	 * set the default protection type
924 	 */
925 	prot_type = PROT_TYPE_0;
926 
927 	/*
928 	 * Check if the protect information of this disk is enabled
929 	 */
930 	if (uscsi_inquiry(cur_file, rawbuf, sizeof (rawbuf))) {
931 		err_print("Inquiry failed\n");
932 		return (-1);
933 	}
934 	inq = (struct scsi_inquiry *)rawbuf;
935 	protect = inq->inq_protect;
936 	if (protect == 0) {
937 		fmt_print("The protection information is not enabled\n");
938 		fmt_print(
939 		    "The disk will be formatted with protection type 0\n");
940 	} else {
941 		(void) memset(rawbuf, 0, MAX_MODE_SENSE_SIZE);
942 		if (uscsi_inquiry_page_86h(cur_file, rawbuf, sizeof (rawbuf))) {
943 			err_print("Inquiry with page 86h failed\n");
944 			return (-1);
945 		}
946 		vpdhdr = (struct vpd_hdr *)rawbuf;
947 		pagecode = vpdhdr->page_code;
948 		if (pagecode != 0x86) {
949 			err_print("Inquiry with page 86h failed\n");
950 			return (-1);
951 		}
952 		spt = (rawbuf[4] << 2) >> 5;
953 		fmt_print("This disk can support protection types:\n");
954 
955 		switch (spt) {
956 		case 0:
957 			prot_flag[1] = 1;
958 			break;
959 		case 1:
960 			prot_flag[1] = 1;
961 			prot_flag[2] = 1;
962 			break;
963 		case 2:
964 			prot_flag[2] = 1;
965 			break;
966 		case 3:
967 			prot_flag[1] = 1;
968 			prot_flag[3] = 1;
969 			break;
970 		case 4:
971 			prot_flag[3] = 1;
972 			break;
973 		case 5:
974 			prot_flag[2] = 1;
975 			prot_flag[3] = 1;
976 			break;
977 		case 7:
978 			prot_flag[1] = 1;
979 			prot_flag[2] = 1;
980 			prot_flag[3] = 1;
981 			break;
982 		default:
983 			err_print(
984 			    "Invalid supported protection types\n");
985 			return (-1);
986 		}
987 		for (i = 0; i < NUM_PROT_TYPE; i++) {
988 			if (prot_flag[i] == 1) {
989 				fmt_print("[%d] TYPE_%d : ", i, i);
990 				fmt_print("%s\n", prot_descriptor[i]);
991 			}
992 		}
993 
994 		/*
995 		 * Get the current protection type
996 		 */
997 		if (uscsi_read_capacity_16(cur_file, &capacity)) {
998 			err_print("Read capacity_16 failed\n");
999 			return (-1);
1000 		}
1001 		p_type = get_cur_protection_type(&capacity);
1002 		fmt_print("\nThe disk is currently formatted with TYPE_%d.\n",
1003 		    p_type);
1004 
1005 		/*
1006 		 * Ask user what protection type to use
1007 		 */
1008 		ioparam.io_bounds.lower = PROT_TYPE_0;
1009 		ioparam.io_bounds.upper = PROT_TYPE_3;
1010 		prot_type = input(FIO_INT, "Specify the New Protection Type",
1011 		    ':', &ioparam, NULL, DATA_INPUT);
1012 		/*
1013 		 * if get a unsupported protection type, then use the
1014 		 * current type: p_type.
1015 		 */
1016 		if (prot_flag[prot_type] == 0) {
1017 			fmt_print("Unsupported protection type.\n");
1018 			prot_type = p_type;
1019 		}
1020 		fmt_print("The disk will be formatted to type %d\n", prot_type);
1021 	}
1022 
1023 	if (SCSI && (format_time = scsi_format_time()) > 0) {
1024 		fmt_print(
1025 		    "\nReady to format.  Formatting cannot be interrupted\n"
1026 		    "and takes %d minutes (estimated). ", format_time);
1027 
1028 	} else if (cur_dtype->dtype_options & SUP_FMTTIME) {
1029 		/*
1030 		 * Formatting time is (2 * time of 1 spin * number of
1031 		 * tracks) + (step rate * number of cylinders) rounded
1032 		 * up to the nearest minute.  Note, a 10% fudge factor
1033 		 * is thrown in for insurance.
1034 		 */
1035 		if (cur_dtype->dtype_fmt_time == 0)
1036 			cur_dtype->dtype_fmt_time = 2;
1037 
1038 		format_tracks = ((end-start) / cur_dtype->dtype_nsect) + 1;
1039 		format_cyls = format_tracks / cur_dtype->dtype_nhead;
1040 		format_tracks = format_tracks * cur_dtype->dtype_fmt_time;
1041 
1042 		/*
1043 		 * ms.
1044 		 */
1045 		format_time = ((60000 / cur_dtype->dtype_rpm) +1) *
1046 		    format_tracks + format_cyls * 7;
1047 		/*
1048 		 * 20% done tick (sec)
1049 		 */
1050 		format_interval = format_time / 5000;
1051 		/*
1052 		 * min.
1053 		 */
1054 		format_time = (format_time + 59999) / 60000;
1055 
1056 		/*
1057 		 * Check format time values and make adjustments
1058 		 * to prevent sleeping too long (forever?) or
1059 		 * too short.
1060 		 */
1061 		if (format_time <= 1) {
1062 			/*
1063 			 * Format time is less than 1 min..
1064 			 */
1065 			format_time = 1;
1066 		}
1067 
1068 		if (format_interval < 11) {
1069 			/* Format time is less than 1 minute. */
1070 			if (format_interval < 2)
1071 				format_interval = 2;	/* failsafe */
1072 			format_interval = 10;
1073 		} else {
1074 			/* Format time is greater than 1 minute. */
1075 			format_interval -= 10;
1076 		}
1077 
1078 		fmt_print(
1079 		    "Ready to format.  Formatting cannot be interrupted\n"
1080 		    "and takes %d minutes (estimated). ", format_time);
1081 	} else {
1082 		fmt_print(
1083 		    "Ready to format.  Formatting cannot be interrupted.\n");
1084 	}
1085 	if (check("Continue")) {
1086 		return (-1);
1087 	}
1088 
1089 	/*
1090 	 * Print the time so that the user will know when format started.
1091 	 * Lock out interrupts.  This could be a problem, since it could
1092 	 * cause the user to sit for quite awhile with no control, but we
1093 	 * don't have any other good way of keeping his gun from going off.
1094 	 */
1095 	clock = time((time_t *)0);
1096 	fmt_print("Beginning format. The current time is %s\n",
1097 	    ctime(&clock));
1098 	enter_critical();
1099 	/*
1100 	 * Mark the defect list dirty so it will be rewritten when we are
1101 	 * done.  It is possible to qualify this so it doesn't always
1102 	 * get rewritten, but it's not worth the trouble.
1103 	 * Note: no defect lists for embedded scsi drives.
1104 	 */
1105 	if (!EMBEDDED_SCSI) {
1106 		cur_list.flags |= LIST_DIRTY;
1107 	}
1108 	/*
1109 	 * If we are formatting over any of the labels, mark the label
1110 	 * dirty so it will be rewritten.
1111 	 */
1112 	if (cur_disk->label_type == L_TYPE_SOLARIS) {
1113 		if (start < totalsects() && end >= datasects()) {
1114 			if (cur_disk->disk_flags & DSK_LABEL)
1115 				cur_flags |= LABEL_DIRTY;
1116 		}
1117 	} else if (cur_disk->label_type == L_TYPE_EFI) {
1118 		if (start < 34) {
1119 			if (cur_disk->disk_flags & DSK_LABEL)
1120 				cur_flags |= LABEL_DIRTY;
1121 		}
1122 	}
1123 	if (start == 0) {
1124 		cur_flags |= LABEL_DIRTY;
1125 	}
1126 	/*
1127 	 * Do the format. bugid 1009138 removed the use of fork to
1128 	 * background the format and print a tick.
1129 	 */
1130 
1131 	status = (*cur_ops->op_format)(start, end, &cur_list);
1132 	if (status) {
1133 		exit_critical();
1134 		err_print("failed\n");
1135 		return (-1);
1136 	}
1137 	fmt_print("done\n");
1138 	if (option_msg && diag_msg) {
1139 		clock = time((time_t *)0);
1140 		fmt_print("The current time is %s\n", ctime(&clock));
1141 	}
1142 	cur_flags |= DISK_FORMATTED;
1143 	/*
1144 	 * If the defect list or label is dirty, write them out again.
1145 	 * Note, for SCSI we have to wait til now to load defect list
1146 	 * since we can't access it until after formatting a virgin disk.
1147 	 */
1148 	/* enter_critical(); */
1149 	if (cur_list.flags & LIST_RELOAD) {
1150 		assert(!EMBEDDED_SCSI);
1151 		if (*cur_ops->op_ex_man == NULL ||
1152 		    (*cur_ops->op_ex_man)(&cur_list)) {
1153 			err_print("Warning: unable to reload defect list\n");
1154 			cur_list.flags &= ~LIST_DIRTY;
1155 			return (-1);
1156 		}
1157 		cur_list.flags |= LIST_DIRTY;
1158 	}
1159 
1160 	if (cur_list.flags & LIST_DIRTY) {
1161 		assert(!EMBEDDED_SCSI);
1162 		write_deflist(&cur_list);
1163 		cur_list.flags = 0;
1164 	}
1165 	if (cur_flags & LABEL_DIRTY) {
1166 		(void) write_label();
1167 		cur_flags &= ~LABEL_DIRTY;
1168 	}
1169 	/*
1170 	 * Come up for air, since the verify step does not need to
1171 	 * be atomic (it does it's own lockouts when necessary).
1172 	 */
1173 	exit_critical();
1174 	/*
1175 	 * If we are supposed to verify, we do the 'write' test over
1176 	 * the format zone.  The rest of the analysis parameters are
1177 	 * left the way they were.
1178 	 */
1179 	if (scan_auto) {
1180 		scan_entire = 0;
1181 		scan_lower = start;
1182 		scan_upper = end;
1183 		fmt_print("\nVerifying media...");
1184 		status = do_scan(SCAN_PATTERN, F_SILENT);
1185 	}
1186 	/*
1187 	 * If the defect list or label is dirty, write them out again.
1188 	 */
1189 	if (cur_list.flags & LIST_DIRTY) {
1190 		assert(!EMBEDDED_SCSI);
1191 		cur_list.flags = 0;
1192 		write_deflist(&cur_list);
1193 	}
1194 	if (cur_flags & LABEL_DIRTY) {
1195 		cur_flags &= ~LABEL_DIRTY;
1196 		(void) write_label();
1197 	}
1198 	return (status);
1199 }
1200 
1201 /*
1202  * This routine implements the 'repair' command.  It allows the user
1203  * to reallocate sectors on the disk that have gone bad.
1204  */
1205 int
1206 c_repair()
1207 {
1208 	diskaddr_t	bn;
1209 	int		status;
1210 	u_ioparam_t	ioparam;
1211 	char		*buf;
1212 	int		buf_is_good;
1213 	int		block_has_error;
1214 	int		i;
1215 
1216 	/*
1217 	 * There must be a current disk type (and therefore a current disk).
1218 	 */
1219 	if (cur_dtype == NULL) {
1220 		err_print("Current Disk Type is not set.\n");
1221 		return (-1);
1222 	}
1223 	/*
1224 	 * The current disk must be formatted for repair to work.
1225 	 */
1226 	if (!(cur_flags & DISK_FORMATTED)) {
1227 		err_print("Current Disk is unformatted.\n");
1228 		return (-1);
1229 	}
1230 	/*
1231 	 * Check for a valid fdisk table entry for Solaris
1232 	 */
1233 	if (!good_fdisk()) {
1234 		return (-1);
1235 	}
1236 	/*
1237 	 * Repair is an optional command for controllers, so it may
1238 	 * not be supported.
1239 	 */
1240 	if (cur_ops->op_repair == NULL) {
1241 		err_print("Controller does not support repairing.\n");
1242 		err_print("or disk supports automatic defect management.\n");
1243 		return (-1);
1244 	}
1245 	/*
1246 	 * There must be a defect list for non-embedded scsi devices,
1247 	 * since we will add to it.
1248 	 */
1249 	if (!EMBEDDED_SCSI && cur_list.list == NULL) {
1250 		err_print("Current Defect List must be initialized.\n");
1251 		return (-1);
1252 	}
1253 	/*
1254 	 * Ask the user which sector has gone bad.
1255 	 */
1256 	ioparam.io_bounds.lower = 0;
1257 	if (cur_disk->label_type == L_TYPE_SOLARIS) {
1258 		ioparam.io_bounds.upper = physsects() - 1;
1259 	} else {
1260 		ioparam.io_bounds.upper = cur_parts->etoc->efi_last_lba;
1261 	}
1262 	bn = input(FIO_BN,
1263 	    "Enter absolute block number of defect", ':',
1264 	    &ioparam, (int *)NULL, DATA_INPUT);
1265 	/*
1266 	 * Check to see if there is a mounted file system over the
1267 	 * specified sector.  If there is, make sure the user is
1268 	 * really serious.
1269 	 */
1270 	if (checkmount(bn, bn)) {
1271 		if (check("Repair is in a mounted partition, continue"))
1272 			return (-1);
1273 	}
1274 	/*
1275 	 * check for partitions being used for swapping in format zone
1276 	 */
1277 	if (checkswap(bn, bn)) {
1278 		if (check("Repair is in a partition which is currently \
1279 being used for swapping.\ncontinue"))
1280 		return (-1);
1281 	}
1282 
1283 	if (checkdevinuse(cur_disk->disk_name, bn, bn, 0, 0)) {
1284 		if (check("Repair is in a partition which is currently "
1285 		    "in use.\ncontinue"))
1286 			return (-1);
1287 	}
1288 
1289 	buf = zalloc((cur_disk->disk_lbasize == 0) ?
1290 	    SECSIZE : cur_disk->disk_lbasize);
1291 
1292 	/*
1293 	 * Try to read the sector before repairing it.  If we can
1294 	 * get good data out of it, we can write that data back
1295 	 * after the repair.  If the sector looks ok, ask the
1296 	 * user to confirm the repair, since it doesn't appear
1297 	 * necessary.  Try reading the block several times to
1298 	 * see if we can read it consistently.
1299 	 *
1300 	 * First, let's see if the block appears to have problems...
1301 	 */
1302 	block_has_error = 1;
1303 	for (i = 0; i < 5; i++) {
1304 		status = (*cur_ops->op_rdwr)(DIR_READ, cur_file, bn,
1305 		    1, buf, (F_SILENT | F_ALLERRS), NULL);
1306 		if (status)
1307 			break;		/* one of the tries failed */
1308 	}
1309 	if (status == 0) {
1310 		block_has_error = 0;
1311 		if (check("\
1312 This block doesn't appear to be bad.  Repair it anyway")) {
1313 			free(buf);
1314 			return (0);
1315 		}
1316 	}
1317 	/*
1318 	 * Last chance...
1319 	 */
1320 	if (check("Ready to repair defect, continue")) {
1321 		free(buf);
1322 		return (-1);
1323 	}
1324 	/*
1325 	 * We're committed to repairing it.  Try to get any good
1326 	 * data out of the block if possible.  Note that we do
1327 	 * not set the F_ALLERRS flag.
1328 	 */
1329 	buf_is_good = 0;
1330 	for (i = 0; i < 5; i++) {
1331 		status = (*cur_ops->op_rdwr)(DIR_READ, cur_file, bn,
1332 		    1, buf, F_SILENT, NULL);
1333 		if (status == 0) {
1334 			buf_is_good = 1;
1335 			break;
1336 		}
1337 	}
1338 	/*
1339 	 * Lock out interrupts so the disk can't get out of sync with
1340 	 * the defect list.
1341 	 */
1342 	enter_critical();
1343 
1344 	fmt_print("Repairing ");
1345 	if (block_has_error) {
1346 		fmt_print("%s error on ", buf_is_good ? "soft" : "hard");
1347 	}
1348 	fmt_print("block %llu (", bn);
1349 	pr_dblock(fmt_print, bn);
1350 	fmt_print(")...");
1351 	/*
1352 	 * Do the repair.
1353 	 */
1354 	status = (*cur_ops->op_repair)(bn, F_NORMAL);
1355 	if (status) {
1356 		fmt_print("failed.\n\n");
1357 	} else {
1358 		/*
1359 		 * The repair worked.  Write the old data to the new
1360 		 * block if we were able to read it, otherwise
1361 		 * zero out the new block.  If it looks like the
1362 		 * new block is bad, let the user know that, too.
1363 		 * Should we attempt auto-repair in this case?
1364 		 */
1365 		fmt_print("ok.\n");
1366 		if (!buf_is_good) {
1367 			bzero(buf, cur_disk->disk_lbasize);
1368 		}
1369 		status = (*cur_ops->op_rdwr)(DIR_WRITE, cur_file, bn,
1370 		    1, buf, (F_SILENT | F_ALLERRS), NULL);
1371 		if (status == 0) {
1372 			status = (*cur_ops->op_rdwr)(DIR_READ, cur_file,
1373 			    bn, 1, buf, (F_SILENT | F_ALLERRS), NULL);
1374 		}
1375 		if (status) {
1376 			fmt_print("The new block %llu (", bn);
1377 			pr_dblock(fmt_print, bn);
1378 			fmt_print(") also appears defective.\n");
1379 		}
1380 		fmt_print("\n");
1381 		/*
1382 		 * Add the bad sector to the defect list, write out
1383 		 * the defect list, and kill off the working list so
1384 		 * it will get synced up with the current defect list
1385 		 * next time we need it.
1386 		 *
1387 		 * For embedded scsi, we don't require a defect list.
1388 		 * However, if we have one, add the defect if the
1389 		 * list includes the grown list.  If not, kill it
1390 		 * to force a resync if we need the list later.
1391 		 */
1392 		if (EMBEDDED_SCSI) {
1393 			if (cur_list.list != NULL) {
1394 				if (cur_list.flags & LIST_PGLIST) {
1395 					add_ldef(bn, &cur_list);
1396 				} else {
1397 					kill_deflist(&cur_list);
1398 				}
1399 			}
1400 		} else if (cur_ctype->ctype_flags & CF_WLIST) {
1401 			kill_deflist(&cur_list);
1402 			if (*cur_ops->op_ex_cur != NULL) {
1403 				(*cur_ops->op_ex_cur)(&cur_list);
1404 				fmt_print("Current list updated\n");
1405 			}
1406 		} else {
1407 			add_ldef(bn, &cur_list);
1408 			write_deflist(&cur_list);
1409 		}
1410 		kill_deflist(&work_list);
1411 	}
1412 	exit_critical();
1413 	free(buf);
1414 
1415 	/*
1416 	 * Return status.
1417 	 */
1418 	return (status);
1419 }
1420 
1421 /*
1422  * This routine implements the 'show' command.  It translates a disk
1423  * block given in any format into decimal, hexadecimal, and
1424  * cylinder/head/sector format.
1425  */
1426 int
1427 c_show()
1428 {
1429 	u_ioparam_t	ioparam;
1430 	diskaddr_t	bn;
1431 
1432 	/*
1433 	 * There must be a current disk type, so we will know the geometry.
1434 	 */
1435 	if (cur_dtype == NULL) {
1436 		err_print("Current Disk Type is not set.\n");
1437 		return (-1);
1438 	}
1439 	/*
1440 	 * Ask the user for a disk block.
1441 	 */
1442 	ioparam.io_bounds.lower = 0;
1443 	if (cur_disk->label_type == L_TYPE_SOLARIS) {
1444 		ioparam.io_bounds.upper = physsects() - 1;
1445 	} else {
1446 		ioparam.io_bounds.upper = cur_parts->etoc->efi_last_lba;
1447 	}
1448 	bn = input(FIO_BN, "Enter a disk block", ':',
1449 	    &ioparam, (int *)NULL, DATA_INPUT);
1450 	/*
1451 	 * Echo it back.
1452 	 */
1453 	fmt_print("Disk block = %lld = 0x%llx = (", bn, bn);
1454 	pr_dblock(fmt_print, bn);
1455 	fmt_print(")\n\n");
1456 	return (0);
1457 }
1458 
1459 /*
1460  * This routine implements the 'label' command.  It writes the
1461  * primary and backup labels onto the current disk.
1462  */
1463 int
1464 c_label()
1465 {
1466 	int			status;
1467 	int			deflt, *defltptr = NULL;
1468 
1469 	/*
1470 	 * There must be a current disk type (and therefore a current disk).
1471 	 */
1472 	if (cur_dtype == NULL) {
1473 		err_print("Current Disk Type is not set.\n");
1474 		return (-1);
1475 	}
1476 	/*
1477 	 * The current disk must be formatted to label it.
1478 	 */
1479 	if (!(cur_flags & DISK_FORMATTED)) {
1480 		err_print("Current Disk is unformatted.\n");
1481 		return (-1);
1482 	}
1483 	/*
1484 	 * Check for a valid fdisk table entry for Solaris
1485 	 */
1486 	if (!good_fdisk()) {
1487 		return (-1);
1488 	}
1489 	/*
1490 	 * Check to see if there are any mounted file systems anywhere
1491 	 * on the current disk.  If so, refuse to label the disk, but
1492 	 * only if the partitions would change for the mounted partitions.
1493 	 *
1494 	 */
1495 	if (checkmount((diskaddr_t)-1, (diskaddr_t)-1)) {
1496 		/* Bleagh, too descriptive */
1497 		if (check_label_with_mount()) {
1498 			err_print("Cannot label disk while it has "
1499 			    "mounted partitions.\n\n");
1500 			return (-1);
1501 		}
1502 	}
1503 
1504 	/*
1505 	 * check to see if there any partitions being used for swapping
1506 	 * on the current disk.  If so, refuse to label the disk, but
1507 	 * only if the partitions would change for the mounted partitions.
1508 	 */
1509 	if (checkswap((diskaddr_t)-1, (diskaddr_t)-1)) {
1510 		if (check_label_with_swap()) {
1511 			err_print("Cannot label disk while its "
1512 			    "partitions are currently being used for "
1513 			    "swapping.\n");
1514 			return (-1);
1515 		}
1516 	}
1517 
1518 	/*
1519 	 * Check to see if any partitions used for svm, vxvm or live upgrade
1520 	 * are on the disk. If so, refuse to label the disk, but only
1521 	 * if we are trying to shrink a partition in use.
1522 	 */
1523 	if (checkdevinuse(cur_disk->disk_name, (diskaddr_t)-1,
1524 	    (diskaddr_t)-1, 0, 1)) {
1525 		err_print("Cannot label disk when "
1526 		    "partitions are in use as described.\n");
1527 		return (-1);
1528 	}
1529 
1530 	/*
1531 	 * If there is not a current partition map, warn the user we
1532 	 * are going to use the default.  The default is the first
1533 	 * partition map we encountered in the data file.  If there is
1534 	 * no default we give up.
1535 	 */
1536 	if (cur_parts == NULL) {
1537 		fmt_print("Current Partition Table is not set, "
1538 		    "using default.\n");
1539 		cur_disk->disk_parts = cur_parts = cur_dtype->dtype_plist;
1540 		if (cur_parts == NULL) {
1541 			err_print("No default available, cannot label.\n");
1542 			return (-1);
1543 		}
1544 	}
1545 	/*
1546 	 * If expert (-e) mode, then ask user if they wish
1547 	 * to change the current solaris label into an EFI one
1548 	 */
1549 	if (expert_mode) {
1550 #if defined(_SUNOS_VTOC_8)
1551 		int 		i;
1552 #endif
1553 		int 		choice;
1554 		u_ioparam_t		ioparam;
1555 		struct extvtoc	vtoc;
1556 		struct dk_label	label;
1557 		struct dk_gpt	*vtoc64;
1558 		struct efi_info	efinfo;
1559 		struct disk_type	*dptr;
1560 
1561 		/* Ask user what label to use */
1562 		fmt_print("[0] SMI Label\n");
1563 		fmt_print("[1] EFI Label\n");
1564 		ioparam.io_bounds.lower = 0;
1565 		ioparam.io_bounds.upper = 1;
1566 		if (cur_label == L_TYPE_SOLARIS)
1567 			deflt = 0;
1568 		else
1569 			deflt = 1;
1570 		defltptr = &deflt;
1571 		choice = input(FIO_INT, "Specify Label type", ':',
1572 		    &ioparam, defltptr, DATA_INPUT);
1573 		if ((choice == 0) && (cur_label == L_TYPE_SOLARIS)) {
1574 			goto expert_end;
1575 		} else if ((choice == 1) && (cur_label == L_TYPE_EFI)) {
1576 			goto expert_end;
1577 		}
1578 		switch (choice) {
1579 		case 0:
1580 		/*
1581 		 * EFI label to SMI label
1582 		 */
1583 		if (cur_dtype->capacity > INFINITY) {
1584 			fmt_print("Warning: SMI labels only support up to "
1585 			    "2 TB.\n");
1586 		}
1587 
1588 		if (cur_disk->fdisk_part.systid == EFI_PMBR) {
1589 			fmt_print("Warning: This disk has an EFI label. "
1590 			    "Changing to SMI label will erase all\n"
1591 			    "current partitions.\n");
1592 			if (check("Continue"))
1593 			return (-1);
1594 #if defined(_FIRMWARE_NEEDS_FDISK)
1595 			fmt_print("You must use fdisk to delete the current "
1596 			    "EFI partition and create a new\n"
1597 			    "Solaris partition before you can convert the "
1598 			    "label.\n");
1599 			return (-1);
1600 #endif
1601 		}
1602 
1603 #if defined(_FIRMWARE_NEEDS_FDISK)
1604 		if (!(((cur_disk->fdisk_part.systid != SUNIXOS) ||
1605 		    (cur_disk->fdisk_part.systid != SUNIXOS2)) &&
1606 		    (cur_disk->fdisk_part.numsect > 0))) {
1607 			fmt_print("You must use fdisk to create a Solaris "
1608 			    "partition before you can convert the label.\n");
1609 			return (-1);
1610 		}
1611 #endif
1612 
1613 		(void) memset((char *)&label, 0, sizeof (struct dk_label));
1614 
1615 		(void) strcpy(x86_devname, cur_disk->disk_name);
1616 		if (cur_ctype->ctype_ctype == DKC_DIRECT)
1617 			dptr = auto_direct_get_geom_label(cur_file,  &label);
1618 		else
1619 			dptr = auto_sense(cur_file, 1, &label);
1620 		if (dptr == NULL) {
1621 			fmt_print("Autoconfiguration failed.\n");
1622 			return (-1);
1623 		}
1624 
1625 		pcyl = label.dkl_pcyl;
1626 		ncyl = label.dkl_ncyl;
1627 		acyl = label.dkl_acyl;
1628 		nhead = label.dkl_nhead;
1629 		nsect = label.dkl_nsect;
1630 
1631 		if (delete_disk_type(cur_disk->disk_type) == 0) {
1632 			cur_label = L_TYPE_SOLARIS;
1633 			cur_disk->label_type = L_TYPE_SOLARIS;
1634 			cur_disk->disk_type = dptr;
1635 			cur_disk->disk_parts = dptr->dtype_plist;
1636 			cur_dtype = dptr;
1637 			cur_parts = dptr->dtype_plist;
1638 
1639 			if (status = write_label())
1640 				err_print("Label failed.\n");
1641 			else
1642 				cur_disk->disk_flags &= ~DSK_LABEL_DIRTY;
1643 
1644 			return (status);
1645 		} else {
1646 			err_print("Label failed.\n");
1647 			return (-1);
1648 		}
1649 
1650 
1651 		case 1:
1652 		/*
1653 		 * SMI label to EFI label
1654 		 */
1655 
1656 
1657 		fmt_print("Warning: This disk has an SMI label. Changing to "
1658 		    "EFI label will erase all\ncurrent partitions.\n");
1659 
1660 		if (check("Continue")) {
1661 			return (-1);
1662 		}
1663 
1664 		if (get_disk_info(cur_file, &efinfo) != 0) {
1665 			return (-1);
1666 		}
1667 		(void) memset((char *)&label, 0, sizeof (struct dk_label));
1668 		label.dkl_pcyl = pcyl;
1669 		label.dkl_ncyl = ncyl;
1670 		label.dkl_acyl = acyl;
1671 #if defined(_SUNOS_VTOC_16)
1672 		label.dkl_bcyl = bcyl;
1673 #endif			/* defined(_SUNOC_VTOC_16) */
1674 		label.dkl_nhead = nhead;
1675 		label.dkl_nsect = nsect;
1676 #if defined(_SUNOS_VTOC_8)
1677 		for (i = 0; i < NDKMAP; i++) {
1678 			label.dkl_map[i] = cur_parts->pinfo_map[i];
1679 		}
1680 #endif			/* defined(_SUNOS_VTOC_8) */
1681 		label.dkl_magic = DKL_MAGIC;
1682 		label.dkl_vtoc = cur_parts->vtoc;
1683 		if (label_to_vtoc(&vtoc, &label) == -1) {
1684 			return (-1);
1685 		}
1686 		if (SMI_vtoc_to_EFI(cur_file, &vtoc64) == -1) {
1687 			return (-1);
1688 		}
1689 		if (efi_write(cur_file, vtoc64) != 0) {
1690 			err_check(vtoc64);
1691 			err_print("Warning: error writing EFI.\n");
1692 			return (-1);
1693 		} else {
1694 			cur_disk->disk_flags &= ~DSK_LABEL_DIRTY;
1695 		}
1696 		/*
1697 		 * copy over the EFI vtoc onto the SMI vtoc and return
1698 		 * okay.
1699 		 */
1700 		dptr = auto_efi_sense(cur_file, &efinfo);
1701 		if (dptr == NULL) {
1702 			fmt_print("Autoconfiguration failed.\n");
1703 			return (-1);
1704 		}
1705 
1706 		cur_label = L_TYPE_EFI;
1707 		cur_disk->label_type = L_TYPE_EFI;
1708 		cur_disk->disk_type = dptr;
1709 		cur_disk->disk_parts = dptr->dtype_plist;
1710 		cur_dtype = dptr;
1711 		cur_parts = dptr->dtype_plist;
1712 		cur_parts->etoc = vtoc64;
1713 
1714 		ncyl = pcyl = nsect = psect = acyl = phead = 0;
1715 
1716 		/*
1717 		 * Get the Solais Fdisk Partition information.
1718 		 */
1719 		(void) copy_solaris_part(&cur_disk->fdisk_part);
1720 
1721 		return (0);
1722 		}
1723 	}
1724 
1725 expert_end:
1726 	/*
1727 	 * Make sure the user is serious.
1728 	 */
1729 	if (check("Ready to label disk, continue")) {
1730 		return (-1);
1731 	}
1732 	/*
1733 	 * Write the labels out (this will also notify unix) and
1734 	 * return status.
1735 	 */
1736 	fmt_print("\n");
1737 	if (status = write_label())
1738 		err_print("Label failed.\n");
1739 	return (status);
1740 }
1741 
1742 /*
1743  * This routine implements the 'analyze' command.  It simply runs
1744  * the analyze menu.
1745  */
1746 int
1747 c_analyze()
1748 {
1749 
1750 	/*
1751 	 * There must be a current disk type (and therefor a current disk).
1752 	 */
1753 	if (cur_dtype == NULL) {
1754 		err_print("Current Disk Type is not set.\n");
1755 		return (-1);
1756 	}
1757 	cur_menu++;
1758 	last_menu = cur_menu;
1759 
1760 	/*
1761 	 * Run the menu.
1762 	 */
1763 	run_menu(menu_analyze, "ANALYZE", "analyze", 0);
1764 	cur_menu--;
1765 	return (0);
1766 }
1767 
1768 /*
1769  * This routine implements the 'defect' command.  It simply runs
1770  * the defect menu.
1771  */
1772 int
1773 c_defect()
1774 {
1775 	int	i;
1776 
1777 	/*
1778 	 * There must be a current disk type (and therefor a current disk).
1779 	 */
1780 	if (cur_dtype == NULL) {
1781 		err_print("Current Disk Type is not set.\n");
1782 		return (-1);
1783 	}
1784 
1785 	/*
1786 	 * Check for the defect management and list management ops and
1787 	 * display appropriate message.
1788 	 */
1789 	if ((cur_ops->op_ex_man == NULL) && (cur_ops->op_ex_cur == NULL) &&
1790 	    (cur_ops->op_create == NULL) && (cur_ops->op_wr_cur == NULL)) {
1791 		err_print("Controller does not support defect management\n");
1792 		err_print("or disk supports automatic defect management.\n");
1793 		return (-1);
1794 	}
1795 	cur_menu++;
1796 	last_menu = cur_menu;
1797 
1798 	/*
1799 	 * Lock out interrupt while we manipulate the defect lists.
1800 	 */
1801 	enter_critical();
1802 	/*
1803 	 * If the working list is null but there is a current list,
1804 	 * update the working list to be a copy of the current list.
1805 	 */
1806 	if ((work_list.list == NULL) && (cur_list.list != NULL)) {
1807 		work_list.header = cur_list.header;
1808 		work_list.list = (struct defect_entry *)zalloc(
1809 		    deflist_size(cur_blksz, work_list.header.count) *
1810 		    cur_blksz);
1811 		for (i = 0; i < work_list.header.count; i++)
1812 			*(work_list.list + i) = *(cur_list.list + i);
1813 		work_list.flags = cur_list.flags & LIST_PGLIST;
1814 	}
1815 	exit_critical();
1816 	/*
1817 	 * Run the menu.
1818 	 */
1819 	run_menu(menu_defect, "DEFECT", "defect", 0);
1820 	cur_menu--;
1821 
1822 	/*
1823 	 * If the user has modified the working list but not committed
1824 	 * it, warn him that he is probably making a mistake.
1825 	 */
1826 	if (work_list.flags & LIST_DIRTY) {
1827 		if (!EMBEDDED_SCSI) {
1828 			err_print(
1829 		"Warning: working defect list modified; but not committed.\n");
1830 			if (!check(
1831 		"Do you wish to commit changes to current defect list"))
1832 			(void) do_commit();
1833 		}
1834 	}
1835 	return (0);
1836 }
1837 
1838 /*
1839  * This routine implements the 'backup' command.  It allows the user
1840  * to search for backup labels on the current disk.  This is useful
1841  * if the primary label was lost and the user wishes to recover the
1842  * partition information for the disk. The disk is relabeled and
1843  * the current defect list is written out if a backup label is found.
1844  */
1845 int
1846 c_backup()
1847 {
1848 	struct	dk_label label;
1849 	struct	disk_type *dtype;
1850 	struct	partition_info *parts, *plist;
1851 	diskaddr_t	bn;
1852 	int	sec, head, i;
1853 	char	*buf;
1854 
1855 	/*
1856 	 * There must be a current disk type (and therefore a current disk).
1857 	 */
1858 	if (cur_dtype == NULL) {
1859 		err_print("Current Disk Type is not set.\n");
1860 		return (-1);
1861 	}
1862 	/*
1863 	 * The disk must be formatted to read backup labels.
1864 	 */
1865 	if (!(cur_flags & DISK_FORMATTED)) {
1866 		err_print("Current Disk is unformatted.\n");
1867 		return (-1);
1868 	}
1869 	/*
1870 	 * Check for a valid fdisk table entry for Solaris
1871 	 */
1872 	if (!good_fdisk()) {
1873 		return (-1);
1874 	}
1875 	/*
1876 	 * If we found a primary label on this disk, make sure
1877 	 * the user is serious.
1878 	 */
1879 	if (cur_disk->label_type == L_TYPE_EFI) {
1880 		if (((cur_disk->disk_parts->etoc->efi_flags &
1881 		    EFI_GPT_PRIMARY_CORRUPT) == 0) &&
1882 		    check("Disk has a primary label, still continue"))
1883 			return (-1);
1884 		fmt_print("Restoring primary label.\n");
1885 		if (write_label()) {
1886 			err_print("Failed\n");
1887 			return (-1);
1888 		}
1889 		return (0);
1890 	} else if (((cur_disk->disk_flags & (DSK_LABEL | DSK_LABEL_DIRTY)) ==
1891 	    DSK_LABEL) &&
1892 	    (check("Disk has a primary label, still continue"))) {
1893 		return (-1);
1894 	}
1895 
1896 	buf = zalloc(cur_blksz);
1897 	fmt_print("Searching for backup labels...");
1898 	(void) fflush(stdout);
1899 
1900 	/*
1901 	 * Some disks have the backup labels in a strange place.
1902 	 */
1903 	if (cur_ctype->ctype_flags & CF_BLABEL)
1904 		head = 2;
1905 	else
1906 		head = nhead - 1;
1907 	/*
1908 	 * Loop through each copy of the backup label.
1909 	 */
1910 	for (sec = 1; ((sec < BAD_LISTCNT * 2 + 1) && (sec < nsect));
1911 	    sec += 2) {
1912 		bn = chs2bn(ncyl + acyl - 1, head, sec) + solaris_offset;
1913 		/*
1914 		 * Attempt to read it.
1915 		 */
1916 		if ((*cur_ops->op_rdwr)(DIR_READ, cur_file, bn,
1917 		    1, buf, F_NORMAL, NULL)) {
1918 			continue;
1919 		}
1920 
1921 		(void *) memcpy((char *)&label, buf, sizeof (struct dk_label));
1922 
1923 		/*
1924 		 * Verify that it is a reasonable label.
1925 		 */
1926 		if (!checklabel(&label))
1927 			continue;
1928 		if (trim_id(label.dkl_asciilabel))
1929 			continue;
1930 		/*
1931 		 * Lock out interrupts while we manipulate lists.
1932 		 */
1933 		enter_critical();
1934 		fmt_print("found.\n");
1935 		/*
1936 		 * Find out which disk type the backup label claims.
1937 		 */
1938 		for (dtype = cur_ctype->ctype_dlist; dtype != NULL;
1939 		    dtype = dtype->dtype_next)
1940 			if (dtype_match(&label, dtype))
1941 				break;
1942 		/*
1943 		 * If it disagrees with our current type, something
1944 		 * real bad is happening.
1945 		 */
1946 		if (dtype != cur_dtype) {
1947 			if (dtype == NULL) {
1948 				fmt_print("\
1949 Unknown disk type in backup label\n");
1950 				exit_critical();
1951 				free(buf);
1952 				return (-1);
1953 			}
1954 			fmt_print("Backup label claims different type:\n");
1955 			fmt_print("    <%s cyl %d alt %d hd %d sec %d>\n",
1956 			    label.dkl_asciilabel, label.dkl_ncyl,
1957 			    label.dkl_acyl, label.dkl_nhead,
1958 			    label.dkl_nsect);
1959 			if (check("Continue")) {
1960 				exit_critical();
1961 				free(buf);
1962 				return (-1);
1963 			}
1964 			cur_dtype = dtype;
1965 		}
1966 		/*
1967 		 * Try to match the partition map with a known map.
1968 		 */
1969 		for (parts = dtype->dtype_plist; parts != NULL;
1970 		    parts = parts->pinfo_next)
1971 			if (parts_match(&label, parts))
1972 				break;
1973 		/*
1974 		 * If we couldn't match it, allocate space for a new one,
1975 		 * fill in the info, and add it to the list.  The name
1976 		 * for the new map is derived from the disk name.
1977 		 */
1978 		if (parts == NULL) {
1979 			parts = (struct partition_info *)
1980 			    zalloc(sizeof (struct partition_info));
1981 			plist = dtype->dtype_plist;
1982 			if (plist == NULL)
1983 				dtype->dtype_plist = parts;
1984 			else {
1985 				while (plist->pinfo_next != NULL)
1986 					plist = plist->pinfo_next;
1987 				plist->pinfo_next = parts;
1988 			}
1989 			parts->pinfo_name = alloc_string("original");
1990 			for (i = 0; i < NDKMAP; i++)
1991 
1992 #if defined(_SUNOS_VTOC_8)
1993 				parts->pinfo_map[i] = label.dkl_map[i];
1994 
1995 #elif defined(_SUNOS_VTOC_16)
1996 				parts->pinfo_map[i].dkl_cylno  =
1997 				    label.dkl_vtoc.v_part[i].p_start / spc();
1998 				parts->pinfo_map[i].dkl_nblk =
1999 				    label.dkl_vtoc.v_part[i].p_size;
2000 #else
2001 #error No VTOC layout defined.
2002 #endif /* defined(_SUNOS_VTOC_8) */
2003 			parts->vtoc = label.dkl_vtoc;
2004 		}
2005 		/*
2006 		 * We now have a partition map.  Make it the current map.
2007 		 */
2008 		cur_disk->disk_parts = cur_parts = parts;
2009 		exit_critical();
2010 		/*
2011 		 * Rewrite the labels and defect lists, as appropriate.
2012 		 */
2013 		if (EMBEDDED_SCSI) {
2014 			fmt_print("Restoring primary label.\n");
2015 			if (write_label()) {
2016 				free(buf);
2017 				return (-1);
2018 			}
2019 		} else {
2020 			fmt_print("Restoring primary label and defect list.\n");
2021 			if (write_label()) {
2022 				free(buf);
2023 				return (-1);
2024 			}
2025 			if (cur_list.list != NULL)
2026 				write_deflist(&cur_list);
2027 		}
2028 		fmt_print("\n");
2029 		free(buf);
2030 		return (0);
2031 	}
2032 	/*
2033 	 * If we didn't find any backup labels, say so.
2034 	 */
2035 	fmt_print("not found.\n\n");
2036 	free(buf);
2037 	return (0);
2038 }
2039 
2040 /*
2041  * This routine is called by c_verify() for an EFI labeled disk
2042  */
2043 static int
2044 c_verify_efi()
2045 {
2046 	struct efi_info efi_info;
2047 	struct	partition_info	tmp_pinfo;
2048 	int status;
2049 
2050 	status = read_efi_label(cur_file, &efi_info);
2051 	if (status != 0) {
2052 		err_print("Warning: Could not read label.\n");
2053 		return (-1);
2054 	}
2055 	if (cur_parts->etoc->efi_flags & EFI_GPT_PRIMARY_CORRUPT) {
2056 		err_print("Reading the primary EFI GPT label ");
2057 		err_print("failed.  Using backup label.\n");
2058 		err_print("Use the 'backup' command to restore ");
2059 		err_print("the primary label.\n");
2060 	}
2061 	tmp_pinfo.etoc = efi_info.e_parts;
2062 	fmt_print("\n");
2063 	if (cur_parts->etoc->efi_parts[8].p_name) {
2064 		fmt_print("Volume name = <%8s>\n",
2065 		    cur_parts->etoc->efi_parts[8].p_name);
2066 	} else {
2067 		fmt_print("Volume name = <        >\n");
2068 	}
2069 	fmt_print("ascii name  = ");
2070 	print_efi_string(efi_info.vendor, efi_info.product,
2071 	    efi_info.revision, efi_info.capacity);
2072 	fmt_print("\n");
2073 
2074 	fmt_print("bytes/sector	=  %d\n", cur_blksz);
2075 	fmt_print("sectors = %llu\n", cur_parts->etoc->efi_last_lba);
2076 	fmt_print("accessible sectors = %llu\n",
2077 	    cur_parts->etoc->efi_last_u_lba);
2078 
2079 	print_map(&tmp_pinfo);
2080 	return (0);
2081 }
2082 
2083 /*
2084  * This routine implements the 'verify' command.  It allows the user
2085  * to read the labels on the current disk.
2086  */
2087 int
2088 c_verify()
2089 {
2090 	struct	dk_label p_label, b_label, *label;
2091 	struct	partition_info tmp_pinfo;
2092 	diskaddr_t	bn;
2093 	int	sec, head, i, status;
2094 	int	p_label_bad = 0;
2095 	int	b_label_bad = 0;
2096 	int	p_label_found = 0;
2097 	int	b_label_found = 0;
2098 	char	id_str[128];
2099 	char	*buf;
2100 
2101 	/*
2102 	 * There must be a current disk type (and therefore a current disk).
2103 	 */
2104 	if (cur_dtype == NULL) {
2105 		err_print("Current Disk Type is not set.\n");
2106 		return (-1);
2107 	}
2108 	/*
2109 	 * The disk must be formatted to read labels.
2110 	 */
2111 	if (!(cur_flags & DISK_FORMATTED)) {
2112 		err_print("Current Disk is unformatted.\n");
2113 		return (-1);
2114 	}
2115 	/*
2116 	 * Check for a valid fdisk table entry for Solaris
2117 	 */
2118 	if (!good_fdisk()) {
2119 		return (-1);
2120 	}
2121 	/*
2122 	 * Branch off here if the disk is EFI labelled.
2123 	 */
2124 	if (cur_label == L_TYPE_EFI) {
2125 		return (c_verify_efi());
2126 	}
2127 	/*
2128 	 * Attempt to read the primary label.
2129 	 */
2130 	status = read_label(cur_file, &p_label);
2131 	if (status == -1) {
2132 		err_print("Warning: Could not read primary label.\n");
2133 		p_label_bad = 1;
2134 	} else {
2135 		/*
2136 		 * Verify that it is a reasonable label.
2137 		 */
2138 		/*
2139 		 * Save complete ascii string for printing later.
2140 		 */
2141 		(void) strncpy(id_str, p_label.dkl_asciilabel, 128);
2142 
2143 		if ((!checklabel((struct dk_label *)&p_label)) ||
2144 		    (trim_id(p_label.dkl_asciilabel))) {
2145 			err_print("\
2146 Warning: Primary label appears to be corrupt.\n");
2147 			p_label_bad = 1;
2148 		} else {
2149 			p_label_found = 1;
2150 			/*
2151 			 * Make sure it matches current label
2152 			 */
2153 			if ((!dtype_match(&p_label, cur_dtype)) ||
2154 			    (!parts_match(&p_label, cur_parts))) {
2155 				err_print("\
2156 Warning: Primary label on disk appears to be different from\ncurrent label.\n");
2157 				p_label_bad = 1;
2158 			}
2159 		}
2160 	}
2161 
2162 	/*
2163 	 * Read backup labels.
2164 	 * Some disks have the backup labels in a strange place.
2165 	 */
2166 	if (cur_ctype->ctype_flags & CF_BLABEL)
2167 		head = 2;
2168 	else
2169 		head = nhead - 1;
2170 
2171 	buf = zalloc(cur_blksz);
2172 	/*
2173 	 * Loop through each copy of the backup label.
2174 	 */
2175 	for (sec = 1; ((sec < BAD_LISTCNT * 2 + 1) && (sec < nsect));
2176 	    sec += 2) {
2177 		bn = chs2bn(ncyl + acyl - 1, head, sec) + solaris_offset;
2178 		/*
2179 		 * Attempt to read it.
2180 		 */
2181 		if ((*cur_ops->op_rdwr)(DIR_READ, cur_file, bn,
2182 		    1, buf, F_NORMAL, NULL))
2183 			continue;
2184 
2185 		(void *) memcpy((char *)&b_label, buf,
2186 		    sizeof (struct dk_label));
2187 
2188 		/*
2189 		 * Verify that it is a reasonable label.
2190 		 */
2191 		if (!checklabel(&b_label))
2192 			continue;
2193 
2194 		/*
2195 		 * Save complete label only if no primary label exists
2196 		 */
2197 		if (!p_label_found)
2198 			(void) strncpy(id_str, b_label.dkl_asciilabel, 128);
2199 
2200 		if (trim_id(b_label.dkl_asciilabel))
2201 			continue;
2202 		b_label_found = 1;
2203 		/*
2204 		 * Compare against primary label
2205 		 */
2206 		if (p_label_found) {
2207 			if ((strcmp(b_label.dkl_asciilabel,
2208 			    p_label.dkl_asciilabel) != 0) ||
2209 			    (b_label.dkl_ncyl != p_label.dkl_ncyl) ||
2210 			    (b_label.dkl_acyl != p_label.dkl_acyl) ||
2211 			    (b_label.dkl_nhead != p_label.dkl_nhead) ||
2212 			    (b_label.dkl_nsect != p_label.dkl_nsect)) {
2213 				b_label_bad = 1;
2214 			} else {
2215 				for (i = 0; i < NDKMAP; i++) {
2216 #if defined(_SUNOS_VTOC_8)
2217 					if ((b_label.dkl_map[i].dkl_cylno !=
2218 					    p_label.dkl_map[i].dkl_cylno) ||
2219 					    (b_label.dkl_map[i].dkl_nblk !=
2220 					    p_label.dkl_map[i].dkl_nblk)) {
2221 						b_label_bad = 1;
2222 						break;
2223 					}
2224 
2225 #elif defined(_SUNOS_VTOC_16)
2226 					if ((b_label.dkl_vtoc.v_part[i].p_tag !=
2227 					    p_label.dkl_vtoc.v_part[i].p_tag) ||
2228 					    (b_label.dkl_vtoc.v_part[i].p_flag
2229 					    != p_label.dkl_vtoc.v_part[i].
2230 					    p_flag) ||
2231 					    (b_label.dkl_vtoc.v_part[i].p_start
2232 					    != p_label.dkl_vtoc.v_part[i].
2233 					    p_start) ||
2234 					    (b_label.dkl_vtoc.v_part[i].p_size
2235 					    != p_label.dkl_vtoc.v_part[i].
2236 					    p_size)) {
2237 						b_label_bad = 1;
2238 						break;
2239 					}
2240 #else
2241 #error No VTOC layout defined.
2242 #endif /* defined(_SUNOS_VTOC_8) */
2243 				}
2244 			}
2245 		}
2246 		if (b_label_bad)
2247 			err_print(
2248 "Warning: Primary and backup labels do not match.\n");
2249 		break;
2250 	}
2251 	/*
2252 	 * If we didn't find any backup labels, say so.
2253 	 */
2254 	if (!b_label_found)
2255 		err_print("Warning: Could not read backup labels.\n");
2256 
2257 	if ((!b_label_found) || (p_label_bad) || (b_label_bad))
2258 		err_print("\n\
2259 Warning: Check the current partitioning and 'label' the disk or use the\n\
2260 \t 'backup' command.\n");
2261 
2262 	/*
2263 	 * Print label information.
2264 	 */
2265 	if (p_label_found) {
2266 		fmt_print("\nPrimary label contents:\n");
2267 		label = &p_label;
2268 	} else if (b_label_found) {
2269 		fmt_print("\nBackup label contents:\n");
2270 		label = &b_label;
2271 	} else {
2272 		free(buf);
2273 		return (0);
2274 	}
2275 
2276 	/*
2277 	 * Must put info into partition_info struct for
2278 	 * for print routine.
2279 	 */
2280 	bzero(&tmp_pinfo, sizeof (struct partition_info));
2281 	for (i = 0; i < NDKMAP; i++) {
2282 
2283 #if defined(_SUNOS_VTOC_8)
2284 		tmp_pinfo.pinfo_map[i] = label->dkl_map[i];
2285 
2286 #elif defined(_SUNOS_VTOC_16)
2287 		tmp_pinfo.pinfo_map[i].dkl_cylno =
2288 		    label->dkl_vtoc.v_part[i].p_start / spc();
2289 		tmp_pinfo.pinfo_map[i].dkl_nblk =
2290 		    label->dkl_vtoc.v_part[i].p_size;
2291 #else
2292 #error No VTOC layout defined.
2293 #endif /* defined(_SUNOS_VTOC_8) */
2294 	}
2295 	tmp_pinfo.vtoc = label->dkl_vtoc;
2296 
2297 	fmt_print("\n");
2298 	fmt_print("Volume name = <%8s>\n", label->dkl_vtoc.v_volume);
2299 	fmt_print("ascii name  = <%s>\n", id_str);
2300 	fmt_print("pcyl        = %4d\n", label->dkl_pcyl);
2301 	fmt_print("ncyl        = %4d\n", label->dkl_ncyl);
2302 	fmt_print("acyl        = %4d\n", label->dkl_acyl);
2303 
2304 #if defined(_SUNOS_VTOC_16)
2305 	fmt_print("bcyl        = %4d\n", label->dkl_bcyl);
2306 #endif /* defined(_SUNOS_VTOC_16) */
2307 
2308 	fmt_print("nhead       = %4d\n", label->dkl_nhead);
2309 	fmt_print("nsect       = %4d\n", label->dkl_nsect);
2310 
2311 	print_map(&tmp_pinfo);
2312 	free(buf);
2313 	return (0);
2314 }
2315 
2316 
2317 /*
2318  * This command implements the inquiry command, for embedded SCSI
2319  * disks only, which issues a SCSI inquiry command, and
2320  * displays the resulting vendor, product id and revision level.
2321  */
2322 int
2323 c_inquiry()
2324 {
2325 	char			inqbuf[255];
2326 	struct scsi_inquiry	*inq;
2327 
2328 	assert(SCSI);
2329 
2330 	inq = (struct scsi_inquiry *)inqbuf;
2331 
2332 	if (uscsi_inquiry(cur_file, inqbuf, sizeof (inqbuf))) {
2333 		err_print("Failed\n");
2334 		return (-1);
2335 	} else {
2336 		fmt_print("Vendor:   ");
2337 		print_buf(inq->inq_vid, sizeof (inq->inq_vid));
2338 		fmt_print("\nProduct:  ");
2339 		print_buf(inq->inq_pid, sizeof (inq->inq_pid));
2340 		fmt_print("\nRevision: ");
2341 		print_buf(inq->inq_revision, sizeof (inq->inq_revision));
2342 		fmt_print("\n");
2343 	}
2344 
2345 	return (0);
2346 }
2347 
2348 
2349 /*
2350  * This routine allows the user to set the 8-character
2351  * volume name in the vtoc.  It then writes both the
2352  * primary and backup labels onto the current disk.
2353  */
2354 int
2355 c_volname()
2356 {
2357 	int	 status;
2358 	char	*prompt;
2359 	union {
2360 		int	xfoo;
2361 		char	defvolname[LEN_DKL_VVOL+1];
2362 	} x;
2363 	char    s1[MAXPATHLEN], nclean[MAXPATHLEN];
2364 	char	*volname;
2365 
2366 
2367 	/*
2368 	 * There must be a current disk type (and therefore a current disk).
2369 	 */
2370 	if (cur_dtype == NULL) {
2371 		err_print("Current Disk Type is not set.\n");
2372 		return (-1);
2373 	}
2374 	/*
2375 	 * The current disk must be formatted to label it.
2376 	 */
2377 	if (!(cur_flags & DISK_FORMATTED)) {
2378 		err_print("Current Disk is unformatted.\n");
2379 		return (-1);
2380 	}
2381 	/*
2382 	 * Check for a valid fdisk table entry for Solaris
2383 	 */
2384 	if (!good_fdisk()) {
2385 		return (-1);
2386 	}
2387 	/*
2388 	 * The current disk must be formatted to label it.
2389 	 */
2390 	if (cur_parts == NULL) {
2391 	err_print(
2392 "Please select a partition map for the disk first.\n");
2393 	return (-1);
2394 	}
2395 
2396 	/*
2397 	 * Check to see if there are any mounted file systems anywhere
2398 	 * on the current disk.  If so, refuse to label the disk, but
2399 	 * only if the partitions would change for the mounted partitions.
2400 	 *
2401 	 */
2402 	if (checkmount((diskaddr_t)-1, (diskaddr_t)-1)) {
2403 		/* Bleagh, too descriptive */
2404 		if (check_label_with_mount()) {
2405 			err_print(
2406 "Cannot label disk while it has mounted partitions.\n\n");
2407 			return (-1);
2408 		}
2409 	}
2410 
2411 	/*
2412 	 * Check to see if there are partitions being used for swapping
2413 	 * on the current disk.  If so, refuse to label the disk, but
2414 	 * only if the partitions would change for the swap partitions.
2415 	 *
2416 	 */
2417 	if (checkswap((diskaddr_t)-1, (diskaddr_t)-1)) {
2418 		/* Bleagh, too descriptive */
2419 		if (check_label_with_swap()) {
2420 			err_print(
2421 "Cannot label disk while its partitions are currently \
2422 being used for swapping.\n\n");
2423 			return (-1);
2424 		}
2425 	}
2426 
2427 	/*
2428 	 * Check to see if any partitions used for svm, vxvm, ZFS zpool
2429 	 * or live upgrade are on the disk. If so, refuse to label the
2430 	 * disk, but only if we are trying to shrink a partition in
2431 	 * use.
2432 	 */
2433 	if (checkdevinuse(cur_disk->disk_name, (diskaddr_t)-1,
2434 	    (diskaddr_t)-1, 0, 1)) {
2435 		err_print("Cannot label disk while its partitions "
2436 		    "are in use as described.\n");
2437 		return (-1);
2438 	}
2439 
2440 	/*
2441 	 * Prompt for the disk volume name.
2442 	 */
2443 	prompt = "Enter 8-character volume name (remember quotes)";
2444 	bzero(x.defvolname, LEN_DKL_VVOL+1);
2445 	bcopy(cur_disk->v_volume, x.defvolname, LEN_DKL_VVOL);
2446 	/*
2447 	 *  Get the input using "get_inputline" since
2448 	 *  input would never return null string.
2449 	 */
2450 	fmt_print("%s[\"%s\"]:", prompt, x.defvolname);
2451 
2452 	/*
2453 	 * Get input from the user.
2454 	 */
2455 	get_inputline(nclean, MAXPATHLEN);
2456 	clean_token(s1, nclean);
2457 	/*
2458 	 * check for return.
2459 	 */
2460 	if (s1[0] == 0) {
2461 		volname = x.defvolname;
2462 	} else {
2463 		/*
2464 		 * remove the " mark from volname.
2465 		 */
2466 		if (s1[0] == '"') {
2467 			int i = 1;
2468 			volname = &s1[1];
2469 			while (s1[i] != '"' && s1[i] != '\0')
2470 				i++;
2471 			s1[i] = '\0';
2472 			clean_token(nclean, volname);
2473 			volname = nclean;
2474 		} else {
2475 			(void) sscanf(&s1[0], "%1024s", nclean);
2476 			volname = nclean;
2477 		};
2478 	}
2479 	/*
2480 	 * Make sure the user is serious.
2481 	 */
2482 	if (check("Ready to label disk, continue")) {
2483 		fmt_print("\n");
2484 		return (-1);
2485 	}
2486 	/*
2487 	 * Use the volume name chosen above
2488 	 */
2489 	bzero(cur_disk->v_volume, LEN_DKL_VVOL);
2490 	bcopy(volname, cur_disk->v_volume, min((int)strlen(volname),
2491 	    LEN_DKL_VVOL));
2492 	if (cur_label == L_TYPE_EFI) {
2493 		bzero(cur_parts->etoc->efi_parts[8].p_name, LEN_DKL_VVOL);
2494 		bcopy(volname, cur_parts->etoc->efi_parts[8].p_name,
2495 		    LEN_DKL_VVOL);
2496 	}
2497 	/*
2498 	 * Write the labels out (this will also notify unix) and
2499 	 * return status.
2500 	 */
2501 	fmt_print("\n");
2502 	if (status = write_label())
2503 		err_print("Label failed.\n");
2504 	return (status);
2505 }
2506