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 2015 Nexenta Systems, Inc. All rights reserved.
23 *
24 * Copyright (c) 2011 Gary Mills
25 *
26 * Copyright (c) 1993, 2010, Oracle and/or its affiliates. All rights reserved.
27 */
28
29 /*
30 * This file contains the code to perform program startup. This
31 * includes reading the data file and the search for disks.
32 */
33 #include "global.h"
34
35 #include <ctype.h>
36 #include <stdlib.h>
37 #include <unistd.h>
38 #include <string.h>
39 #include <strings.h>
40 #include <fcntl.h>
41 #include <errno.h>
42 #include <memory.h>
43 #include <dirent.h>
44 #include <sys/fcntl.h>
45 #include <sys/param.h>
46 #include <sys/stat.h>
47
48 #include "startup.h"
49 #include "param.h"
50 #include "label.h"
51 #include "misc.h"
52 #include "menu_command.h"
53 #include "partition.h"
54 #include "ctlr_scsi.h"
55
56 #include "auto_sense.h"
57
58 extern struct ctlr_type ctlr_types[];
59 extern int nctypes;
60 extern struct ctlr_ops genericops;
61 extern long strtol();
62
63 extern int errno;
64
65 char *file_name;
66 char *option_d;
67 char *option_f;
68 char *option_l;
69 char *option_p;
70 char option_s;
71 char *option_t;
72 char *option_x;
73 char diag_msg;
74 char option_msg;
75 int need_newline;
76 int dev_expert;
77 int expert_mode;
78 uint_t cur_blksz;
79 struct ctlr_info *ctlr_list;
80 struct disk_info *disk_list;
81 struct mctlr_list *controlp;
82 char x86_devname[MAXNAMELEN];
83 FILE *data_file;
84
85 #ifdef __STDC__
86
87 /* Function prototypes for ANSI C Compilers */
88 static void usage(void);
89 static int sup_prxfile(void);
90 static void sup_setpath(void);
91 static void sup_setdtype(void);
92 static int sup_change_spec(struct disk_type *, char *);
93 static void sup_setpart(void);
94 static void search_for_logical_dev(char *devname);
95 static void add_device_to_disklist(char *devname, char *devpath);
96 static int disk_is_known(struct dk_cinfo *dkinfo);
97 static void datafile_error(char *errmsg, char *token);
98 static void search_duplicate_dtypes(void);
99 static void search_duplicate_pinfo(void);
100 static void check_dtypes_for_inconsistency(struct disk_type *dp1,
101 struct disk_type *dp2);
102 static void check_pinfo_for_inconsistency(struct partition_info *pp1,
103 struct partition_info *pp2);
104 static uint_t str2blks(char *str);
105 static int str2cyls(char *str);
106 static struct chg_list *new_chg_list(struct disk_type *);
107 static char *get_physical_name(char *);
108 static void sort_disk_list(void);
109 static int disk_name_compare(const void *, const void *);
110 static void make_controller_list(void);
111 static void check_for_duplicate_disknames(char *arglist[]);
112
113 #else /* __STDC__ */
114
115 /* Function prototypes for non-ANSI C Compilers */
116 static void usage();
117 static int sup_prxfile();
118 static void sup_setpath();
119 static void sup_setdtype();
120 static int sup_change_spec();
121 static void sup_setpart();
122 static void search_for_logical_dev();
123 static void add_device_to_disklist();
124 static int disk_is_known();
125 static void datafile_error();
126 static void search_duplicate_dtypes();
127 static void search_duplicate_pinfo();
128 static void check_dtypes_for_inconsistency();
129 static void check_pinfo_for_inconsistency();
130 static uint_t str2blks();
131 static int str2cyls();
132 static struct chg_list *new_chg_list();
133 static char *get_physical_name();
134 static void sort_disk_list();
135 static int disk_name_compare();
136 static void make_controller_list();
137 static void check_for_duplicate_disknames();
138
139 #endif /* __STDC__ */
140
141 #if defined(sparc)
142 static char *other_ctlrs[] = {
143 "ata"
144 };
145 #define OTHER_CTLRS 1
146
147 #elif defined(i386)
148 static char *other_ctlrs[] = {
149 "ISP-80"
150 };
151 #define OTHER_CTLRS 2
152
153 #else
154 #error No Platform defined.
155 #endif
156
157
158 /*
159 * This global is used to store the current line # in the data file.
160 * It must be global because the I/O routines are allowed to side
161 * effect it to keep track of backslashed newlines.
162 */
163 int data_lineno; /* current line # in data file */
164
165 /*
166 * Search path as defined in the format.dat files
167 */
168 static char **search_path = NULL;
169
170
171 static int name_represents_wholedisk(char *name);
172
173 static void get_disk_name(int fd, char *disk_name, struct disk_info *disk_info);
174
175 /*
176 * This routine digests the options on the command line. It returns
177 * the index into argv of the first string that is not an option. If
178 * there are none, it returns -1.
179 */
180 int
do_options(int argc,char * argv[])181 do_options(int argc, char *argv[])
182 {
183 char *ptr;
184 int i;
185 int next;
186
187 /*
188 * Default is no extended messages. Can be enabled manually.
189 */
190 option_msg = 0;
191 diag_msg = 0;
192 expert_mode = 0;
193 need_newline = 0;
194 dev_expert = 0;
195
196 /*
197 * Loop through the argument list, incrementing each time by
198 * an amount determined by the options found.
199 */
200 for (i = 1; i < argc; i = next) {
201 /*
202 * Start out assuming an increment of 1.
203 */
204 next = i + 1;
205 /*
206 * As soon as we hit a non-option, we're done.
207 */
208 if (*argv[i] != '-')
209 return (i);
210 /*
211 * Loop through all the characters in this option string.
212 */
213 for (ptr = argv[i] + 1; *ptr != '\0'; ptr++) {
214 /*
215 * Determine each option represented. For options
216 * that use a second string, increase the increment
217 * of the main loop so they aren't re-interpreted.
218 */
219 switch (*ptr) {
220 case 's':
221 case 'S':
222 option_s = 1;
223 break;
224 case 'f':
225 case 'F':
226 option_f = argv[next++];
227 if (next > argc)
228 goto badopt;
229 break;
230 case 'l':
231 case 'L':
232 option_l = argv[next++];
233 if (next > argc)
234 goto badopt;
235 break;
236 case 'x':
237 case 'X':
238 option_x = argv[next++];
239 if (next > argc)
240 goto badopt;
241 break;
242 case 'd':
243 case 'D':
244 option_d = argv[next++];
245 if (next > argc)
246 goto badopt;
247 break;
248 case 't':
249 case 'T':
250 option_t = argv[next++];
251 if (next > argc)
252 goto badopt;
253 break;
254 case 'p':
255 case 'P':
256 option_p = argv[next++];
257 if (next > argc)
258 goto badopt;
259 break;
260 case 'm':
261 option_msg = 1;
262 break;
263 case 'M':
264 option_msg = 1;
265 diag_msg = 1;
266 break;
267 case 'e':
268 expert_mode = 1;
269 break;
270 #ifdef DEBUG
271 case 'z':
272 dev_expert = 1;
273 break;
274 #endif
275 default:
276 badopt:
277 usage();
278 break;
279 }
280 }
281 }
282 /*
283 * All the command line strings were options. Return that fact.
284 */
285 return (-1);
286 }
287
288
289 static void
usage()290 usage()
291 {
292 err_print("Usage: format [-s][-d disk_name]");
293 err_print("[-t disk_type][-p partition_name]\n");
294 err_print("\t[-f cmd_file][-l log_file]");
295 err_print("[-x data_file] [-m] [-M] [-e] disk_list\n");
296 fullabort();
297 }
298
299
300 /*
301 * This routine reads in and digests the data file. The data file contains
302 * definitions for the search path, known disk types, and known partition
303 * maps.
304 *
305 * Note: for each file being processed, file_name is a pointer to that
306 * file's name. We are careful to make sure that file_name points to
307 * globally-accessible data, not data on the stack, because each
308 * disk/partition/controller definition now keeps a pointer to the
309 * filename in which it was defined. In the case of duplicate,
310 * conflicting definitions, we can thus tell the user exactly where
311 * the problem is occurring.
312 */
313 void
sup_init()314 sup_init()
315 {
316 int nopened_files = 0;
317 char fname[MAXPATHLEN];
318 char *path;
319 char *p;
320 struct stat stbuf;
321
322
323 /*
324 * Create a singly-linked list of controller types so that we may
325 * dynamically add unknown controllers to this for 3'rd
326 * party disk support.
327 */
328
329 make_controller_list();
330
331 /*
332 * If a data file was specified on the command line, use it first
333 * If the file cannot be opened, fail. We want to guarantee
334 * that, if the user explicitly names a file, they can
335 * access it.
336 *
337 * option_x is already global, no need to dup it on the heap.
338 */
339 if (option_x) {
340 file_name = option_x;
341 if (sup_prxfile()) {
342 nopened_files++;
343 } else {
344 err_print("Unable to open data file '%s' - %s.\n",
345 file_name, strerror(errno));
346 fullabort();
347 }
348 }
349
350 /*
351 * Now look for an environment variable FORMAT_PATH.
352 * If found, we use it as a colon-separated list
353 * of directories. If no such environment variable
354 * is defined, use a default path of "/etc".
355 */
356 path = getenv("FORMAT_PATH");
357 if (path == NULL) {
358 path = "/etc";
359 }
360 /*
361 * Traverse the path one file at a time. Pick off
362 * the file name, and append the name "format.dat"
363 * at the end of the pathname.
364 * Whatever string we construct, duplicate it on the
365 * heap, so that file_name is globally accessible.
366 */
367 while (*path != 0) {
368 p = fname;
369 while (*path != 0 && *path != ':')
370 *p++ = *path++;
371 if (p == fname)
372 continue;
373 *p = 0;
374 if (*path == ':')
375 path++;
376 /*
377 * If the path we have so far is a directory,
378 * look for a format.dat file in that directory,
379 * otherwise try using the path name specified.
380 * This permits arbitrary file names in the
381 * path specification, if this proves useful.
382 */
383 if (stat(fname, &stbuf) == -1) {
384 err_print("Unable to access '%s' - %s.\n",
385 fname, strerror(errno));
386 } else {
387 if (S_ISDIR(stbuf.st_mode)) {
388 if (*(p-1) != '/')
389 *p++ = '/';
390 (void) strcpy(p, "format.dat");
391 }
392 file_name = alloc_string(fname);
393 if (sup_prxfile()) {
394 nopened_files++;
395 }
396 }
397 }
398
399 /*
400 * Check for duplicate disk or partitions definitions
401 * that are inconsistent - this would be very confusing.
402 */
403 search_duplicate_dtypes();
404 search_duplicate_pinfo();
405 }
406
407
408 /*
409 * Open and process a format data file. Unfortunately, we use
410 * globals: file_name for the file name, and data_file
411 * for the descriptor. Return true if able to open the file.
412 */
413 static int
sup_prxfile()414 sup_prxfile()
415 {
416 int status;
417 TOKEN token;
418 TOKEN cleaned;
419
420 /*
421 * Open the data file. Return 0 if unable to do so.
422 */
423 data_file = fopen(file_name, "r");
424 if (data_file == NULL) {
425 return (0);
426 }
427 /*
428 * Step through the data file a meta-line at a time. There are
429 * typically several backslashed newlines in each meta-line,
430 * so data_lineno will be getting side effected along the way.
431 */
432 data_lineno = 0;
433 for (;;) {
434 data_lineno++;
435 /*
436 * Get the keyword.
437 */
438 status = sup_gettoken(token);
439 /*
440 * If we hit the end of the data file, we're done.
441 */
442 if (status == SUP_EOF)
443 break;
444 /*
445 * If the line is blank, skip it.
446 */
447 if (status == SUP_EOL)
448 continue;
449 /*
450 * If the line starts with some key character, it's an error.
451 */
452 if (status != SUP_STRING) {
453 datafile_error("Expecting keyword, found '%s'", token);
454 continue;
455 }
456 /*
457 * Clean up the token and see which keyword it is. Call
458 * the appropriate routine to process the rest of the line.
459 */
460 clean_token(cleaned, token);
461 if (strcmp(cleaned, "search_path") == 0)
462 sup_setpath();
463 else if (strcmp(cleaned, "disk_type") == 0)
464 sup_setdtype();
465 else if (strcmp(cleaned, "partition") == 0)
466 sup_setpart();
467 else {
468 datafile_error("Unknown keyword '%s'", cleaned);
469 }
470 }
471 /*
472 * Close the data file.
473 */
474 (void) fclose(data_file);
475
476 return (1);
477 }
478
479 /*
480 * This routine processes a 'search_path' line in the data file. The
481 * search path is a list of disk names that will be searched for by the
482 * program.
483 *
484 * The static path_size and path_alloc are used to build up the
485 * list of files comprising the search path. The static definitions
486 * enable supporting multiple search path definitions.
487 */
488 static void
sup_setpath()489 sup_setpath()
490 {
491 TOKEN token;
492 TOKEN cleaned;
493 int status;
494 static int path_size;
495 static int path_alloc;
496
497 /*
498 * Pull in some grammar.
499 */
500 status = sup_gettoken(token);
501 if (status != SUP_EQL) {
502 datafile_error("Expecting '=', found '%s'", token);
503 return;
504 }
505 /*
506 * Loop through the entries.
507 */
508 for (;;) {
509 /*
510 * Pull in the disk name.
511 */
512 status = sup_gettoken(token);
513 /*
514 * If we hit end of line, we're done.
515 */
516 if (status == SUP_EOL)
517 break;
518 /*
519 * If we hit some key character, it's an error.
520 */
521 if (status != SUP_STRING) {
522 datafile_error("Expecting value, found '%s'", token);
523 break;
524 }
525 clean_token(cleaned, token);
526 /*
527 * Build the string into an argvlist. This array
528 * is dynamically sized, as necessary, and terminated
529 * with a null. Each name is alloc'ed on the heap,
530 * so no dangling references.
531 */
532 search_path = build_argvlist(search_path, &path_size,
533 &path_alloc, cleaned);
534 /*
535 * Pull in some grammar.
536 */
537 status = sup_gettoken(token);
538 if (status == SUP_EOL)
539 break;
540 if (status != SUP_COMMA) {
541 datafile_error("Expecting ', ', found '%s'", token);
542 break;
543 }
544 }
545 }
546
547 /*
548 * This routine processes a 'disk_type' line in the data file. It defines
549 * the physical attributes of a brand of disk when connected to a specific
550 * controller type.
551 */
552 static void
sup_setdtype()553 sup_setdtype()
554 {
555 TOKEN token, cleaned, ident;
556 int val, status, i;
557 ulong_t flags = 0;
558 struct disk_type *dtype, *type;
559 struct ctlr_type *ctype;
560 char *dtype_name, *ptr;
561 struct mctlr_list *mlp;
562
563 /*
564 * Pull in some grammar.
565 */
566 status = sup_gettoken(token);
567 if (status != SUP_EQL) {
568 datafile_error("Expecting '=', found '%s'", token);
569 return;
570 }
571 /*
572 * Pull in the name of the disk type.
573 */
574 status = sup_gettoken(token);
575 if (status != SUP_STRING) {
576 datafile_error("Expecting value, found '%s'", token);
577 return;
578 }
579 clean_token(cleaned, token);
580 /*
581 * Allocate space for the disk type and copy in the name.
582 */
583 dtype_name = (char *)zalloc(strlen(cleaned) + 1);
584 (void) strcpy(dtype_name, cleaned);
585 dtype = (struct disk_type *)zalloc(sizeof (struct disk_type));
586 dtype->dtype_asciilabel = dtype_name;
587 /*
588 * Save the filename/linenumber where this disk was defined
589 */
590 dtype->dtype_filename = file_name;
591 dtype->dtype_lineno = data_lineno;
592 /*
593 * Loop for each attribute.
594 */
595 for (;;) {
596 /*
597 * Pull in some grammar.
598 */
599 status = sup_gettoken(token);
600 /*
601 * If we hit end of line, we're done.
602 */
603 if (status == SUP_EOL)
604 break;
605 if (status != SUP_COLON) {
606 datafile_error("Expecting ':', found '%s'", token);
607 return;
608 }
609 /*
610 * Pull in the attribute.
611 */
612 status = sup_gettoken(token);
613 /*
614 * If we hit end of line, we're done.
615 */
616 if (status == SUP_EOL)
617 break;
618 /*
619 * If we hit a key character, it's an error.
620 */
621 if (status != SUP_STRING) {
622 datafile_error("Expecting keyword, found '%s'", token);
623 return;
624 }
625 clean_token(ident, token);
626 /*
627 * Check to see if we've got a change specification
628 * If so, this routine will parse the entire
629 * specification, so just restart at top of loop
630 */
631 if (sup_change_spec(dtype, ident)) {
632 continue;
633 }
634 /*
635 * Pull in more grammar.
636 */
637 status = sup_gettoken(token);
638 if (status != SUP_EQL) {
639 datafile_error("Expecting '=', found '%s'", token);
640 return;
641 }
642 /*
643 * Pull in the value of the attribute.
644 */
645 status = sup_gettoken(token);
646 if (status != SUP_STRING) {
647 datafile_error("Expecting value, found '%s'", token);
648 return;
649 }
650 clean_token(cleaned, token);
651 /*
652 * If the attribute defined the ctlr...
653 */
654 if (strcmp(ident, "ctlr") == 0) {
655 /*
656 * Match the value with a ctlr type.
657 */
658 mlp = controlp;
659
660 while (mlp != NULL) {
661 if (strcmp(mlp->ctlr_type->ctype_name,
662 cleaned) == 0)
663 break;
664 mlp = mlp->next;
665 }
666 /*
667 * If we couldn't match it, it's an error.
668 */
669 if (mlp == NULL) {
670 for (i = 0; i < OTHER_CTLRS; i++) {
671 if (strcmp(other_ctlrs[i], cleaned)
672 == 0) {
673 datafile_error(NULL, NULL);
674 return;
675 }
676 }
677 if (i == OTHER_CTLRS) {
678 datafile_error(
679 "Unknown controller '%s'",
680 cleaned);
681 return;
682 }
683 }
684 /*
685 * Found a match. Add this disk type to the list
686 * for the ctlr type if we can complete the
687 * disk specification correctly.
688 */
689 ctype = mlp->ctlr_type;
690 flags |= SUP_CTLR;
691 continue;
692 }
693 /*
694 * All other attributes require a numeric value. Convert
695 * the value to a number.
696 */
697 val = (int)strtol(cleaned, &ptr, 0);
698 if (*ptr != '\0') {
699 datafile_error("Expecting an integer, found '%s'",
700 cleaned);
701 return;
702 }
703 /*
704 * Figure out which attribute it was and fill in the
705 * appropriate value. Also note that the attribute
706 * has been defined.
707 */
708 if (strcmp(ident, "ncyl") == 0) {
709 dtype->dtype_ncyl = val;
710 flags |= SUP_NCYL;
711 } else if (strcmp(ident, "acyl") == 0) {
712 dtype->dtype_acyl = val;
713 flags |= SUP_ACYL;
714 } else if (strcmp(ident, "pcyl") == 0) {
715 dtype->dtype_pcyl = val;
716 flags |= SUP_PCYL;
717 } else if (strcmp(ident, "nhead") == 0) {
718 dtype->dtype_nhead = val;
719 flags |= SUP_NHEAD;
720 } else if (strcmp(ident, "nsect") == 0) {
721 dtype->dtype_nsect = val;
722 flags |= SUP_NSECT;
723 } else if (strcmp(ident, "rpm") == 0) {
724 dtype->dtype_rpm = val;
725 flags |= SUP_RPM;
726 } else if (strcmp(ident, "bpt") == 0) {
727 dtype->dtype_bpt = val;
728 flags |= SUP_BPT;
729 } else if (strcmp(ident, "bps") == 0) {
730 dtype->dtype_bps = val;
731 flags |= SUP_BPS;
732 } else if (strcmp(ident, "drive_type") == 0) {
733 dtype->dtype_dr_type = val;
734 flags |= SUP_DRTYPE;
735 } else if (strcmp(ident, "cache") == 0) {
736 dtype->dtype_cache = val;
737 flags |= SUP_CACHE;
738 } else if (strcmp(ident, "prefetch") == 0) {
739 dtype->dtype_threshold = val;
740 flags |= SUP_PREFETCH;
741 } else if (strcmp(ident, "read_retries") == 0) {
742 dtype->dtype_read_retries = val;
743 flags |= SUP_READ_RETRIES;
744 } else if (strcmp(ident, "write_retries") == 0) {
745 dtype->dtype_write_retries = val;
746 flags |= SUP_WRITE_RETRIES;
747 } else if (strcmp(ident, "min_prefetch") == 0) {
748 dtype->dtype_prefetch_min = val;
749 flags |= SUP_CACHE_MIN;
750 } else if (strcmp(ident, "max_prefetch") == 0) {
751 dtype->dtype_prefetch_max = val;
752 flags |= SUP_CACHE_MAX;
753 } else if (strcmp(ident, "trks_zone") == 0) {
754 dtype->dtype_trks_zone = val;
755 flags |= SUP_TRKS_ZONE;
756 } else if (strcmp(ident, "atrks") == 0) {
757 dtype->dtype_atrks = val;
758 flags |= SUP_ATRKS;
759 } else if (strcmp(ident, "asect") == 0) {
760 dtype->dtype_asect = val;
761 flags |= SUP_ASECT;
762 } else if (strcmp(ident, "psect") == 0) {
763 dtype->dtype_psect = val;
764 flags |= SUP_PSECT;
765 } else if (strcmp(ident, "phead") == 0) {
766 dtype->dtype_phead = val;
767 flags |= SUP_PHEAD;
768 } else if (strcmp(ident, "fmt_time") == 0) {
769 dtype->dtype_fmt_time = val;
770 flags |= SUP_FMTTIME;
771 } else if (strcmp(ident, "cyl_skew") == 0) {
772 dtype->dtype_cyl_skew = val;
773 flags |= SUP_CYLSKEW;
774 } else if (strcmp(ident, "trk_skew") == 0) {
775 dtype->dtype_trk_skew = val;
776 flags |= SUP_TRKSKEW;
777 } else {
778 datafile_error("Unknown keyword '%s'", ident);
779 }
780 }
781 /*
782 * Check to be sure all the necessary attributes have been defined.
783 * If any are missing, it's an error. Also, log options for later
784 * use by specific driver.
785 */
786 dtype->dtype_options = flags;
787 if ((flags & SUP_MIN_DRIVE) != SUP_MIN_DRIVE) {
788 datafile_error("Incomplete specification", "");
789 return;
790 }
791 if ((!(ctype->ctype_flags & CF_SCSI)) && (!(flags & SUP_BPT)) &&
792 (!(ctype->ctype_flags & CF_NOFORMAT))) {
793 datafile_error("Incomplete specification", "");
794 return;
795 }
796 if ((ctype->ctype_flags & CF_SMD_DEFS) && (!(flags & SUP_BPS))) {
797 datafile_error("Incomplete specification", "");
798 return;
799 }
800 /*
801 * Add this disk type to the list for the ctlr type
802 */
803 assert(flags & SUP_CTLR);
804 type = ctype->ctype_dlist;
805 if (type == NULL) {
806 ctype->ctype_dlist = dtype;
807 } else {
808 while (type->dtype_next != NULL)
809 type = type->dtype_next;
810 type->dtype_next = dtype;
811 }
812 }
813
814
815 /*
816 * Parse a SCSI mode page change specification.
817 *
818 * Return:
819 * 0: not change specification, continue parsing
820 * 1: was change specification, it was ok,
821 * or we already handled the error.
822 */
823 static int
sup_change_spec(struct disk_type * disk,char * id)824 sup_change_spec(struct disk_type *disk, char *id)
825 {
826 char *p;
827 char *p2;
828 int pageno;
829 int byteno;
830 int mode;
831 int value;
832 TOKEN token;
833 TOKEN ident;
834 struct chg_list *cp;
835 int tilde;
836 int i;
837
838 /*
839 * Syntax: p[<nn>|0x<xx>]
840 */
841 if (*id != 'p') {
842 return (0);
843 }
844 pageno = (int)strtol(id+1, &p2, 0);
845 if (*p2 != 0) {
846 return (0);
847 }
848 /*
849 * Once we get this far, we know we have the
850 * beginnings of a change specification.
851 * If there's a problem now, report the problem,
852 * and return 1, so that the caller can restart
853 * parsing at the next expression.
854 */
855 if (!scsi_supported_page(pageno)) {
856 datafile_error("Unsupported mode page '%s'", id);
857 return (1);
858 }
859 /*
860 * Next token should be the byte offset
861 */
862 if (sup_gettoken(token) != SUP_STRING) {
863 datafile_error("Unexpected value '%s'", token);
864 return (1);
865 }
866 clean_token(ident, token);
867
868 /*
869 * Syntax: b[<nn>|0x<xx>]
870 */
871 p = ident;
872 if (*p++ != 'b') {
873 datafile_error("Unknown keyword '%s'", ident);
874 return (1);
875 }
876 byteno = (int)strtol(p, &p2, 10);
877 if (*p2 != 0) {
878 datafile_error("Unknown keyword '%s'", ident);
879 return (1);
880 }
881 if (byteno == 0 || byteno == 1) {
882 datafile_error("Unsupported byte offset '%s'", ident);
883 return (1);
884 }
885
886 /*
887 * Get the operator for this expression
888 */
889 mode = CHG_MODE_UNDEFINED;
890 switch (sup_gettoken(token)) {
891 case SUP_EQL:
892 mode = CHG_MODE_ABS;
893 break;
894 case SUP_OR:
895 if (sup_gettoken(token) == SUP_EQL)
896 mode = CHG_MODE_SET;
897 break;
898 case SUP_AND:
899 if (sup_gettoken(token) == SUP_EQL)
900 mode = CHG_MODE_CLR;
901 break;
902 }
903 if (mode == CHG_MODE_UNDEFINED) {
904 datafile_error("Unexpected operator: '%s'", token);
905 return (1);
906 }
907
908 /*
909 * Get right-hand of expression - accept optional tilde
910 */
911 tilde = 0;
912 if ((i = sup_gettoken(token)) == SUP_TILDE) {
913 tilde = 1;
914 i = sup_gettoken(token);
915 }
916 if (i != SUP_STRING) {
917 datafile_error("Expecting value, found '%s'", token);
918 return (1);
919 }
920 clean_token(ident, token);
921 value = (int)strtol(ident, &p, 0);
922 if (*p != 0) {
923 datafile_error("Expecting value, found '%s'", token);
924 return (1);
925 }
926
927 /*
928 * Apply the tilde operator, if found.
929 * Constrain to a byte value.
930 */
931 if (tilde) {
932 value = ~value;
933 }
934 value &= 0xff;
935
936 /*
937 * We parsed a successful change specification expression.
938 * Add it to the list for this disk type.
939 */
940 cp = new_chg_list(disk);
941 cp->pageno = pageno;
942 cp->byteno = byteno;
943 cp->mode = mode;
944 cp->value = value;
945 return (1);
946 }
947
948
949 /*
950 * This routine processes a 'partition' line in the data file. It defines
951 * a known partition map for a particular disk type on a particular
952 * controller type.
953 */
954 static void
sup_setpart()955 sup_setpart()
956 {
957 TOKEN token, cleaned, disk, ctlr, ident;
958 struct disk_type *dtype = NULL;
959 struct ctlr_type *ctype = NULL;
960 struct partition_info *pinfo, *parts;
961 char *pinfo_name;
962 int i, index, status, flags = 0;
963 uint_t val1, val2;
964 ushort_t vtoc_tag;
965 ushort_t vtoc_flag;
966 struct mctlr_list *mlp;
967
968 /*
969 * Pull in some grammar.
970 */
971 status = sup_gettoken(token);
972 if (status != SUP_EQL) {
973 datafile_error("Expecting '=', found '%s'", token);
974 return;
975 }
976 /*
977 * Pull in the name of the map.
978 */
979 status = sup_gettoken(token);
980 if (status != SUP_STRING) {
981 datafile_error("Expecting value, found '%s'", token);
982 return;
983 }
984 clean_token(cleaned, token);
985 /*
986 * Allocate space for the partition map and fill in the name.
987 */
988 pinfo_name = (char *)zalloc(strlen(cleaned) + 1);
989 (void) strcpy(pinfo_name, cleaned);
990 pinfo = (struct partition_info *)zalloc(sizeof (struct partition_info));
991 pinfo->pinfo_name = pinfo_name;
992 /*
993 * Save the filename/linenumber where this partition was defined
994 */
995 pinfo->pinfo_filename = file_name;
996 pinfo->pinfo_lineno = data_lineno;
997
998 /*
999 * Install default vtoc information into the new partition table
1000 */
1001 set_vtoc_defaults(pinfo);
1002
1003 /*
1004 * Loop for each attribute in the line.
1005 */
1006 for (;;) {
1007 /*
1008 * Pull in some grammar.
1009 */
1010 status = sup_gettoken(token);
1011 /*
1012 * If we hit end of line, we're done.
1013 */
1014 if (status == SUP_EOL)
1015 break;
1016 if (status != SUP_COLON) {
1017 datafile_error("Expecting ':', found '%s'", token);
1018 return;
1019 }
1020 /*
1021 * Pull in the attribute.
1022 */
1023 status = sup_gettoken(token);
1024 /*
1025 * If we hit end of line, we're done.
1026 */
1027 if (status == SUP_EOL)
1028 break;
1029 if (status != SUP_STRING) {
1030 datafile_error("Expecting keyword, found '%s'", token);
1031 return;
1032 }
1033 clean_token(ident, token);
1034 /*
1035 * Pull in more grammar.
1036 */
1037 status = sup_gettoken(token);
1038 if (status != SUP_EQL) {
1039 datafile_error("Expecting '=', found '%s'", token);
1040 return;
1041 }
1042 /*
1043 * Pull in the value of the attribute.
1044 */
1045 status = sup_gettoken(token);
1046 /*
1047 * If we hit a key character, it's an error.
1048 */
1049 if (status != SUP_STRING) {
1050 datafile_error("Expecting value, found '%s'", token);
1051 return;
1052 }
1053 clean_token(cleaned, token);
1054 /*
1055 * If the attribute is the ctlr, save the ctlr name and
1056 * mark it defined.
1057 */
1058 if (strcmp(ident, "ctlr") == 0) {
1059 (void) strcpy(ctlr, cleaned);
1060 flags |= SUP_CTLR;
1061 continue;
1062 /*
1063 * If the attribute is the disk, save the disk name and
1064 * mark it defined.
1065 */
1066 } else if (strcmp(ident, "disk") == 0) {
1067 (void) strcpy(disk, cleaned);
1068 flags |= SUP_DISK;
1069 continue;
1070 }
1071 /*
1072 * If we now know both the controller name and the
1073 * disk name, let's see if we can find the controller
1074 * and disk type. This will give us the geometry,
1075 * which can permit us to accept partitions specs
1076 * in cylinders or blocks.
1077 */
1078 if (((flags & (SUP_DISK|SUP_CTLR)) == (SUP_DISK|SUP_CTLR)) &&
1079 dtype == NULL && ctype == NULL) {
1080 /*
1081 * Attempt to match the specified ctlr to a known type.
1082 */
1083 mlp = controlp;
1084
1085 while (mlp != NULL) {
1086 if (strcmp(mlp->ctlr_type->ctype_name,
1087 ctlr) == 0)
1088 break;
1089 mlp = mlp->next;
1090 }
1091 /*
1092 * If no match is found, it's an error.
1093 */
1094 if (mlp == NULL) {
1095 for (i = 0; i < OTHER_CTLRS; i++) {
1096 if (strcmp(other_ctlrs[i], ctlr) == 0) {
1097 datafile_error(NULL, NULL);
1098 return;
1099 }
1100 }
1101 if (i == OTHER_CTLRS) {
1102 datafile_error(
1103 "Unknown controller '%s'", ctlr);
1104 return;
1105 }
1106 }
1107 ctype = mlp->ctlr_type;
1108 /*
1109 * Attempt to match the specified disk to a known type.
1110 */
1111 for (dtype = ctype->ctype_dlist; dtype != NULL;
1112 dtype = dtype->dtype_next) {
1113 if (strcmp(dtype->dtype_asciilabel, disk) == 0)
1114 break;
1115 }
1116 /*
1117 * If no match is found, it's an error.
1118 */
1119 if (dtype == NULL) {
1120 datafile_error("Unknown disk '%s'", disk);
1121 return;
1122 }
1123 /*
1124 * Now that we know the disk type, set up the
1125 * globals that let that magic macro "spc()"
1126 * do it's thing. Sorry that this is glued
1127 * together so poorly...
1128 */
1129 nhead = dtype->dtype_nhead;
1130 nsect = dtype->dtype_nsect;
1131 acyl = dtype->dtype_acyl;
1132 ncyl = dtype->dtype_ncyl;
1133 }
1134 /*
1135 * By now, the disk and controller type must be defined
1136 */
1137 if (dtype == NULL || ctype == NULL) {
1138 datafile_error("Incomplete specification", "");
1139 return;
1140 }
1141 /*
1142 * The rest of the attributes are all single letters.
1143 * Make sure the specified attribute is a single letter.
1144 */
1145 if (strlen(ident) != 1) {
1146 datafile_error("Unknown keyword '%s'", ident);
1147 return;
1148 }
1149 /*
1150 * Also make sure it is within the legal range of letters.
1151 */
1152 if (ident[0] < PARTITION_BASE || ident[0] > PARTITION_BASE+9) {
1153 datafile_error("Unknown keyword '%s'", ident);
1154 return;
1155 }
1156 /*
1157 * Here's the index of the partition we're dealing with
1158 */
1159 index = ident[0] - PARTITION_BASE;
1160 /*
1161 * For SunOS 5.0, we support the additional syntax:
1162 * [<tag>, ] [<flag>, ] <start>, <end>
1163 * instead of:
1164 * <start>, <end>
1165 *
1166 * <tag> may be one of: boot, root, swap, etc.
1167 * <flag> consists of two characters:
1168 * W (writable) or R (read-only)
1169 * M (mountable) or U (unmountable)
1170 *
1171 * Start with the defaults assigned above:
1172 */
1173 vtoc_tag = pinfo->vtoc.v_part[index].p_tag;
1174 vtoc_flag = pinfo->vtoc.v_part[index].p_flag;
1175
1176 /*
1177 * First try to match token against possible tag values
1178 */
1179 if (find_value(ptag_choices, cleaned, &i) == 1) {
1180 /*
1181 * Found valid tag. Use it and advance parser
1182 */
1183 vtoc_tag = (ushort_t)i;
1184 status = sup_gettoken(token);
1185 if (status != SUP_COMMA) {
1186 datafile_error(
1187 "Expecting ', ', found '%s'", token);
1188 return;
1189 }
1190 status = sup_gettoken(token);
1191 if (status != SUP_STRING) {
1192 datafile_error("Expecting value, found '%s'",
1193 token);
1194 return;
1195 }
1196 clean_token(cleaned, token);
1197 }
1198
1199 /*
1200 * Try to match token against possible flag values
1201 */
1202 if (find_value(pflag_choices, cleaned, &i) == 1) {
1203 /*
1204 * Found valid flag. Use it and advance parser
1205 */
1206 vtoc_flag = (ushort_t)i;
1207 status = sup_gettoken(token);
1208 if (status != SUP_COMMA) {
1209 datafile_error("Expecting ', ', found '%s'",
1210 token);
1211 return;
1212 }
1213 status = sup_gettoken(token);
1214 if (status != SUP_STRING) {
1215 datafile_error("Expecting value, found '%s'",
1216 token);
1217 return;
1218 }
1219 clean_token(cleaned, token);
1220 }
1221 /*
1222 * All other attributes have a pair of numeric values.
1223 * Convert the first value to a number. This value
1224 * is the starting cylinder number of the partition.
1225 */
1226 val1 = str2cyls(cleaned);
1227 if (val1 == (uint_t)(-1)) {
1228 datafile_error("Expecting an integer, found '%s'",
1229 cleaned);
1230 return;
1231 }
1232 /*
1233 * Pull in some grammar.
1234 */
1235 status = sup_gettoken(token);
1236 if (status != SUP_COMMA) {
1237 datafile_error("Expecting ', ', found '%s'", token);
1238 return;
1239 }
1240 /*
1241 * Pull in the second value.
1242 */
1243 status = sup_gettoken(token);
1244 if (status != SUP_STRING) {
1245 datafile_error("Expecting value, found '%s'", token);
1246 return;
1247 }
1248 clean_token(cleaned, token);
1249 /*
1250 * Convert the second value to a number. This value
1251 * is the number of blocks composing the partition.
1252 * If the token is terminated with a 'c', the units
1253 * are cylinders, not blocks. Also accept a 'b', if
1254 * they choose to be so specific.
1255 */
1256 val2 = str2blks(cleaned);
1257 if (val2 == (uint_t)(-1)) {
1258 datafile_error("Expecting an integer, found '%s'",
1259 cleaned);
1260 return;
1261 }
1262 /*
1263 * Fill in the appropriate map entry with the values.
1264 */
1265 pinfo->pinfo_map[index].dkl_cylno = val1;
1266 pinfo->pinfo_map[index].dkl_nblk = val2;
1267 pinfo->vtoc.v_part[index].p_tag = vtoc_tag;
1268 pinfo->vtoc.v_part[index].p_flag = vtoc_flag;
1269
1270 #if defined(_SUNOS_VTOC_16)
1271 pinfo->vtoc.v_part[index].p_start = val1 * (nhead * nsect);
1272 pinfo->vtoc.v_part[index].p_size = val2;
1273
1274 if (val2 == 0) {
1275 pinfo->vtoc.v_part[index].p_tag = 0;
1276 pinfo->vtoc.v_part[index].p_flag = 0;
1277 pinfo->vtoc.v_part[index].p_start = 0;
1278 pinfo->pinfo_map[index].dkl_cylno = 0;
1279 }
1280 #endif /* defined(_SUNOS_VTOC_16) */
1281
1282 }
1283 /*
1284 * Check to be sure that all necessary attributes were defined.
1285 */
1286 if ((flags & SUP_MIN_PART) != SUP_MIN_PART) {
1287 datafile_error("Incomplete specification", "");
1288 return;
1289 }
1290 /*
1291 * Add this partition map to the list of known maps for the
1292 * specified disk/ctlr.
1293 */
1294 parts = dtype->dtype_plist;
1295 if (parts == NULL)
1296 dtype->dtype_plist = pinfo;
1297 else {
1298 while (parts->pinfo_next != NULL)
1299 parts = parts->pinfo_next;
1300 parts->pinfo_next = pinfo;
1301 }
1302 }
1303
1304 /*
1305 * Open the disk device - just a wrapper for open.
1306 */
1307 int
open_disk(char * diskname,int flags)1308 open_disk(char *diskname, int flags)
1309 {
1310 return (open(diskname, flags));
1311 }
1312
1313 /*
1314 * This routine performs the disk search during startup. It looks for
1315 * all the disks in the search path, and creates a list of those that
1316 * are found.
1317 */
1318 void
do_search(char * arglist[])1319 do_search(char *arglist[])
1320 {
1321 char **sp;
1322 DIR *dir;
1323 struct dirent *dp;
1324 char s[MAXPATHLEN];
1325 char path[MAXPATHLEN];
1326 char curdir[MAXPATHLEN];
1327 char *directory = "/dev/rdsk";
1328 struct disk_info *disk;
1329 int i;
1330
1331 /*
1332 * Change directory to the device directory. This
1333 * gives us the most efficient access to that directory.
1334 * Remember where we were, and return there when finished.
1335 */
1336 if (getcwd(curdir, sizeof (curdir)) == NULL) {
1337 err_print("Cannot get current directory - %s\n",
1338 strerror(errno));
1339 fullabort();
1340 }
1341 if (chdir(directory) == -1) {
1342 err_print("Cannot set directory to %s - %s\n",
1343 directory, strerror(errno));
1344 fullabort();
1345 }
1346
1347 /*
1348 * If there were disks specified on the command line,
1349 * use those disks, and nothing but those disks.
1350 */
1351 if (arglist != NULL) {
1352 check_for_duplicate_disknames(arglist);
1353 for (; *arglist != NULL; arglist++) {
1354 search_for_logical_dev(*arglist);
1355 }
1356 } else {
1357 /*
1358 * If there were no disks specified on the command line,
1359 * search for all disks attached to the system.
1360 */
1361 fmt_print("Searching for disks...");
1362 (void) fflush(stdout);
1363 need_newline = 1;
1364
1365 /*
1366 * Find all disks specified in search_path definitions
1367 * in whatever format.dat files were processed.
1368 */
1369 sp = search_path;
1370 if (sp != NULL) {
1371 while (*sp != NULL) {
1372 search_for_logical_dev(*sp++);
1373 }
1374 }
1375
1376 /*
1377 * Open the device directory
1378 */
1379 if ((dir = opendir(".")) == NULL) {
1380 err_print("Cannot open %s - %s\n",
1381 directory, strerror(errno));
1382 fullabort();
1383 }
1384
1385 /*
1386 * Now find all usable nodes in /dev/rdsk (or /dev, if 4.x)
1387 * First find all nodes which do not conform to
1388 * standard disk naming conventions. This permits
1389 * all user-defined names to override the default names.
1390 */
1391 while ((dp = readdir(dir)) != NULL) {
1392 if (strcmp(dp->d_name, ".") == 0 ||
1393 strcmp(dp->d_name, "..") == 0)
1394 continue;
1395 if (!conventional_name(dp->d_name)) {
1396 if (!fdisk_physical_name(dp->d_name)) {
1397 /*
1398 * If non-conventional name represents
1399 * a link to non-s2 slice , ignore it.
1400 */
1401 if (!name_represents_wholedisk
1402 (dp->d_name)) {
1403 (void) strcpy(path, directory);
1404 (void) strcat(path, "/");
1405 (void) strcat(path, dp->d_name);
1406 add_device_to_disklist(
1407 dp->d_name, path);
1408 }
1409 }
1410 }
1411 }
1412 rewinddir(dir);
1413
1414
1415 /*
1416 * Now find all nodes corresponding to the standard
1417 * device naming conventions.
1418 */
1419 while ((dp = readdir(dir)) != NULL) {
1420 if (strcmp(dp->d_name, ".") == 0 ||
1421 strcmp(dp->d_name, "..") == 0)
1422 continue;
1423 if (whole_disk_name(dp->d_name)) {
1424 (void) strcpy(path, directory);
1425 (void) strcat(path, "/");
1426 (void) strcat(path, dp->d_name);
1427 canonicalize_name(s, dp->d_name);
1428 add_device_to_disklist(s, path);
1429 }
1430 }
1431 /*
1432 * Close the directory
1433 */
1434 if (closedir(dir) == -1) {
1435 err_print("Cannot close directory %s - %s\n",
1436 directory, strerror(errno));
1437 fullabort();
1438 }
1439
1440 need_newline = 0;
1441 fmt_print("done\n");
1442 }
1443
1444 /*
1445 * Return to whence we came
1446 */
1447 if (chdir(curdir) == -1) {
1448 err_print("Cannot set directory to %s - %s\n",
1449 curdir, strerror(errno));
1450 fullabort();
1451 }
1452
1453 /*
1454 * If we didn't find any disks, give up.
1455 */
1456 if (disk_list == NULL) {
1457 if (geteuid() == 0) {
1458 err_print("No disks found!\n");
1459 } else {
1460 err_print("No permission (or no disks found)!\n");
1461 }
1462 (void) fflush(stdout);
1463 fullabort();
1464 }
1465
1466 sort_disk_list();
1467
1468 /*
1469 * Tell user the results of the auto-configure process
1470 */
1471 i = 0;
1472 for (disk = disk_list; disk != NULL; disk = disk->disk_next) {
1473 float scaled;
1474 diskaddr_t nblks;
1475 struct disk_type *type;
1476 if (disk->disk_flags & DSK_AUTO_CONFIG) {
1477 if (i++ == 0) {
1478 fmt_print("\n");
1479 }
1480 fmt_print("%s: ", disk->disk_name);
1481 if (disk->disk_flags & DSK_LABEL_DIRTY) {
1482 fmt_print("configured ");
1483 } else {
1484 fmt_print("configured and labeled ");
1485 }
1486 type = disk->disk_type;
1487 nblks = type->dtype_ncyl * type->dtype_nhead *
1488 type->dtype_nsect;
1489 if (disk->label_type == L_TYPE_SOLARIS)
1490 scaled = bn2mb(nblks);
1491 else
1492 scaled = bn2mb(type->capacity);
1493 fmt_print("with capacity of ");
1494 if (scaled > 1024.0) {
1495 fmt_print("%1.2fGB\n", scaled/1024.0);
1496 } else {
1497 fmt_print("%1.2fMB\n", scaled);
1498 }
1499 }
1500 }
1501 }
1502
1503
1504 /*
1505 * For a given "logical" disk name as specified in a format.dat
1506 * search path, try to find the device it actually refers to.
1507 * Since we are trying to maintain 4.x naming convention
1508 * compatibility in 5.0, this involves a little bit of work.
1509 * We also want to be able to function under 4.x, if needed.
1510 *
1511 * canonical: standard name reference. append a partition
1512 * reference, and open that file in the device directory.
1513 * examples: SVR4: c0t0d0
1514 * 4.x: sd0
1515 *
1516 * absolute: begins with a '/', and is assumed to be an
1517 * absolute pathname to some node.
1518 *
1519 * relative: non-canonical, doesn't begin with a '/'.
1520 * assumed to be the name of a file in the appropriate
1521 * device directory.
1522 */
1523 static void
search_for_logical_dev(char * devname)1524 search_for_logical_dev(char *devname)
1525 {
1526 char path[MAXPATHLEN];
1527 char *directory = "/dev/rdsk/";
1528 char *partition = "s2";
1529
1530 /*
1531 * If the name is an absolute path name, accept it as is
1532 */
1533 if (*devname == '/') {
1534 (void) strcpy(path, devname);
1535 } else if (canonical_name(devname)) {
1536 /*
1537 * If canonical name, construct a standard path name.
1538 */
1539 (void) strcpy(path, directory);
1540 (void) strcat(path, devname);
1541 (void) strcat(path, partition);
1542 } else if (canonical4x_name(devname)) {
1543 /*
1544 * Check to see if it's a 4.x file name in the /dev
1545 * directory on 5.0. Here, we only accept the
1546 * canonicalized form: sd0.
1547 */
1548 (void) strcpy(path, "/dev/r");
1549 (void) strcat(path, devname);
1550 (void) strcat(path, "c");
1551 } else {
1552 /*
1553 * If it's not a canonical name, then it may be a
1554 * reference to an actual file name in the device
1555 * directory itself.
1556 */
1557 (void) strcpy(path, directory);
1558 (void) strcat(path, devname);
1559 }
1560
1561 /* now add the device */
1562 add_device_to_disklist(devname, path);
1563 }
1564
1565 /*
1566 * Get the disk name from the inquiry data
1567 */
1568 static void
get_disk_name(int fd,char * disk_name,struct disk_info * disk_info)1569 get_disk_name(int fd, char *disk_name, struct disk_info *disk_info)
1570 {
1571 struct scsi_inquiry inquiry;
1572 char *vid, *pid, *rid;
1573
1574 if (get_disk_inquiry_prop(disk_info->devfs_name, &vid, &pid, &rid)
1575 == 0) {
1576 (void) snprintf(disk_name, MAXNAMELEN - 1, "%s-%s-%s",
1577 vid, pid, rid);
1578 free(vid);
1579 free(pid);
1580 free(rid);
1581
1582 return;
1583 }
1584
1585 if (uscsi_inquiry(fd, (char *)&inquiry, sizeof (inquiry))) {
1586 if (option_msg)
1587 err_print("\nInquiry failed - %s\n", strerror(errno));
1588 (void) strcpy(disk_name, "Unknown-Unknown-0001");
1589 return;
1590 }
1591
1592 (void) get_generic_disk_name(disk_name, &inquiry);
1593 }
1594
1595 /*
1596 * Add a device to the disk list, if it appears to be a disk,
1597 * and we haven't already found it under some other name.
1598 */
1599 static void
add_device_to_disklist(char * devname,char * devpath)1600 add_device_to_disklist(char *devname, char *devpath)
1601 {
1602 struct disk_info *search_disk;
1603 struct ctlr_info *search_ctlr;
1604 struct disk_type *search_dtype, *efi_disk;
1605 struct partition_info *search_parts;
1606 struct disk_info *dptr;
1607 struct ctlr_info *cptr;
1608 struct disk_type *type;
1609 struct partition_info *parts;
1610 struct dk_label search_label;
1611 struct dk_cinfo dkinfo;
1612 struct stat stbuf;
1613 struct ctlr_type *ctlr, *tctlr;
1614 struct mctlr_list *mlp;
1615 struct efi_info efi_info;
1616 struct dk_minfo mediainfo;
1617 int search_file;
1618 int status;
1619 int i;
1620 int access_flags = 0;
1621 char disk_name[MAXNAMELEN];
1622
1623 /*
1624 * Attempt to open the disk. If it fails, skip it.
1625 */
1626 if ((search_file = open_disk(devpath, O_RDWR | O_NDELAY)) < 0) {
1627 return;
1628 }
1629 /*
1630 * Must be a character device
1631 */
1632 if (fstat(search_file, &stbuf) == -1 || !S_ISCHR(stbuf.st_mode)) {
1633 (void) close(search_file);
1634 return;
1635 }
1636 /*
1637 * Attempt to read the configuration info on the disk.
1638 * Again, if it fails, we assume the disk's not there.
1639 * Note we must close the file for the disk before we
1640 * continue.
1641 */
1642 if (ioctl(search_file, DKIOCINFO, &dkinfo) < 0) {
1643 (void) close(search_file);
1644 return;
1645 }
1646
1647 /* If it is a removable media, skip it. */
1648
1649 if (!expert_mode) {
1650 int isremovable, ret;
1651 ret = ioctl(search_file, DKIOCREMOVABLE, &isremovable);
1652 if ((ret >= 0) && (isremovable != 0)) {
1653 (void) close(search_file);
1654 return;
1655 }
1656 }
1657
1658 if (ioctl(search_file, DKIOCGMEDIAINFO, &mediainfo) == -1) {
1659 cur_blksz = DEV_BSIZE;
1660 } else {
1661 cur_blksz = mediainfo.dki_lbsize;
1662 }
1663
1664 /*
1665 * If the type of disk is one we don't know about,
1666 * add it to the list.
1667 */
1668 mlp = controlp;
1669
1670 while (mlp != NULL) {
1671 if (mlp->ctlr_type->ctype_ctype == dkinfo.dki_ctype) {
1672 break;
1673 }
1674 mlp = mlp->next;
1675 }
1676
1677 if (mlp == NULL) {
1678 if (dkinfo.dki_ctype == DKC_CDROM) {
1679 if (ioctl(search_file, DKIOCGMEDIAINFO,
1680 &mediainfo) < 0) {
1681 mediainfo.dki_media_type = DK_UNKNOWN;
1682 }
1683 }
1684 /*
1685 * Skip CDROM devices, they are read only.
1686 * But not devices like Iomega Rev Drive which
1687 * identifies itself as a CDROM, but has a removable
1688 * disk.
1689 */
1690 if ((dkinfo.dki_ctype == DKC_CDROM) &&
1691 (mediainfo.dki_media_type != DK_REMOVABLE_DISK)) {
1692 (void) close(search_file);
1693 return;
1694 }
1695 /*
1696 * create the new ctlr_type structure and fill it in.
1697 */
1698 tctlr = zalloc(sizeof (struct ctlr_type));
1699 tctlr->ctype_ctype = dkinfo.dki_ctype;
1700 tctlr->ctype_name = zalloc(DK_DEVLEN);
1701 if (strlcpy(tctlr->ctype_name, dkinfo.dki_cname,
1702 DK_DEVLEN) > DK_DEVLEN) {
1703 /*
1704 * DKIOCINFO returned a controller name longer
1705 * than DK_DEVLEN bytes, which means more of the
1706 * dk_cinfo structure may be corrupt. We don't
1707 * allow the user to perform any operations on
1708 * the device in this case
1709 */
1710 err_print("\nError: Device %s: controller "
1711 "name (%s)\nis invalid. Device will not "
1712 "be displayed.\n", devname, dkinfo.dki_cname);
1713 (void) close(search_file);
1714 destroy_data(tctlr->ctype_name);
1715 destroy_data((char *)tctlr);
1716 return;
1717 } else {
1718 tctlr->ctype_ops = zalloc(sizeof (struct ctlr_ops));
1719
1720 /*
1721 * copy the generic disk ops structure into local copy.
1722 */
1723 *(tctlr->ctype_ops) = genericops;
1724
1725 tctlr->ctype_flags = CF_WLIST;
1726
1727 mlp = controlp;
1728
1729 while (mlp->next != NULL) {
1730 mlp = mlp->next;
1731 }
1732
1733 mlp->next = zalloc(sizeof (struct mctlr_list));
1734 mlp->next->ctlr_type = tctlr;
1735 }
1736 }
1737
1738 /*
1739 * Search through all disks known at this time, to
1740 * determine if we're already identified this disk.
1741 * If so, then there's no need to include it a
1742 * second time. This permits the user-defined names
1743 * to supercede the standard conventional names.
1744 */
1745 if (disk_is_known(&dkinfo)) {
1746 (void) close(search_file);
1747 return;
1748 }
1749 #if defined(sparc)
1750 /*
1751 * Because opening id with FNDELAY always succeeds,
1752 * read the label early on to see whether the device
1753 * really exists. A result of DSK_RESERVED
1754 * means the disk may be reserved.
1755 * In the future, it will be good
1756 * to move these into controller specific files and have a common
1757 * generic check for reserved disks here, including intel disks.
1758 */
1759 if (dkinfo.dki_ctype == DKC_SCSI_CCS) {
1760 char *first_sector;
1761
1762 first_sector = zalloc(cur_blksz);
1763 i = scsi_rdwr(DIR_READ, search_file, (diskaddr_t)0,
1764 1, first_sector, F_SILENT, NULL);
1765 switch (i) {
1766 case DSK_RESERVED:
1767 access_flags |= DSK_RESERVED;
1768 break;
1769 case DSK_UNAVAILABLE:
1770 access_flags |= DSK_UNAVAILABLE;
1771 break;
1772 default:
1773 break;
1774 }
1775 free(first_sector);
1776 }
1777 #endif /* defined(sparc) */
1778
1779 /*
1780 * The disk appears to be present. Allocate space for the
1781 * disk structure and add it to the list of found disks.
1782 */
1783 search_disk = (struct disk_info *)zalloc(sizeof (struct disk_info));
1784 if (disk_list == NULL)
1785 disk_list = search_disk;
1786 else {
1787 for (dptr = disk_list; dptr->disk_next != NULL;
1788 dptr = dptr->disk_next)
1789 ;
1790 dptr->disk_next = search_disk;
1791 }
1792 /*
1793 * Fill in some info from the ioctls.
1794 */
1795 search_disk->disk_dkinfo = dkinfo;
1796 if (is_efi_type(search_file)) {
1797 search_disk->label_type = L_TYPE_EFI;
1798 } else {
1799 search_disk->label_type = L_TYPE_SOLARIS;
1800 }
1801 /*
1802 * Remember the names of the disk
1803 */
1804 search_disk->disk_name = alloc_string(devname);
1805 search_disk->disk_path = alloc_string(devpath);
1806
1807 /*
1808 * Remember the lba size of the disk
1809 */
1810 search_disk->disk_lbasize = cur_blksz;
1811
1812 (void) strcpy(x86_devname, devname);
1813
1814 /*
1815 * Determine if this device is linked to a physical name.
1816 */
1817 search_disk->devfs_name = get_physical_name(devpath);
1818
1819 /*
1820 * Try to match the ctlr for this disk with a ctlr we
1821 * have already found. A match is assumed if the ctlrs
1822 * are at the same address && ctypes agree
1823 */
1824 for (search_ctlr = ctlr_list; search_ctlr != NULL;
1825 search_ctlr = search_ctlr->ctlr_next)
1826 if (search_ctlr->ctlr_addr == dkinfo.dki_addr &&
1827 search_ctlr->ctlr_space == dkinfo.dki_space &&
1828 search_ctlr->ctlr_ctype->ctype_ctype ==
1829 dkinfo.dki_ctype)
1830 break;
1831 /*
1832 * If no match was found, we need to identify this ctlr.
1833 */
1834 if (search_ctlr == NULL) {
1835 /*
1836 * Match the type of the ctlr to a known type.
1837 */
1838 mlp = controlp;
1839
1840 while (mlp != NULL) {
1841 if (mlp->ctlr_type->ctype_ctype == dkinfo.dki_ctype)
1842 break;
1843 mlp = mlp->next;
1844 }
1845 /*
1846 * If no match was found, it's an error.
1847 * Close the disk and report the error.
1848 */
1849 if (mlp == NULL) {
1850 err_print("\nError: found disk attached to ");
1851 err_print("unsupported controller type '%d'.\n",
1852 dkinfo.dki_ctype);
1853 (void) close(search_file);
1854 return;
1855 }
1856 /*
1857 * Allocate space for the ctlr structure and add it
1858 * to the list of found ctlrs.
1859 */
1860 search_ctlr = (struct ctlr_info *)
1861 zalloc(sizeof (struct ctlr_info));
1862 search_ctlr->ctlr_ctype = mlp->ctlr_type;
1863 if (ctlr_list == NULL)
1864 ctlr_list = search_ctlr;
1865 else {
1866 for (cptr = ctlr_list; cptr->ctlr_next != NULL;
1867 cptr = cptr->ctlr_next)
1868 ;
1869 cptr->ctlr_next = search_ctlr;
1870 }
1871 /*
1872 * Fill in info from the ioctl.
1873 */
1874 for (i = 0; i < DK_DEVLEN; i++) {
1875 search_ctlr->ctlr_cname[i] = dkinfo.dki_cname[i];
1876 search_ctlr->ctlr_dname[i] = dkinfo.dki_dname[i];
1877 }
1878 /*
1879 * Make sure these can be used as simple strings
1880 */
1881 search_ctlr->ctlr_cname[i] = 0;
1882 search_ctlr->ctlr_dname[i] = 0;
1883
1884 search_ctlr->ctlr_flags = dkinfo.dki_flags;
1885 search_ctlr->ctlr_num = dkinfo.dki_cnum;
1886 search_ctlr->ctlr_addr = dkinfo.dki_addr;
1887 search_ctlr->ctlr_space = dkinfo.dki_space;
1888 search_ctlr->ctlr_prio = dkinfo.dki_prio;
1889 search_ctlr->ctlr_vec = dkinfo.dki_vec;
1890 }
1891 /*
1892 * By this point, we have a known ctlr. Link the disk
1893 * to the ctlr.
1894 */
1895 search_disk->disk_ctlr = search_ctlr;
1896 if (access_flags & (DSK_RESERVED | DSK_UNAVAILABLE)) {
1897 if (access_flags & DSK_RESERVED)
1898 search_disk->disk_flags |= DSK_RESERVED;
1899 else
1900 search_disk->disk_flags |= DSK_UNAVAILABLE;
1901 (void) close(search_file);
1902 return;
1903 } else {
1904 search_disk->disk_flags &= ~(DSK_RESERVED | DSK_UNAVAILABLE);
1905 }
1906
1907 /*
1908 * Attempt to read the primary label.
1909 * (Note that this is really through the DKIOCGVTOC
1910 * ioctl, then converted from vtoc to label.)
1911 */
1912 if (search_disk->label_type == L_TYPE_SOLARIS) {
1913 status = read_label(search_file, &search_label);
1914 } else {
1915 status = read_efi_label(search_file, &efi_info, search_disk);
1916 }
1917 /*
1918 * If reading the label failed, and this is a SCSI
1919 * disk, we can attempt to auto-sense the disk
1920 * Configuration.
1921 */
1922 ctlr = search_ctlr->ctlr_ctype;
1923 if ((status == -1) && (ctlr->ctype_ctype == DKC_SCSI_CCS)) {
1924 if (option_msg && diag_msg) {
1925 err_print("%s: attempting auto configuration\n",
1926 search_disk->disk_name);
1927 }
1928
1929 switch (search_disk->label_type) {
1930 case (L_TYPE_SOLARIS):
1931 if (auto_sense(search_file, 0, &search_label) != NULL) {
1932 /*
1933 * Auto config worked, so we now have
1934 * a valid label for the disk. Mark
1935 * the disk as needing the label flushed.
1936 */
1937 status = 0;
1938 search_disk->disk_flags |=
1939 (DSK_LABEL_DIRTY | DSK_AUTO_CONFIG);
1940 }
1941 break;
1942 case (L_TYPE_EFI):
1943 efi_disk = auto_efi_sense(search_file, &efi_info);
1944 if (efi_disk != NULL) {
1945 /*
1946 * Auto config worked, so we now have
1947 * a valid label for the disk.
1948 */
1949 status = 0;
1950 search_disk->disk_flags |=
1951 (DSK_LABEL_DIRTY | DSK_AUTO_CONFIG);
1952 }
1953 break;
1954 default:
1955 /* Should never happen */
1956 break;
1957 }
1958 }
1959
1960 /*
1961 * If we didn't successfully read the label, or the label
1962 * appears corrupt, just leave the disk as an unknown type.
1963 */
1964 if (status == -1) {
1965 (void) close(search_file);
1966 return;
1967 }
1968
1969 if (search_disk->label_type == L_TYPE_SOLARIS) {
1970 if (!checklabel(&search_label)) {
1971 (void) close(search_file);
1972 return;
1973 }
1974 if (trim_id(search_label.dkl_asciilabel)) {
1975 (void) close(search_file);
1976 return;
1977 }
1978 }
1979 /*
1980 * The label looks ok. Mark the disk as labeled.
1981 */
1982 search_disk->disk_flags |= DSK_LABEL;
1983
1984 if (search_disk->label_type == L_TYPE_EFI) {
1985 search_dtype = (struct disk_type *)
1986 zalloc(sizeof (struct disk_type));
1987 type = search_ctlr->ctlr_ctype->ctype_dlist;
1988 if (type == NULL) {
1989 search_ctlr->ctlr_ctype->ctype_dlist =
1990 search_dtype;
1991 } else {
1992 while (type->dtype_next != NULL) {
1993 type = type->dtype_next;
1994 }
1995 type->dtype_next = search_dtype;
1996 }
1997 search_dtype->dtype_next = NULL;
1998
1999 search_dtype->vendor = strdup(efi_info.vendor);
2000 search_dtype->product = strdup(efi_info.product);
2001 search_dtype->revision = strdup(efi_info.revision);
2002
2003 if (search_dtype->vendor == NULL ||
2004 search_dtype->product == NULL ||
2005 search_dtype->revision == NULL) {
2006 free(search_dtype->vendor);
2007 free(search_dtype->product);
2008 free(search_dtype->revision);
2009 free(search_dtype);
2010 goto out;
2011 }
2012
2013 search_dtype->capacity = efi_info.capacity;
2014 search_disk->disk_type = search_dtype;
2015
2016 search_parts = (struct partition_info *)
2017 zalloc(sizeof (struct partition_info));
2018 search_dtype->dtype_plist = search_parts;
2019
2020 search_parts->pinfo_name = alloc_string("original");
2021 search_parts->pinfo_next = NULL;
2022 search_parts->etoc = efi_info.e_parts;
2023 search_disk->disk_parts = search_parts;
2024
2025 /*
2026 * Copy the volume name, if present
2027 */
2028 for (i = 0; i < search_parts->etoc->efi_nparts; i++) {
2029 if (search_parts->etoc->efi_parts[i].p_tag ==
2030 V_RESERVED) {
2031 if (search_parts->etoc->efi_parts[i].p_name) {
2032 bcopy(search_parts->etoc->efi_parts[i]
2033 .p_name, search_disk->v_volume,
2034 LEN_DKL_VVOL);
2035 } else {
2036 bzero(search_disk->v_volume,
2037 LEN_DKL_VVOL);
2038 }
2039 break;
2040 }
2041 }
2042 out:
2043 (void) close(search_file);
2044
2045 free(efi_info.vendor);
2046 free(efi_info.product);
2047 free(efi_info.revision);
2048 return;
2049 }
2050
2051 /*
2052 * Attempt to match the disk type in the label with a
2053 * known disk type.
2054 */
2055 for (search_dtype = search_ctlr->ctlr_ctype->ctype_dlist;
2056 search_dtype != NULL;
2057 search_dtype = search_dtype->dtype_next)
2058 if (dtype_match(&search_label, search_dtype))
2059 break;
2060 /*
2061 * If no match was found, we need to create a disk type
2062 * for this disk.
2063 */
2064 if (search_dtype == NULL) {
2065 /*
2066 * Allocate space for the disk type and add it
2067 * to the list of disk types for this ctlr type.
2068 */
2069 search_dtype = (struct disk_type *)
2070 zalloc(sizeof (struct disk_type));
2071 type = search_ctlr->ctlr_ctype->ctype_dlist;
2072 if (type == NULL)
2073 search_ctlr->ctlr_ctype->ctype_dlist =
2074 search_dtype;
2075 else {
2076 while (type->dtype_next != NULL)
2077 type = type->dtype_next;
2078 type->dtype_next = search_dtype;
2079 }
2080 /*
2081 * Fill in the drive info from the disk label.
2082 */
2083 search_dtype->dtype_next = NULL;
2084 if (strncmp(search_label.dkl_asciilabel, "DEFAULT",
2085 strlen("DEFAULT")) == 0) {
2086 (void) get_disk_name(search_file, disk_name,
2087 search_disk);
2088 search_dtype->dtype_asciilabel = (char *)
2089 zalloc(strlen(disk_name) + 1);
2090 (void) strcpy(search_dtype->dtype_asciilabel,
2091 disk_name);
2092 } else {
2093 search_dtype->dtype_asciilabel = (char *)
2094 zalloc(strlen(search_label.dkl_asciilabel) + 1);
2095 (void) strcpy(search_dtype->dtype_asciilabel,
2096 search_label.dkl_asciilabel);
2097 }
2098 search_dtype->dtype_pcyl = search_label.dkl_pcyl;
2099 search_dtype->dtype_ncyl = search_label.dkl_ncyl;
2100 search_dtype->dtype_acyl = search_label.dkl_acyl;
2101 search_dtype->dtype_nhead = search_label.dkl_nhead;
2102 search_dtype->dtype_nsect = search_label.dkl_nsect;
2103 search_dtype->dtype_rpm = search_label.dkl_rpm;
2104 /*
2105 * Mark the disk as needing specification of
2106 * ctlr specific attributes. This is necessary
2107 * because the label doesn't contain these attributes,
2108 * and they aren't known at this point. They will
2109 * be asked for if this disk is ever selected by
2110 * the user.
2111 * Note: for SCSI, we believe the label.
2112 */
2113 if ((search_ctlr->ctlr_ctype->ctype_ctype != DKC_SCSI_CCS) &&
2114 (search_ctlr->ctlr_ctype->ctype_ctype != DKC_DIRECT) &&
2115 (search_ctlr->ctlr_ctype->ctype_ctype != DKC_VBD) &&
2116 (search_ctlr->ctlr_ctype->ctype_ctype != DKC_PCMCIA_ATA) &&
2117 (search_ctlr->ctlr_ctype->ctype_ctype != DKC_BLKDEV)) {
2118 search_dtype->dtype_flags |= DT_NEED_SPEFS;
2119 }
2120 }
2121 /*
2122 * By this time we have a known disk type. Link the disk
2123 * to the disk type.
2124 */
2125 search_disk->disk_type = search_dtype;
2126
2127 /*
2128 * Close the file for this disk
2129 */
2130 (void) close(search_file);
2131
2132 /*
2133 * Attempt to match the partition map in the label with
2134 * a known partition map for this disk type.
2135 */
2136 for (search_parts = search_dtype->dtype_plist;
2137 search_parts != NULL;
2138 search_parts = search_parts->pinfo_next)
2139 if (parts_match(&search_label, search_parts)) {
2140 break;
2141 }
2142 /*
2143 * If no match was made, we need to create a partition
2144 * map for this disk.
2145 */
2146 if (search_parts == NULL) {
2147 /*
2148 * Allocate space for the partition map and add
2149 * it to the list of maps for this disk type.
2150 */
2151 search_parts = (struct partition_info *)
2152 zalloc(sizeof (struct partition_info));
2153 parts = search_dtype->dtype_plist;
2154 if (parts == NULL)
2155 search_dtype->dtype_plist = search_parts;
2156 else {
2157 while (parts->pinfo_next != NULL)
2158 parts = parts->pinfo_next;
2159 parts->pinfo_next = search_parts;
2160 }
2161 search_parts->pinfo_next = NULL;
2162 /*
2163 * Fill in the name of the map with a name derived
2164 * from the name of this disk. This is necessary
2165 * because the label contains no name for the
2166 * partition map.
2167 */
2168 search_parts->pinfo_name = alloc_string("original");
2169 /*
2170 * Fill in the partition info from the disk label.
2171 */
2172 for (i = 0; i < NDKMAP; i++) {
2173
2174 #if defined(_SUNOS_VTOC_8)
2175 search_parts->pinfo_map[i] =
2176 search_label.dkl_map[i];
2177
2178 #elif defined(_SUNOS_VTOC_16)
2179 search_parts->pinfo_map[i].dkl_cylno =
2180 search_label.dkl_vtoc.v_part[i].p_start /
2181 ((blkaddr32_t)(search_label.dkl_nhead *
2182 search_label.dkl_nsect));
2183 search_parts->pinfo_map[i].dkl_nblk =
2184 search_label.dkl_vtoc.v_part[i].p_size;
2185
2186 #else
2187 #error No VTOC format defined.
2188 #endif
2189 }
2190 }
2191 /*
2192 * If the vtoc looks valid, copy the volume name and vtoc
2193 * info from the label. Otherwise, install a default vtoc.
2194 * This permits vtoc info to automatically appear in the sun
2195 * label, without requiring an upgrade procedure.
2196 */
2197 if (search_label.dkl_vtoc.v_version == V_VERSION) {
2198 bcopy(search_label.dkl_vtoc.v_volume,
2199 search_disk->v_volume, LEN_DKL_VVOL);
2200 search_parts->vtoc = search_label.dkl_vtoc;
2201 } else {
2202 bzero(search_disk->v_volume, LEN_DKL_VVOL);
2203 set_vtoc_defaults(search_parts);
2204 }
2205 /*
2206 * By this time we have a known partitition map. Link the
2207 * disk to the partition map.
2208 */
2209 search_disk->disk_parts = search_parts;
2210 }
2211
2212
2213 /*
2214 * Search the disk list for a disk with the identical configuration.
2215 * Return true if one is found.
2216 */
2217 static int
disk_is_known(struct dk_cinfo * dkinfo)2218 disk_is_known(struct dk_cinfo *dkinfo)
2219 {
2220 struct disk_info *dp;
2221
2222 dp = disk_list;
2223 while (dp != NULL) {
2224 if (dp->disk_dkinfo.dki_ctype == dkinfo->dki_ctype &&
2225 dp->disk_dkinfo.dki_cnum == dkinfo->dki_cnum &&
2226 dp->disk_dkinfo.dki_unit == dkinfo->dki_unit &&
2227 strcmp(dp->disk_dkinfo.dki_dname, dkinfo->dki_dname) == 0) {
2228 return (1);
2229 }
2230 dp = dp->disk_next;
2231 }
2232 return (0);
2233 }
2234
2235
2236 /*
2237 * This routine checks to see if a given disk type matches the type
2238 * in the disk label.
2239 */
2240 int
dtype_match(struct dk_label * label,struct disk_type * dtype)2241 dtype_match(struct dk_label *label, struct disk_type *dtype)
2242 {
2243
2244 if (dtype->dtype_asciilabel == NULL) {
2245 return (0);
2246 }
2247
2248 /*
2249 * If the any of the physical characteristics are different, or
2250 * the name is different, it doesn't match.
2251 */
2252 if ((strcmp(label->dkl_asciilabel, dtype->dtype_asciilabel) != 0) ||
2253 (label->dkl_ncyl != dtype->dtype_ncyl) ||
2254 (label->dkl_acyl != dtype->dtype_acyl) ||
2255 (label->dkl_nhead != dtype->dtype_nhead) ||
2256 (label->dkl_nsect != dtype->dtype_nsect)) {
2257 return (0);
2258 }
2259 /*
2260 * If those are all identical, assume it's a match.
2261 */
2262 return (1);
2263 }
2264
2265 /*
2266 * This routine checks to see if a given partition map matches the map
2267 * in the disk label.
2268 */
2269 int
parts_match(struct dk_label * label,struct partition_info * pinfo)2270 parts_match(struct dk_label *label, struct partition_info *pinfo)
2271 {
2272 int i;
2273
2274 /*
2275 * If any of the partition entries is different, it doesn't match.
2276 */
2277 for (i = 0; i < NDKMAP; i++)
2278
2279 #if defined(_SUNOS_VTOC_8)
2280 if ((label->dkl_map[i].dkl_cylno !=
2281 pinfo->pinfo_map[i].dkl_cylno) ||
2282 (label->dkl_map[i].dkl_nblk !=
2283 pinfo->pinfo_map[i].dkl_nblk))
2284
2285 #elif defined(_SUNOS_VTOC_16)
2286 if ((pinfo->pinfo_map[i].dkl_cylno !=
2287 label->dkl_vtoc.v_part[i].p_start /
2288 (label->dkl_nhead * label->dkl_nsect)) ||
2289 (pinfo->pinfo_map[i].dkl_nblk !=
2290 label->dkl_vtoc.v_part[i].p_size))
2291 #else
2292 #error No VTOC format defined.
2293 #endif
2294 return (0);
2295 /*
2296 * Compare the vtoc information for a match
2297 * Do not require the volume name to be equal, for a match!
2298 */
2299 if (label->dkl_vtoc.v_version != pinfo->vtoc.v_version)
2300 return (0);
2301 if (label->dkl_vtoc.v_nparts != pinfo->vtoc.v_nparts)
2302 return (0);
2303 for (i = 0; i < NDKMAP; i++) {
2304 if (label->dkl_vtoc.v_part[i].p_tag !=
2305 pinfo->vtoc.v_part[i].p_tag)
2306 return (0);
2307 if (label->dkl_vtoc.v_part[i].p_flag !=
2308 pinfo->vtoc.v_part[i].p_flag)
2309 return (0);
2310 }
2311 /*
2312 * If they are all identical, it's a match.
2313 */
2314 return (1);
2315 }
2316
2317 /*
2318 * This routine checks to see if the given disk name refers to the disk
2319 * in the given disk structure.
2320 */
2321 int
diskname_match(char * name,struct disk_info * disk)2322 diskname_match(char *name, struct disk_info *disk)
2323 {
2324 struct dk_cinfo dkinfo;
2325 char s[MAXPATHLEN];
2326 int fd;
2327
2328 /*
2329 * Match the name of the disk in the disk_info structure
2330 */
2331 if (strcmp(name, disk->disk_name) == 0) {
2332 return (1);
2333 }
2334
2335 /*
2336 * Check to see if it's a 4.x file name in the /dev
2337 * directory on 5.0. Here, we only accept the
2338 * canonicalized form: sd0.
2339 */
2340 if (canonical4x_name(name) == 0) {
2341 return (0);
2342 }
2343
2344 (void) strcpy(s, "/dev/r");
2345 (void) strcat(s, name);
2346 (void) strcat(s, "c");
2347
2348 if ((fd = open_disk(s, O_RDWR | O_NDELAY)) < 0) {
2349 return (0);
2350 }
2351
2352 if (ioctl(fd, DKIOCINFO, &dkinfo) < 0) {
2353 (void) close(fd);
2354 return (0);
2355 }
2356 (void) close(fd);
2357
2358 if (disk->disk_dkinfo.dki_ctype == dkinfo.dki_ctype &&
2359 disk->disk_dkinfo.dki_cnum == dkinfo.dki_cnum &&
2360 disk->disk_dkinfo.dki_unit == dkinfo.dki_unit &&
2361 strcmp(disk->disk_dkinfo.dki_dname, dkinfo.dki_dname) == 0) {
2362 return (1);
2363 }
2364 return (0);
2365 }
2366
2367
2368 static void
datafile_error(char * errmsg,char * token)2369 datafile_error(char *errmsg, char *token)
2370 {
2371 int token_type;
2372 TOKEN token_buf;
2373
2374 /*
2375 * Allow us to get by controllers that the other platforms don't
2376 * know about.
2377 */
2378 if (errmsg != NULL) {
2379 err_print(errmsg, token);
2380 err_print(" - %s (%d)\n", file_name, data_lineno);
2381 }
2382
2383 /*
2384 * Re-sync the parsing at the beginning of the next line
2385 * unless of course we're already there.
2386 */
2387 if (last_token_type != SUP_EOF && last_token_type != SUP_EOL) {
2388 do {
2389 token_type = sup_gettoken(token_buf);
2390 } while (token_type != SUP_EOF && token_type != SUP_EOL);
2391
2392 if (token_type == SUP_EOF) {
2393 sup_pushtoken(token_buf, token_type);
2394 }
2395 }
2396 }
2397
2398
2399 /*
2400 * Search through all defined disk types for duplicate entries
2401 * that are inconsistent with each other. Disks with different
2402 * characteristics should be named differently.
2403 * Note that this function only checks for duplicate disks
2404 * for the same controller. It's possible to have two disks with
2405 * the same name, but defined for different controllers.
2406 * That may or may not be a problem...
2407 */
2408 static void
search_duplicate_dtypes()2409 search_duplicate_dtypes()
2410 {
2411 struct disk_type *dp1;
2412 struct disk_type *dp2;
2413 struct mctlr_list *mlp;
2414
2415 mlp = controlp;
2416
2417 while (mlp != NULL) {
2418 dp1 = mlp->ctlr_type->ctype_dlist;
2419 while (dp1 != NULL) {
2420 dp2 = dp1->dtype_next;
2421 while (dp2 != NULL) {
2422 check_dtypes_for_inconsistency(dp1, dp2);
2423 dp2 = dp2->dtype_next;
2424 }
2425 dp1 = dp1->dtype_next;
2426 }
2427 mlp = mlp->next;
2428 }
2429 }
2430
2431
2432 /*
2433 * Search through all defined partition types for duplicate entries
2434 * that are inconsistent with each other. Partitions with different
2435 * characteristics should be named differently.
2436 * Note that this function only checks for duplicate partitions
2437 * for the same disk. It's possible to have two partitions with
2438 * the same name, but defined for different disks.
2439 * That may or may not be a problem...
2440 */
2441 static void
search_duplicate_pinfo()2442 search_duplicate_pinfo()
2443 {
2444 struct disk_type *dp;
2445 struct partition_info *pp1;
2446 struct partition_info *pp2;
2447 struct mctlr_list *mlp;
2448
2449 mlp = controlp;
2450
2451 while (mlp != NULL) {
2452 dp = mlp->ctlr_type->ctype_dlist;
2453 while (dp != NULL) {
2454 pp1 = dp->dtype_plist;
2455 while (pp1 != NULL) {
2456 pp2 = pp1->pinfo_next;
2457 while (pp2 != NULL) {
2458 check_pinfo_for_inconsistency(pp1, pp2);
2459 pp2 = pp2->pinfo_next;
2460 }
2461 pp1 = pp1->pinfo_next;
2462 }
2463 dp = dp->dtype_next;
2464 }
2465 mlp = mlp->next;
2466 }
2467 }
2468
2469
2470 /*
2471 * Determine if two particular disk definitions are inconsistent.
2472 * Ie: same name, but different characteristics.
2473 * If so, print an error message and abort.
2474 */
2475 static void
check_dtypes_for_inconsistency(struct disk_type * dp1,struct disk_type * dp2)2476 check_dtypes_for_inconsistency(struct disk_type *dp1, struct disk_type *dp2)
2477 {
2478 int i;
2479 int result;
2480 struct chg_list *cp1;
2481 struct chg_list *cp2;
2482
2483
2484 /*
2485 * If the name's different, we're ok
2486 */
2487 if (strcmp(dp1->dtype_asciilabel, dp2->dtype_asciilabel) != 0) {
2488 return;
2489 }
2490
2491 /*
2492 * Compare all the disks' characteristics
2493 */
2494 result = 0;
2495 result |= (dp1->dtype_flags != dp2->dtype_flags);
2496 result |= (dp1->dtype_options != dp2->dtype_options);
2497 result |= (dp1->dtype_fmt_time != dp2->dtype_fmt_time);
2498 result |= (dp1->dtype_bpt != dp2->dtype_bpt);
2499 result |= (dp1->dtype_ncyl != dp2->dtype_ncyl);
2500 result |= (dp1->dtype_acyl != dp2->dtype_acyl);
2501 result |= (dp1->dtype_pcyl != dp2->dtype_pcyl);
2502 result |= (dp1->dtype_nhead != dp2->dtype_nhead);
2503 result |= (dp1->dtype_nsect != dp2->dtype_nsect);
2504 result |= (dp1->dtype_rpm != dp2->dtype_rpm);
2505 result |= (dp1->dtype_cyl_skew != dp2->dtype_cyl_skew);
2506 result |= (dp1->dtype_trk_skew != dp2->dtype_trk_skew);
2507 result |= (dp1->dtype_trks_zone != dp2->dtype_trks_zone);
2508 result |= (dp1->dtype_atrks != dp2->dtype_atrks);
2509 result |= (dp1->dtype_asect != dp2->dtype_asect);
2510 result |= (dp1->dtype_cache != dp2->dtype_cache);
2511 result |= (dp1->dtype_threshold != dp2->dtype_threshold);
2512 result |= (dp1->dtype_read_retries != dp2->dtype_read_retries);
2513 result |= (dp1->dtype_write_retries != dp2->dtype_write_retries);
2514 result |= (dp1->dtype_prefetch_min != dp2->dtype_prefetch_min);
2515 result |= (dp1->dtype_prefetch_max != dp2->dtype_prefetch_max);
2516 for (i = 0; i < NSPECIFICS; i++) {
2517 result |= (dp1->dtype_specifics[i] != dp2->dtype_specifics[i]);
2518 }
2519
2520 cp1 = dp1->dtype_chglist;
2521 cp2 = dp2->dtype_chglist;
2522 while (cp1 != NULL && cp2 != NULL) {
2523 if (cp1 == NULL || cp2 == NULL) {
2524 result = 1;
2525 break;
2526 }
2527 result |= (cp1->pageno != cp2->pageno);
2528 result |= (cp1->byteno != cp2->byteno);
2529 result |= (cp1->mode != cp2->mode);
2530 result |= (cp1->value != cp2->value);
2531 cp1 = cp1->next;
2532 cp2 = cp2->next;
2533 }
2534
2535 if (result) {
2536 err_print("Inconsistent definitions for disk type '%s'\n",
2537 dp1->dtype_asciilabel);
2538 if (dp1->dtype_filename != NULL &&
2539 dp2->dtype_filename != NULL) {
2540 err_print("%s (%d) - %s (%d)\n",
2541 dp1->dtype_filename, dp1->dtype_lineno,
2542 dp2->dtype_filename, dp2->dtype_lineno);
2543 }
2544 fullabort();
2545 }
2546 }
2547
2548
2549 /*
2550 * Determine if two particular partition definitions are inconsistent.
2551 * Ie: same name, but different characteristics.
2552 * If so, print an error message and abort.
2553 */
2554 static void
check_pinfo_for_inconsistency(struct partition_info * pp1,struct partition_info * pp2)2555 check_pinfo_for_inconsistency(struct partition_info *pp1,
2556 struct partition_info *pp2)
2557 {
2558 int i;
2559 int result;
2560 struct dk_map32 *map1;
2561 struct dk_map32 *map2;
2562
2563 #if defined(_SUNOS_VTOC_8)
2564 struct dk_map2 *vp1;
2565 struct dk_map2 *vp2;
2566
2567 #elif defined(_SUNOS_VTOC_16)
2568 struct dkl_partition *vp1;
2569 struct dkl_partition *vp2;
2570 #else
2571 #error No VTOC layout defined.
2572 #endif /* defined(_SUNOS_VTOC_8) */
2573
2574 /*
2575 * If the name's different, we're ok
2576 */
2577 if (strcmp(pp1->pinfo_name, pp2->pinfo_name) != 0) {
2578 return;
2579 }
2580
2581 /*
2582 * Compare all the partitions' characteristics
2583 */
2584 result = 0;
2585 map1 = pp1->pinfo_map;
2586 map2 = pp2->pinfo_map;
2587 for (i = 0; i < NDKMAP; i++, map1++, map2++) {
2588 result |= (map1->dkl_cylno != map2->dkl_cylno);
2589 result |= (map1->dkl_nblk != map2->dkl_nblk);
2590 }
2591
2592 /*
2593 * Compare the significant portions of the vtoc information
2594 */
2595 vp1 = pp1->vtoc.v_part;
2596 vp2 = pp2->vtoc.v_part;
2597 for (i = 0; i < NDKMAP; i++, vp1++, vp2++) {
2598 result |= (vp1->p_tag != vp2->p_tag);
2599 result |= (vp1->p_flag != vp2->p_flag);
2600 }
2601
2602 if (result) {
2603 err_print("Inconsistent definitions for partition type '%s'\n",
2604 pp1->pinfo_name);
2605 if (pp1->pinfo_filename != NULL &&
2606 pp2->pinfo_filename != NULL) {
2607 err_print("%s (%d) - %s (%d)\n",
2608 pp1->pinfo_filename, pp1->pinfo_lineno,
2609 pp2->pinfo_filename, pp2->pinfo_lineno);
2610 }
2611 fullabort();
2612 }
2613 }
2614
2615 /*
2616 * Convert a string of digits into a block number.
2617 * The digits are assumed to be a block number unless the
2618 * the string is terminated by 'c', in which case it is
2619 * assumed to be in units of cylinders. Accept a 'b'
2620 * to explictly specify blocks, for consistency.
2621 *
2622 * NB: uses the macro spc(), which requires that the
2623 * globals nhead/nsect/acyl be set up correctly.
2624 *
2625 * Returns -1 in the case of an error.
2626 */
2627 static uint_t
str2blks(char * str)2628 str2blks(char *str)
2629 {
2630 int blks;
2631 char *p;
2632
2633 blks = (int)strtol(str, &p, 0);
2634 /*
2635 * Check what terminated the conversion.
2636 */
2637 if (*p != 0) {
2638 /*
2639 * Units specifier of 'c': convert cylinders to blocks
2640 */
2641 if (*p == 'c') {
2642 p++;
2643 blks = blks * spc();
2644 /*
2645 * Ignore a 'b' specifier.
2646 */
2647 } else if (*p == 'b') {
2648 p++;
2649 }
2650 /*
2651 * Anthing left over is an error
2652 */
2653 if (*p != 0) {
2654 blks = -1;
2655 }
2656 }
2657
2658 return (blks);
2659 }
2660 /*
2661 * Convert a string of digits into a cylinder number.
2662 * Accept a an optional 'c' specifier, for consistency.
2663 *
2664 * Returns -1 in the case of an error.
2665 */
2666 int
str2cyls(char * str)2667 str2cyls(char *str)
2668 {
2669 int cyls;
2670 char *p;
2671
2672 cyls = (int)strtol(str, &p, 0);
2673 /*
2674 * Check what terminated the conversion.
2675 */
2676 if (*p != 0) {
2677 /*
2678 * Units specifier of 'c': accept it.
2679 */
2680 if (*p == 'c') {
2681 p++;
2682 }
2683 /*
2684 * Anthing left over is an error
2685 */
2686 if (*p != 0) {
2687 cyls = -1;
2688 }
2689 }
2690
2691 return (cyls);
2692 }
2693
2694
2695 /*
2696 * Create a new chg_list structure, and append it onto the
2697 * end of the current chg_list under construction. By
2698 * applying changes in the order in which listed in the
2699 * data file, the changes we make are deterministic.
2700 * Return a pointer to the new structure, so that the
2701 * caller can fill in the appropriate information.
2702 */
2703 static struct chg_list *
new_chg_list(struct disk_type * disk)2704 new_chg_list(struct disk_type *disk)
2705 {
2706 struct chg_list *cp;
2707 struct chg_list *nc;
2708
2709 nc = zalloc(sizeof (struct chg_list));
2710
2711 if (disk->dtype_chglist == NULL) {
2712 disk->dtype_chglist = nc;
2713 } else {
2714 for (cp = disk->dtype_chglist; cp->next; cp = cp->next)
2715 ;
2716 cp->next = nc;
2717 }
2718 nc->next = NULL;
2719 return (nc);
2720 }
2721
2722
2723 /*
2724 * Follow symbolic links from the logical device name to
2725 * the /devfs physical device name. To be complete, we
2726 * handle the case of multiple links. This function
2727 * either returns NULL (no links, or some other error),
2728 * or the physical device name, alloc'ed on the heap.
2729 *
2730 * Note that the standard /devices prefix is stripped from
2731 * the final pathname, if present. The trailing options
2732 * are also removed (":c, raw").
2733 */
2734 static char *
get_physical_name(char * path)2735 get_physical_name(char *path)
2736 {
2737 struct stat stbuf;
2738 int i;
2739 int level;
2740 char *p;
2741 char s[MAXPATHLEN];
2742 char buf[MAXPATHLEN];
2743 char dir[MAXPATHLEN];
2744 char savedir[MAXPATHLEN];
2745 char *result = NULL;
2746
2747 if (getcwd(savedir, sizeof (savedir)) == NULL) {
2748 err_print("getcwd() failed - %s\n", strerror(errno));
2749 return (NULL);
2750 }
2751
2752 (void) strcpy(s, path);
2753 if ((p = strrchr(s, '/')) != NULL) {
2754 *p = 0;
2755 }
2756 if (s[0] == 0) {
2757 (void) strcpy(s, "/");
2758 }
2759 if (chdir(s) == -1) {
2760 err_print("cannot chdir() to %s - %s\n",
2761 s, strerror(errno));
2762 goto exit;
2763 }
2764
2765 level = 0;
2766 (void) strcpy(s, path);
2767 for (;;) {
2768 /*
2769 * See if there's a real file out there. If not,
2770 * we have a dangling link and we ignore it.
2771 */
2772 if (stat(s, &stbuf) == -1) {
2773 goto exit;
2774 }
2775 if (lstat(s, &stbuf) == -1) {
2776 err_print("%s: lstat() failed - %s\n",
2777 s, strerror(errno));
2778 goto exit;
2779 }
2780 /*
2781 * If the file is not a link, we're done one
2782 * way or the other. If there were links,
2783 * return the full pathname of the resulting
2784 * file.
2785 */
2786 if (!S_ISLNK(stbuf.st_mode)) {
2787 if (level > 0) {
2788 /*
2789 * Strip trailing options from the
2790 * physical device name
2791 */
2792 if ((p = strrchr(s, ':')) != NULL) {
2793 *p = 0;
2794 }
2795 /*
2796 * Get the current directory, and
2797 * glue the pieces together.
2798 */
2799 if (getcwd(dir, sizeof (dir)) == NULL) {
2800 err_print("getcwd() failed - %s\n",
2801 strerror(errno));
2802 goto exit;
2803 }
2804 (void) strcat(dir, "/");
2805 (void) strcat(dir, s);
2806 /*
2807 * If we have the standard fixed
2808 * /devices prefix, remove it.
2809 */
2810 p = (strstr(dir, DEVFS_PREFIX) == dir) ?
2811 dir+strlen(DEVFS_PREFIX) : dir;
2812 result = alloc_string(p);
2813 }
2814 goto exit;
2815 }
2816 i = readlink(s, buf, sizeof (buf));
2817 if (i == -1) {
2818 err_print("%s: readlink() failed - %s\n",
2819 s, strerror(errno));
2820 goto exit;
2821 }
2822 level++;
2823 buf[i] = 0;
2824
2825 /*
2826 * Break up the pathname into the directory
2827 * reference, if applicable and simple filename.
2828 * chdir()'ing to the directory allows us to
2829 * handle links with relative pathnames correctly.
2830 */
2831 (void) strcpy(dir, buf);
2832 if ((p = strrchr(dir, '/')) != NULL) {
2833 *p = 0;
2834 if (chdir(dir) == -1) {
2835 err_print("cannot chdir() to %s - %s\n",
2836 dir, strerror(errno));
2837 goto exit;
2838 }
2839 (void) strcpy(s, p+1);
2840 } else {
2841 (void) strcpy(s, buf);
2842 }
2843 }
2844
2845 exit:
2846 if (chdir(savedir) == -1) {
2847 err_print("cannot chdir() to %s - %s\n",
2848 savedir, strerror(errno));
2849 }
2850
2851 return (result);
2852 }
2853
2854
2855 static void
sort_disk_list()2856 sort_disk_list()
2857 {
2858 int n;
2859 struct disk_info **disks;
2860 struct disk_info *d;
2861 struct disk_info **dp;
2862 struct disk_info **dp2;
2863
2864 /*
2865 * Count the number of disks in the list
2866 */
2867 n = 0;
2868 for (d = disk_list; d != NULL; d = d->disk_next) {
2869 n++;
2870 }
2871 if (n == 0) {
2872 return;
2873 }
2874
2875 /*
2876 * Allocate a simple disk list array and fill it in
2877 */
2878 disks = (struct disk_info **)
2879 zalloc((n+1) * sizeof (struct disk_info *));
2880
2881 dp = disks;
2882 for (d = disk_list; d != NULL; d = d->disk_next) {
2883 *dp++ = d;
2884 }
2885 *dp = NULL;
2886
2887 /*
2888 * Sort the disk list array
2889 */
2890 qsort((void *) disks, n, sizeof (struct disk_info *),
2891 disk_name_compare);
2892
2893 /*
2894 * Rebuild the linked list disk list structure
2895 */
2896 dp = disks;
2897 disk_list = *dp;
2898 dp2 = dp + 1;
2899 do {
2900 (*dp++)->disk_next = *dp2++;
2901 } while (*dp != NULL);
2902
2903 /*
2904 * Clean up
2905 */
2906 (void) destroy_data((void *)disks);
2907 }
2908
2909
2910 /*
2911 * Compare two disk names
2912 */
2913 static int
disk_name_compare(const void * arg1,const void * arg2)2914 disk_name_compare(
2915 const void *arg1,
2916 const void *arg2)
2917 {
2918 char *s1;
2919 char *s2;
2920 int n1;
2921 int n2;
2922 char *p1;
2923 char *p2;
2924
2925 s1 = (*((struct disk_info **)arg1))->disk_name;
2926 s2 = (*((struct disk_info **)arg2))->disk_name;
2927
2928 for (;;) {
2929 if (*s1 == 0 || *s2 == 0)
2930 break;
2931 if (isdigit(*s1) && isdigit(*s2)) {
2932 n1 = strtol(s1, &p1, 10);
2933 n2 = strtol(s2, &p2, 10);
2934 if (n1 != n2) {
2935 return (n1 - n2);
2936 }
2937 s1 = p1;
2938 s2 = p2;
2939 } else if (*s1 != *s2) {
2940 break;
2941 } else {
2942 s1++;
2943 s2++;
2944 }
2945 }
2946
2947 return (*s1 - *s2);
2948 }
2949
2950 static void
make_controller_list(void)2951 make_controller_list(void)
2952 {
2953 int x;
2954 struct mctlr_list *ctlrp;
2955
2956 ctlrp = controlp;
2957
2958 for (x = nctypes; x != 0; x--) {
2959 ctlrp = zalloc(sizeof (struct mctlr_list));
2960 ctlrp->next = controlp;
2961 ctlrp->ctlr_type = &ctlr_types[x - 1];
2962 controlp = ctlrp;
2963
2964 }
2965 }
2966
2967 static void
check_for_duplicate_disknames(char * arglist[])2968 check_for_duplicate_disknames(char *arglist[])
2969 {
2970 char *directory = "/dev/rdsk/";
2971 char **disklist;
2972 int len;
2973 char s[MAXPATHLEN], t[MAXPATHLEN];
2974 int diskno = 0;
2975 int i;
2976
2977
2978 len = strlen(directory);
2979 disklist = arglist;
2980 for (; *disklist != NULL; disklist++) {
2981 if (strncmp(directory, *disklist, len) == 0) {
2982 /* Disk is in conventional format */
2983 canonicalize_name(s, *disklist);
2984 /*
2985 * check if the disk is already present in
2986 * disk list.
2987 */
2988 for (i = 0; i < diskno; i++) {
2989 canonicalize_name(t, arglist[i]);
2990 if (strncmp(s, t, strlen(t)) == 0)
2991 break;
2992 }
2993 if (i != diskno)
2994 continue;
2995 }
2996 (void) strcpy(arglist[diskno], *disklist);
2997 diskno++;
2998 }
2999 arglist[diskno] = NULL;
3000 }
3001
3002 #define DISK_PREFIX "/dev/rdsk/"
3003
3004 /*
3005 * This Function checks if the non-conventional name is a a link to
3006 * one of the conventional whole disk name.
3007 */
3008 static int
name_represents_wholedisk(char * name)3009 name_represents_wholedisk(char *name)
3010 {
3011 char symname[MAXPATHLEN];
3012 char localname[MAXPATHLEN];
3013 char *nameptr;
3014 ssize_t symname_size;
3015
3016 if (strlcpy(localname, name, MAXPATHLEN) >= MAXPATHLEN)
3017 return (1); /* buffer overflow, reject this name */
3018
3019 while ((symname_size = readlink(
3020 localname, symname, MAXPATHLEN - 1)) != -1) {
3021 symname[symname_size] = '\0';
3022 nameptr = symname;
3023 if (strncmp(symname, DISK_PREFIX,
3024 (sizeof (DISK_PREFIX) - 1)) == 0)
3025 nameptr += (sizeof (DISK_PREFIX) - 1);
3026
3027 if (conventional_name(nameptr)) {
3028 if (whole_disk_name(nameptr))
3029 return (0);
3030 else
3031 return (1);
3032 }
3033
3034 (void) strcpy(localname, symname);
3035 }
3036 return (0);
3037 }
3038