Print this page
4833 Remove volrmmount
4845 rm(u)mount don't always print mount/unmount errors
4846 HAL partition names don't match real parition names
Reviewed by: Dan McDonald <danmcd@omniti.com>
Reviewed by: Josef 'Jeff' Sipek <jeffpc@josefsipek.net>
Split |
Close |
Expand all |
Collapse all |
--- old/usr/src/cmd/rmformat/rmf_misc.c
+++ new/usr/src/cmd/rmformat/rmf_misc.c
1 1 /*
2 2 * CDDL HEADER START
3 3 *
4 4 * The contents of this file are subject to the terms of the
5 5 * Common Development and Distribution License (the "License").
6 6 * You may not use this file except in compliance with the License.
7 7 *
8 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 9 * or http://www.opensolaris.org/os/licensing.
↓ open down ↓ |
9 lines elided |
↑ open up ↑ |
10 10 * See the License for the specific language governing permissions
11 11 * and limitations under the License.
12 12 *
13 13 * When distributing Covered Code, include this CDDL HEADER in each
14 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 15 * If applicable, add the following below this CDDL HEADER, with the
16 16 * fields enclosed by brackets "[]" replaced with your own identifying
17 17 * information: Portions Copyright [yyyy] [name of copyright owner]
18 18 *
19 19 * CDDL HEADER END
20 - */
21 -
22 -/*
20 + *
21 + *
23 22 * Copyright 2009 Sun Microsystems, Inc. All rights reserved.
24 23 * Use is subject to license terms.
24 + *
25 + * Copyright 2014 Andrew Stormont.
25 26 */
26 27
27 28 /*
28 29 * rmf_misc.c :
29 30 * Miscelleneous routines for rmformat.
30 31 */
31 32
32 33 #include <sys/types.h>
33 34 #include <stdio.h>
34 35 #include <sys/mnttab.h>
35 36 #include <volmgt.h>
36 37 #include <sys/dkio.h>
37 38 #include <sys/fdio.h>
38 39 #include <sys/vtoc.h>
39 40 #include <sys/termios.h>
40 41 #include <sys/mount.h>
41 42 #include <ctype.h>
42 43 #include <signal.h>
43 44 #include <sys/wait.h>
44 45 #include <dirent.h>
45 46 #include <priv_utils.h>
46 47 #include <stdarg.h>
47 48 #include "rmformat.h"
48 49
49 50 /*
50 51 * Definitions.
51 52 */
52 53 #define SENSE_KEY(rqbuf) (rqbuf[2] & 0xf) /* scsi error category */
53 54 #define ASC(rqbuf) (rqbuf[12]) /* additional sense code */
54 55 #define ASCQ(rqbuf) (rqbuf[13]) /* ASC qualifier */
55 56
56 57 #define DEFAULT_SCSI_TIMEOUT 60
57 58 #define INQUIRY_CMD 0x12
58 59 #define RQBUFLEN 32
59 60 #define CD_RW 1 /* CD_RW/CD-R */
60 61 #define WRITE_10_CMD 0x2A
61 62 #define READ_INFO_CMD 0x51
62 63 #define SYNC_CACHE_CMD 0x35
63 64 #define CLOSE_TRACK_CMD 0x5B
64 65 #define MODE_SENSE_10_CMD 0x5A
65 66 #define DEVFS_PREFIX "/devices"
66 67
67 68 int uscsi_error; /* used for debugging failed uscsi */
68 69 char rqbuf[RQBUFLEN];
69 70 static uint_t total_retries;
70 71 static struct uscsi_cmd uscmd;
71 72 static char ucdb[16];
72 73 uchar_t uscsi_status, rqstatus, rqresid;
73 74 int total_devices_found = 0;
74 75 int removable_found = 0;
75 76
76 77 extern char *global_intr_msg;
77 78 extern int vol_running;
78 79 extern char *dev_name;
79 80 extern int32_t m_flag;
↓ open down ↓ |
45 lines elided |
↑ open up ↑ |
80 81
81 82 /*
82 83 * ON-private functions from libvolmgt
83 84 */
84 85 int _dev_mounted(char *path);
85 86
86 87 /*
87 88 * Function prototypes.
88 89 */
89 90 static int my_umount(char *mountp);
90 -static int my_volrmmount(char *real_name);
91 91 static int vol_name_to_dev_node(char *vname, char *found);
92 92 static int vol_lookup(char *supplied, char *found);
93 93 static device_t *get_device(char *user_supplied, char *node);
94 94 static char *get_physical_name(char *path);
95 95 static int lookup_device(char *supplied, char *found);
96 96 static void fini_device(device_t *dev);
97 97 static int is_cd(char *node);
98 98 void *my_zalloc(size_t size);
99 99 void err_msg(char *fmt, ...);
100 100 int inquiry(int fd, uchar_t *inq);
101 101 struct uscsi_cmd *get_uscsi_cmd(void);
102 102 int uscsi(int fd, struct uscsi_cmd *scmd);
103 103 int get_mode_page(int fd, int page_no, int pc, int buf_len,
104 104 uchar_t *buffer);
105 105 int mode_sense(int fd, uchar_t pc, int dbd, int page_len,
106 106 uchar_t *buffer);
107 107 uint16_t read_scsi16(void *addr);
108 108 int check_device(device_t *dev, int cond);
109 109 static void get_media_info(device_t *t_dev, char *sdev,
110 110 char *pname, char *sn);
111 111
112 112 extern void process_p_flag(smedia_handle_t handle, int32_t fd);
113 113
114 114 void
115 115 my_perror(char *err_string)
116 116 {
117 117
118 118 int error_no;
119 119 if (errno == 0)
120 120 return;
121 121
122 122 error_no = errno;
123 123 (void) fprintf(stderr, "%s", err_string);
124 124 (void) fprintf(stderr, gettext(" : "));
125 125 errno = error_no;
126 126 perror("");
127 127 }
128 128
129 129 int32_t
130 130 get_confirmation()
131 131 {
132 132 char c;
133 133
134 134 (void) fprintf(stderr, gettext("Do you want to continue? (y/n)"));
135 135 c = getchar();
136 136 if (c == 'y' || c == 'Y')
137 137 return (1);
138 138 else if (c == 'n' || c == 'N')
139 139 return (0);
140 140 else {
141 141 (void) fprintf(stderr, gettext("Invalid choice\n"));
142 142 return (0);
143 143 }
144 144 }
145 145
146 146
147 147 void
148 148 get_passwd(struct smwp_state *wp, int32_t confirm)
149 149 {
150 150 char passwd[256], re_passwd[256];
151 151 int32_t len;
152 152 struct termios tio;
153 153 int32_t echo_off = 0;
154 154 FILE *in, *out;
155 155 char *buf;
156 156
157 157
158 158 in = fopen("/dev/tty", "r+");
159 159 if (in == NULL) {
160 160 in = stdin;
161 161 out = stderr;
162 162 } else {
163 163 out = in;
164 164 }
165 165
166 166 /* Turn echoing off if it is on now. */
167 167
168 168 if (tcgetattr(fileno(in), &tio) < 0) {
169 169 PERROR("Echo off ioctl failed");
170 170 exit(1);
171 171 }
172 172 if (tio.c_lflag & ECHO) {
173 173 tio.c_lflag &= ~ECHO;
174 174 /* echo_off = tcsetattr(fileno(in), TCSAFLUSH, &tio) == 0; */
175 175 echo_off = tcsetattr(fileno(in), TCSAFLUSH, &tio) == 0;
176 176 tio.c_lflag |= ECHO;
177 177 }
178 178
179 179 /* CONSTCOND */
180 180 while (1) {
181 181 (void) fputs(
182 182 gettext("Please enter password (32 chars maximum):"),
183 183 out);
184 184 (void) fflush(out);
185 185 buf = fgets(passwd, (size_t)256, in);
186 186 rewind(in);
187 187 if (buf == NULL) {
188 188 PERROR("Error reading password");
189 189 continue;
190 190 }
191 191 len = strlen(passwd);
192 192 (void) fputc('\n', out);
193 193 len--; /* To offset the \n */
194 194 if ((len <= 0) || (len > 32)) {
195 195 (void) fprintf(stderr,
196 196 gettext("Invalid length of password \n"));
197 197 (void) fputs("Try again\n", out);
198 198 continue;
199 199 }
200 200
201 201 if (!confirm)
202 202 break;
203 203
204 204 (void) fputs("Please reenter password:", out);
205 205 (void) fflush(out);
206 206 buf = fgets(re_passwd, (size_t)256, in);
207 207 rewind(in);
208 208 (void) fputc('\n', out);
209 209 if ((buf == NULL) || strcmp(passwd, re_passwd)) {
210 210 (void) fputs("passwords did not match\n", out);
211 211 (void) fputs("Try again\n", out);
212 212 } else {
213 213 break;
214 214 }
215 215 }
↓ open down ↓ |
115 lines elided |
↑ open up ↑ |
216 216 wp->sm_passwd_len = len;
217 217 (void) strncpy(wp->sm_passwd, passwd, wp->sm_passwd_len);
218 218 wp->sm_version = SMWP_STATE_V_1;
219 219
220 220 /* Restore echoing. */
221 221 if (echo_off)
222 222 (void) tcsetattr(fileno(in), TCSAFLUSH, &tio);
223 223
224 224 }
225 225
226 -int32_t
227 -check_and_unmount_vold(char *device_name, int32_t flag)
228 -{
229 - char *real_name;
230 - char *nm;
231 - char tmp_path_name[PATH_MAX];
232 - struct stat stat_buf;
233 - int32_t ret_val = 0;
234 - struct mnttab *mntp;
235 - FILE *fp;
236 - int nl;
237 -
238 - DPRINTF1("Device name %s\n", device_name);
239 -
240 - if (volmgt_running() == 0) {
241 - DPRINTF("Vold not running\n");
242 - return (0);
243 - }
244 - if ((nm = volmgt_symname(device_name)) == NULL) {
245 - DPRINTF("path not managed\n");
246 - real_name = media_findname(device_name);
247 - } else {
248 - DPRINTF1("path managed as %s\n", nm);
249 - real_name = media_findname(nm);
250 - DPRINTF1("real name %s\n", real_name);
251 - }
252 -
253 - if (real_name == NULL)
254 - return (-1);
255 -
256 - /*
257 - * To find out whether the device has been mounted by
258 - * volume manager...
259 - *
260 - * Convert the real name to a block device address.
261 - * Do a partial match with the mnttab entries.
262 - * Make sure the match is in the beginning to avoid if
263 - * anybody puts a label similiar to volume manager path names.
264 - * Then use "volrmmount -e <dev_name>" if -U flag is set.
265 - */
266 -
267 - nl = strlen("/vol/dev/");
268 -
269 - if (strncmp(real_name, "/vol/dev/", nl) != 0)
270 - return (0);
271 - if (real_name[nl] == 'r') {
272 - (void) snprintf(tmp_path_name, PATH_MAX, "%s%s", "/vol/dev/",
273 - &real_name[nl + 1]);
274 - } else {
275 - (void) snprintf(tmp_path_name, PATH_MAX, "%s", real_name);
276 - }
277 - DPRINTF1("%s \n", tmp_path_name);
278 - ret_val = stat(tmp_path_name, &stat_buf);
279 - if (ret_val < 0) {
280 - PERROR("Could not stat");
281 - return (-1);
282 - }
283 -
284 - fp = fopen("/etc/mnttab", "r");
285 -
286 - if (fp == NULL) {
287 - PERROR("Could not open /etc/mnttab");
288 - return (-1);
289 - }
290 -
291 - mntp = (struct mnttab *)malloc(sizeof (struct mnttab));
292 - if (mntp == NULL) {
293 - PERROR("malloc failed");
294 - (void) fclose(fp);
295 - return (-1);
296 - }
297 - errno = 0;
298 - while (getmntent(fp, mntp) == 0) {
299 - if (errno != 0) {
300 - PERROR("Error with mnttab");
301 - (void) fclose(fp);
302 - return (-1);
303 - }
304 - /* Is it a probable entry? */
305 - DPRINTF1(" %s \n", mntp->mnt_special);
306 - if (strstr(mntp->mnt_special, tmp_path_name) !=
307 - mntp->mnt_special) {
308 - /* Skip to next entry */
309 - continue;
310 - } else {
311 - DPRINTF1("Found!! %s\n", mntp->mnt_special);
312 - ret_val = 1;
313 - break;
314 - }
315 - }
316 -
317 - if (ret_val == 1) {
318 - if (flag) {
319 - if (my_volrmmount(real_name) < 0) {
320 - ret_val = -1;
321 - }
322 - } else {
323 - ret_val = -1;
324 - }
325 - }
326 - (void) fclose(fp);
327 - free(mntp);
328 - return (ret_val);
329 -}
330 -
331 226 /*
332 227 * This routine checks if a device has mounted partitions. The
333 228 * device name is assumed to be /dev/rdsk/cNtNdNsN. So, this can
334 229 * be used for SCSI and PCMCIA cards.
335 230 * Returns
336 231 * 0 : if not mounted
337 232 * 1 : if successfully unmounted
338 233 * -1 : Any error or umount failed
339 234 */
340 235
341 236 int32_t
342 237 check_and_unmount_scsi(char *device_name, int32_t flag)
343 238 {
344 239
345 240 struct mnttab *mntrefp;
346 241 struct mnttab *mntp;
347 242 FILE *fp;
348 243 char block_dev_name[PATH_MAX];
349 244 char tmp_name[PATH_MAX];
350 245 int32_t i, j;
351 246 int32_t unmounted = 0;
352 247
353 248 /*
354 249 * If the device name is not a character special, anyway we
355 250 * can not progress further
356 251 */
357 252
358 253 if (strncmp(device_name, "/dev/rdsk/c", strlen("/dev/rdsk/c")) != 0)
359 254 return (0);
360 255
361 256 (void) snprintf(block_dev_name, PATH_MAX, "/dev/%s",
362 257 &device_name[strlen("/dev/r")]);
363 258 fp = fopen("/etc/mnttab", "r");
364 259
365 260 if (fp == NULL) {
366 261 PERROR("Could not open /etc/mnttab");
367 262 return (-1);
368 263 }
369 264
370 265 mntrefp = (struct mnttab *)malloc(sizeof (struct mnttab));
371 266 if (mntrefp == NULL) {
372 267 PERROR("malloc failed");
373 268 (void) fclose(fp);
374 269 return (-1);
375 270 }
376 271
377 272 mntp = (struct mnttab *)malloc(sizeof (struct mnttab));
378 273 if (mntp == NULL) {
379 274 PERROR("malloc failed");
380 275 (void) fclose(fp);
381 276 free(mntrefp);
382 277 return (-1);
383 278 }
384 279
385 280 /* Try all the partitions */
386 281
387 282 (void) snprintf(tmp_name, PATH_MAX, "/dev/%s",
388 283 &device_name[strlen("/dev/r")]);
389 284
390 285 tmp_name[strlen("/dev/dsk/c0t0d0s")] = '\0';
391 286
392 287 errno = 0;
393 288 while (getmntent(fp, mntp) == 0) {
394 289 if (errno != 0) {
395 290 PERROR("Error with mnttab");
396 291 (void) fclose(fp);
397 292 return (-1);
398 293 }
399 294 /* Is it a probable entry? */
400 295 if (strncmp(mntp->mnt_special, tmp_name, strlen(tmp_name))) {
401 296 /* Skip to next entry */
402 297 continue;
403 298 }
404 299 for (i = 0; i < NDKMAP; i++) {
405 300 /* Check for ufs style mount devices */
406 301 (void) snprintf(block_dev_name, PATH_MAX,
407 302 "%s%d", tmp_name, i);
408 303
409 304 if (strcmp(mntp->mnt_special, block_dev_name) == 0) {
410 305 if (flag) {
411 306 if (my_umount(mntp->mnt_mountp) < 0) {
412 307 (void) fclose(fp);
413 308 return (-1);
414 309 }
415 310 unmounted = 1;
416 311 } else {
417 312 (void) fclose(fp);
418 313 return (-1);
419 314 }
420 315 /* Skip to next entry */
421 316 continue;
422 317 }
423 318
424 319 /* Try for :1 -> :24 for pcfs */
425 320
426 321 for (j = 1; j < 24; j++) {
427 322 (void) snprintf(block_dev_name, PATH_MAX,
428 323 "%s%d:%d", tmp_name, i, j);
429 324
430 325 if (strcmp(mntp->mnt_special,
431 326 block_dev_name) == 0) {
432 327 if (flag) {
433 328 if (my_umount(mntp->mnt_mountp)
434 329 < 0) {
435 330 (void) fclose(fp);
436 331 return (-1);
437 332 }
438 333 unmounted = 1;
439 334 } else {
440 335 (void) fclose(fp);
441 336 return (-1);
442 337 }
443 338 /* Skip to next entry */
444 339 continue;
445 340 }
446 341 (void) snprintf(block_dev_name, PATH_MAX,
447 342 "%s%d:%c", tmp_name, i, 'b' + j);
448 343
449 344 if (strcmp(mntp->mnt_special,
450 345 block_dev_name) == 0) {
451 346 if (flag) {
452 347 if (my_umount(mntp->mnt_mountp)
453 348 < 0) {
454 349 (void) fclose(fp);
455 350 return (-1);
456 351 }
457 352 unmounted = 1;
458 353 } else {
459 354 (void) fclose(fp);
460 355 return (-1);
461 356 }
462 357 /* Skip to next entry */
463 358 continue;
464 359 }
465 360 }
466 361 }
467 362
468 363 }
469 364
470 365 if (unmounted)
471 366 return (1);
472 367 return (0);
473 368 }
474 369
475 370 /*
476 371 * This routine checks if a device has mounted partitions. The
477 372 * device name is assumed to be /dev/rdiskette. So, this can
478 373 * be used for Floppy controllers
479 374 * Returns
480 375 * 0 : if not mounted
481 376 * 1 : if successfully unmounted
482 377 * -1 : Any error or unmount failed
483 378 */
484 379
485 380 int32_t
486 381 check_and_unmount_floppy(int32_t fd, int32_t flag)
487 382 {
488 383 FILE *fp = NULL;
489 384 int32_t mfd;
490 385 struct dk_cinfo dkinfo, dkinfo_tmp;
491 386 struct mnttab mnt_record;
492 387 struct mnttab *mp = &mnt_record;
493 388 struct stat stbuf;
494 389 char raw_device[PATH_MAX];
495 390 int32_t found = 0;
496 391
497 392
498 393 if (ioctl(fd, DKIOCINFO, &dkinfo) < 0) {
499 394 return (-1);
500 395 }
501 396
502 397 if ((fp = fopen(MNTTAB, "r")) == NULL) {
503 398 PERROR("Could not open /etc/mnttab");
504 399 (void) close(fd);
505 400 exit(3);
506 401 }
507 402
508 403 while (getmntent(fp, mp) == 0) {
509 404 if (strstr(mp->mnt_special, "/dev/fd") == NULL &&
510 405 strstr(mp->mnt_special, "/dev/disket") == NULL &&
511 406 strstr(mp->mnt_special, "/dev/c") == NULL) {
512 407 continue;
513 408 }
514 409
515 410 (void) strcpy(raw_device, "/dev/r");
516 411 (void) strcat(raw_device, mp->mnt_special + strlen("/dev/"));
517 412
518 413
519 414 /*
520 415 * Attempt to open the device. If it fails, skip it.
521 416 */
522 417
523 418 /* Turn on the privileges. */
524 419 (void) __priv_bracket(PRIV_ON);
525 420
526 421 mfd = open(raw_device, O_RDWR | O_NDELAY);
527 422
528 423 /* Turn off the privileges. */
529 424 (void) __priv_bracket(PRIV_OFF);
530 425
531 426 if (mfd < 0) {
532 427 continue;
533 428 }
534 429
535 430 /*
536 431 * Must be a character device
537 432 */
538 433 if (fstat(mfd, &stbuf) < 0 || !S_ISCHR(stbuf.st_mode)) {
539 434 (void) close(mfd);
540 435 continue;
541 436 }
542 437 /*
543 438 * Attempt to read the configuration info on the disk.
544 439 */
545 440 if (ioctl(mfd, DKIOCINFO, &dkinfo_tmp) < 0) {
546 441 (void) close(mfd);
547 442 continue;
548 443 }
549 444 /*
550 445 * Finished with the opened device
551 446 */
552 447 (void) close(mfd);
553 448
554 449 /*
555 450 * If it's not the disk we're interested in, it doesn't apply.
556 451 */
557 452 if (dkinfo.dki_ctype != dkinfo_tmp.dki_ctype ||
558 453 dkinfo.dki_cnum != dkinfo_tmp.dki_cnum ||
559 454 dkinfo.dki_unit != dkinfo_tmp.dki_unit) {
560 455 continue;
561 456 }
562 457 /*
563 458 * It's a mount on the disk we're checking. If we are
564 459 * checking whole disk, then we found trouble. We can
565 460 * quit searching.
566 461 */
567 462
568 463 if (flag) {
569 464 if (my_umount(mp->mnt_mountp) < 0) {
570 465 return (-1);
571 466 }
572 467 found = 1;
573 468 } else {
574 469 return (-1);
575 470 }
576 471 }
577 472 return (found);
578 473 }
579 474
580 475
581 476 int32_t
582 477 my_open(char *device_name, int32_t flags)
583 478 {
584 479 char *real_name;
585 480 char *nm;
586 481 char tmp_path_name[PATH_MAX];
587 482 struct stat stat_buf;
588 483 int32_t ret_val;
589 484 int32_t fd;
590 485 int32_t have_read_priv = 0;
591 486 DIR *dirp;
592 487 struct dirent *dp;
593 488
594 489 DPRINTF1("Device name %s\n", device_name);
595 490
596 491 if ((nm = volmgt_symname(device_name)) == NULL) {
597 492 DPRINTF("path not managed\n");
598 493 real_name = media_findname(device_name);
599 494 } else {
600 495 DPRINTF1("path managed as %s\n", nm);
601 496 real_name = media_findname(nm);
602 497 DPRINTF1("real name %s\n", real_name);
603 498 }
604 499
605 500 if (real_name == NULL)
606 501 return (-1);
607 502
608 503 (void) strcpy(tmp_path_name, real_name);
609 504 ret_val = stat(tmp_path_name, &stat_buf);
610 505 if (ret_val < 0) {
611 506 PERROR("Could not stat");
612 507 return (-1);
613 508 }
614 509 if (S_ISDIR(stat_buf.st_mode)) {
615 510
616 511 /*
617 512 * Open the directory and look for the
618 513 * first non '.' entry.
619 514 * Since raw_read and raw_writes are used, we don't
620 515 * need to access the backup slice.
621 516 * For PCMCIA Memory cards, raw_read and raw_writes are
622 517 * not supported, but that is not a problem as, only slice2
623 518 * is allowed on PCMCIA memory cards.
624 519 */
625 520
626 521 /*
627 522 * First make sure we are operating with a /vol/....
628 523 * Otherwise it can dangerous,
629 524 * e.g. rmformat -s /dev/rdsk
630 525 * We should not look into the directory contents here.
631 526 */
632 527 if (strncmp(tmp_path_name, "/vol/dev/", strlen("/vol/dev/"))
633 528 != 0) {
634 529 (void) fprintf(stderr, gettext("The specified device \
635 530 is not a raw device.\n"));
636 531 exit(1);
637 532 }
638 533
639 534 /* Turn on the privileges. */
640 535 (void) __priv_bracket(PRIV_ON);
641 536
642 537 dirp = opendir(tmp_path_name);
643 538
644 539 /* Turn off the privileges. */
645 540 (void) __priv_bracket(PRIV_OFF);
646 541
647 542 if (dirp == NULL) {
648 543 return (-1);
649 544 }
650 545
651 546 /* Turn on the privileges. */
652 547 (void) __priv_bracket(PRIV_ON);
653 548 have_read_priv = 1;
654 549
655 550 while ((dp = readdir(dirp)) != NULL) {
656 551
657 552 /* Turn off the privileges. */
658 553 (void) __priv_bracket(PRIV_OFF);
659 554 have_read_priv = 0;
660 555
661 556 DPRINTF1("Found %s\n", dp->d_name);
662 557 if ((strcmp(dp->d_name, ".") != 0) &&
663 558 (strcmp(dp->d_name, "..") != 0)) {
664 559 (void) snprintf(tmp_path_name, PATH_MAX,
665 560 "%s/%s", tmp_path_name, dp->d_name);
666 561
667 562 DPRINTF1("tmp_pathname is %s\n", tmp_path_name);
668 563 break;
669 564 }
670 565
671 566 /* Turn on the privileges. */
672 567 (void) __priv_bracket(PRIV_ON);
673 568 have_read_priv = 1;
674 569 }
675 570
676 571 if (have_read_priv) {
677 572 /* drop the file_dac_read privilege */
678 573 (void) __priv_bracket(PRIV_OFF);
679 574 have_read_priv = 0;
680 575 }
681 576
682 577 (void) closedir(dirp);
683 578 }
684 579
685 580
686 581 if (volmgt_running() == 0) {
687 582 /* Turn on privileges. */
688 583 (void) __priv_bracket(PRIV_ON);
689 584 have_read_priv = 1;
690 585 }
691 586
692 587 fd = open(tmp_path_name, flags);
693 588
694 589 if (have_read_priv) {
695 590 /* Turn off privileges. */
696 591 (void) __priv_bracket(PRIV_OFF);
697 592 have_read_priv = 0;
698 593 }
699 594
700 595 DPRINTF1("path opened %s\n", tmp_path_name);
701 596
702 597 return (fd);
703 598 }
704 599
705 600 uint64_t
706 601 my_atoll(char *ptr)
707 602 {
708 603 char *tmp_ptr = ptr;
709 604 int32_t base = 10;
710 605 uint64_t ret_val;
711 606
712 607 while (*tmp_ptr) {
713 608 if (isdigit(*tmp_ptr))
714 609 tmp_ptr++;
715 610 else {
716 611 base = 16;
717 612 break;
718 613 }
719 614 }
720 615 tmp_ptr = ptr;
721 616 if (base == 16) {
722 617 if (strlen(tmp_ptr) < 3) {
723 618 return (-1);
724 619 }
725 620 if (*tmp_ptr++ != '0' || (*tmp_ptr != 'x' && *tmp_ptr != 'X')) {
726 621 return (-1);
727 622 }
728 623 tmp_ptr++;
729 624 while (*tmp_ptr) {
730 625 if (isxdigit(*tmp_ptr))
731 626 tmp_ptr++;
732 627 else {
733 628 return (-1);
734 629 }
735 630 }
736 631 }
737 632 ret_val = (uint64_t)strtoull(ptr, (char **)NULL, 0);
738 633 return (ret_val);
739 634 }
740 635
741 636 int32_t
742 637 write_sunos_label(int32_t fd, int32_t media_type)
743 638 {
744 639
745 640 struct extvtoc v_toc;
746 641 int32_t ret;
747 642
748 643 (void) memset(&v_toc, 0, sizeof (struct extvtoc));
749 644
750 645 /* Initialize the vtoc information */
751 646
752 647 if (media_type == SM_FLOPPY) {
753 648 struct fd_char fdchar;
754 649 int32_t mult_factor;
755 650
756 651 if (ioctl(fd, FDIOGCHAR, &fdchar) < 0) {
757 652 PERROR("FDIOGCHAR failed");
758 653 return (-1);
759 654 }
760 655
761 656 /* SPARC and x86 fd drivers use fdc_medium differently */
762 657 #if defined(__sparc)
763 658 mult_factor = (fdchar.fdc_medium) ? 2 : 1;
764 659 #elif defined(__x86)
765 660 mult_factor = (fdchar.fdc_medium == 5) ? 2 : 1;
766 661 #else
767 662 #error No Platform defined
768 663 #endif /* defined(__sparc) */
769 664
770 665 /* initialize the vtoc structure */
771 666 v_toc.v_nparts = 3;
772 667
773 668 v_toc.v_part[0].p_start = 0;
774 669 v_toc.v_part[0].p_size = (fdchar.fdc_ncyl - 1) * 2 *
775 670 fdchar.fdc_secptrack * mult_factor;
776 671 v_toc.v_part[1].p_start = (fdchar.fdc_ncyl - 1) * 2 *
777 672 fdchar.fdc_secptrack * mult_factor;
778 673 v_toc.v_part[1].p_size = 2 * fdchar.fdc_secptrack * mult_factor;
779 674
780 675 v_toc.v_part[2].p_start = 0;
781 676 v_toc.v_part[2].p_size = fdchar.fdc_ncyl * 2 *
782 677 fdchar.fdc_secptrack * mult_factor;
783 678
784 679 } else if (media_type == SM_SCSI_FLOPPY) {
785 680
786 681 smedia_handle_t handle;
787 682 smmedium_prop_t med_info;
788 683 struct dk_geom dkgeom;
789 684
790 685
791 686 /*
792 687 * call smedia_get_medium_property to get the
793 688 * correct media information, since DKIOCGMEDIAINFO
794 689 * may fail for unformatted media.
795 690 */
796 691
797 692 handle = smedia_get_handle(fd);
798 693 if (handle == NULL) {
799 694 (void) fprintf(stderr,
800 695 gettext("Failed to get libsmedia handle.\n"));
801 696
802 697 (void) close(fd);
803 698 return (-1);
804 699 }
805 700
806 701
807 702 if (smedia_get_medium_property(handle, &med_info) < 0) {
808 703 (void) fprintf(stderr,
809 704 gettext("Get medium property failed \n"));
810 705
811 706 (void) smedia_release_handle(handle);
812 707 (void) close(fd);
813 708 return (-1);
814 709 }
815 710
816 711 /* Fill in our own geometry information */
817 712
818 713 dkgeom.dkg_pcyl = med_info.sm_pcyl;
819 714 dkgeom.dkg_ncyl = med_info.sm_pcyl;
820 715 dkgeom.dkg_nhead = med_info.sm_nhead;
821 716 dkgeom.dkg_nsect = med_info.sm_nsect;
822 717 dkgeom.dkg_acyl = 0;
823 718 dkgeom.dkg_bcyl = 0;
824 719 dkgeom.dkg_intrlv = 0;
825 720 dkgeom.dkg_apc = 0;
826 721
827 722 /*
828 723 * Try to set vtoc, if not successful we will
829 724 * continue to use the faked geometry information.
830 725 */
831 726
832 727 (void) ioctl(fd, DKIOCSGEOM, &dkgeom);
833 728
834 729 (void) smedia_release_handle(handle);
835 730
836 731 /* we want the same partitioning as used for normal floppies */
837 732
838 733 v_toc.v_part[0].p_start = 0;
839 734 v_toc.v_part[0].p_size = (diskaddr_t)(dkgeom.dkg_ncyl - 1) *
840 735 dkgeom.dkg_nhead * dkgeom.dkg_nsect;
841 736
842 737 v_toc.v_part[1].p_start = (diskaddr_t)(dkgeom.dkg_ncyl - 1) *
843 738 dkgeom.dkg_nhead * dkgeom.dkg_nsect;
844 739 v_toc.v_part[1].p_size = dkgeom.dkg_nhead * dkgeom.dkg_nsect;
845 740
846 741 v_toc.v_part[2].p_start = 0;
847 742 v_toc.v_part[2].p_size = (diskaddr_t)dkgeom.dkg_ncyl *
848 743 dkgeom.dkg_nhead * dkgeom.dkg_nsect;
849 744
850 745 /* both write_vtoc and DKIOCSVTOC require V_NUMPAR partitions */
851 746 v_toc.v_nparts = V_NUMPAR;
852 747
853 748 } else {
854 749
855 750 return (0);
856 751 }
857 752
858 753 v_toc.v_sanity = VTOC_SANE;
859 754 v_toc.v_version = V_VERSION;
860 755
861 756 /*
862 757 * The label structure is set up for DEV_BSIZE(512 byte) blocks,
863 758 * even though a medium density diskette has 1024 byte blocks
864 759 * See dklabel.h for more details.
865 760 */
866 761 v_toc.v_sectorsz = DEV_BSIZE;
867 762
868 763 /* let the fd driver finish constructing the label and writing it. */
869 764
870 765
871 766 /* Turn on the privileges. */
872 767 (void) __priv_bracket(PRIV_ON);
873 768
874 769 ret = write_extvtoc(fd, &v_toc);
875 770
876 771 /* Turn off the privileges. */
877 772 (void) __priv_bracket(PRIV_OFF);
878 773
879 774 if (ret < 0) {
880 775 PERROR("Write vtoc");
881 776 DPRINTF1("Write vtoc failed errno:%d\n", errno);
882 777 return (-1);
883 778 }
884 779
885 780 return (0);
886 781 }
887 782
888 783 static void
889 784 intr_sig_handler()
890 785 {
891 786 char c;
892 787
893 788 (void) fprintf(stderr, gettext(global_intr_msg));
894 789 (void) fprintf(stderr,
895 790 gettext("\nDo you want to stop formatting?(y/n)"));
896 791 (void) fflush(stdout);
897 792 rewind(stdin);
898 793 while ((c = getchar()) == -1)
899 794 ;
900 795 if (c == 'y' || c == 'Y') {
901 796 (void) fprintf(stderr, gettext("Format interrupted\n"));
902 797 exit(1);
903 798 } else if (c == 'n' || c == 'N')
904 799 return;
905 800 else {
906 801 (void) fprintf(stderr, gettext("Did not interrupt\n"));
907 802 return;
908 803 }
909 804 }
910 805
911 806 static struct sigaction act, oact;
912 807 void
913 808 trap_SIGINT()
914 809 {
915 810
916 811 act.sa_handler = intr_sig_handler;
917 812 (void) memset(&act.sa_mask, 0, sizeof (sigset_t));
918 813 act.sa_flags = SA_RESTART; /* | SA_NODEFER; */
919 814 if (sigaction(SIGINT, &act, &oact) < 0) {
920 815 DPRINTF("sigset failed\n");
921 816 return;
922 817 }
923 818 }
924 819
925 820 void
926 821 release_SIGINT()
927 822 {
928 823 if (sigaction(SIGINT, &oact, (struct sigaction *)NULL) < 0) {
929 824 DPRINTF("sigunset failed\n");
930 825 return;
931 826 }
932 827 }
933 828
934 829 int32_t
935 830 verify(smedia_handle_t handle, int32_t fd, diskaddr_t start_sector,
936 831 uint32_t nblocks, char *buf,
937 832 int32_t flag, int32_t blocksize, int32_t no_raw_rw)
938 833 {
939 834 uint64_t ret;
940 835
941 836 DPRINTF("ANALYSE MEDIA \n");
942 837
943 838
944 839 if ((flag == VERIFY_READ) && (!no_raw_rw)) {
945 840
946 841 /* Turn on the privileges. */
947 842 (void) __priv_bracket(PRIV_ON);
948 843
949 844 ret = smedia_raw_read(handle, start_sector, buf, nblocks *
950 845 blocksize);
951 846
952 847 /* Turn off the privileges. */
953 848 (void) __priv_bracket(PRIV_OFF);
954 849
955 850 if (ret != (nblocks * blocksize))
956 851 return (-1);
957 852 return (0);
958 853
959 854 } else if ((flag == VERIFY_WRITE) && (!no_raw_rw)) {
960 855
961 856 /* Turn on privileges. */
962 857 (void) __priv_bracket(PRIV_ON);
963 858
964 859 ret = smedia_raw_write(handle, start_sector, buf, nblocks *
965 860 blocksize);
966 861
967 862 /* Turn off the privileges. */
968 863 (void) __priv_bracket(PRIV_OFF);
969 864
970 865 if (ret != (blocksize * nblocks))
971 866 return (-1);
972 867 return (0);
973 868
974 869 } else if ((flag == VERIFY_READ) && (no_raw_rw)) {
975 870 ret = llseek(fd, start_sector * blocksize, SEEK_SET);
976 871 if (ret != start_sector * blocksize) {
977 872 (void) fprintf(stderr, gettext("Seek failed\n"));
978 873 return (-2);
979 874 }
980 875
981 876 /* Turn on the privileges. */
982 877 (void) __priv_bracket(PRIV_ON);
983 878
984 879 ret = read(fd, buf, nblocks * blocksize);
985 880
986 881 /* Turn off the privileges. */
987 882 (void) __priv_bracket(PRIV_OFF);
988 883
989 884 if (ret != nblocks * blocksize) {
990 885 return (-1);
991 886 }
992 887 return (0);
993 888 } else if ((flag == VERIFY_WRITE) && (no_raw_rw)) {
994 889 ret = llseek(fd, start_sector * blocksize, SEEK_SET);
995 890 if (ret != start_sector * blocksize) {
996 891 (void) fprintf(stderr, gettext("Seek failed\n"));
997 892 return (-2);
998 893 }
999 894
1000 895 /* Turn on the privileges. */
1001 896 (void) __priv_bracket(PRIV_ON);
1002 897
1003 898 ret = write(fd, buf, nblocks * blocksize);
1004 899
1005 900 /* Turn off the privileges. */
1006 901 (void) __priv_bracket(PRIV_OFF);
1007 902
1008 903 if (ret != nblocks * blocksize) {
1009 904 return (-1);
1010 905 }
1011 906 return (0);
1012 907 } else {
1013 908 DPRINTF("Illegal parameter to verify_analysis!\n");
1014 909 return (-1);
1015 910 }
1016 911 }
1017 912
1018 913 static int
1019 914 my_umount(char *mountp)
1020 915 {
1021 916 pid_t pid; /* forked proc's pid */
1022 917 int rval; /* proc's return value */
1023 918
1024 919
1025 920 /* create a child to unmount the path */
1026 921
1027 922 /* Turn on the privileges */
1028 923 (void) __priv_bracket(PRIV_ON);
1029 924
1030 925 pid = fork();
1031 926
1032 927 /* Turn off the privileges. */
1033 928 (void) __priv_bracket(PRIV_OFF);
1034 929
1035 930 if (pid < 0) {
1036 931 PERROR("fork failed");
1037 932 exit(0);
1038 933 }
1039 934
1040 935 if (pid == 0) {
1041 936 /* the child */
1042 937 /* get rid of those nasty err messages */
1043 938 DPRINTF1("call_unmount_prog: calling %s \n", mountp);
1044 939
1045 940 /* Turn on the priviliges. */
1046 941 (void) __priv_bracket(PRIV_ON);
1047 942
1048 943 if (execl("/usr/sbin/umount", "/usr/sbin/umount", mountp,
1049 944 NULL) < 0) {
1050 945 perror("exec failed");
1051 946 /* Turn off the privileges */
1052 947 (void) __priv_bracket(PRIV_OFF);
1053 948 exit(-1);
1054 949 }
1055 950 }
1056 951
1057 952 /* wait for the umount command to exit */
1058 953 rval = 0;
1059 954 if (waitpid(pid, &rval, 0) == pid) {
↓ open down ↓ |
719 lines elided |
↑ open up ↑ |
1060 955 if (WIFEXITED(rval)) {
1061 956 if (WEXITSTATUS(rval) == 0) {
1062 957 DPRINTF("umount : Success\n");
1063 958 return (1);
1064 959 }
1065 960 }
1066 961 }
1067 962 return (-1);
1068 963 }
1069 964
1070 -static int
1071 -my_volrmmount(char *real_name)
1072 -{
1073 - int pid, rval;
1074 -
1075 - /* Turn on the privileges. */
1076 - (void) __priv_bracket(PRIV_ON);
1077 -
1078 - pid = fork();
1079 -
1080 - /* Turn off the privileges. */
1081 - (void) __priv_bracket(PRIV_OFF);
1082 -
1083 - /* create a child to unmount the path */
1084 - if (pid < 0) {
1085 - PERROR("fork failed");
1086 - exit(0);
1087 - }
1088 -
1089 - if (pid == 0) {
1090 - /* the child */
1091 - /* get rid of those nasty err messages */
1092 - DPRINTF1("call_unmount_prog: calling %s \n",
1093 - "/usr/bin/volrmmount");
1094 -
1095 - /* Turn on the privileges. */
1096 - (void) __priv_bracket(PRIV_ON);
1097 - if (execl("/usr/bin/volrmmount", "/usr/bin/volrmmount", "-e",
1098 - real_name, NULL) < 0) {
1099 - PERROR("volrmmount exec failed");
1100 - /* Turn off the privileges */
1101 - (void) __priv_bracket(PRIV_OFF);
1102 - exit(-1);
1103 - }
1104 - } else if (waitpid(pid, &rval, 0) == pid) {
1105 - if (WIFEXITED(rval)) {
1106 - if (WEXITSTATUS(rval) == 0) {
1107 - DPRINTF("volrmmount: Success\n");
1108 - return (1);
1109 - }
1110 - }
1111 - }
1112 - return (-1);
1113 -}
1114 -
1115 965 int
1116 966 find_device(int defer, char *tmpstr)
1117 967 {
1118 968 DIR *dir;
1119 969 struct dirent *dirent;
1120 970 char sdev[PATH_MAX], dev[PATH_MAX], *pname;
1121 971 device_t *t_dev;
1122 972 int removable = 0;
1123 973 int device_type = 0;
1124 974 int hotpluggable = 0;
1125 975 struct dk_minfo mediainfo;
1126 976 static int found = 0;
1127 977
1128 978 dir = opendir("/dev/rdsk");
1129 979 if (dir == NULL)
1130 980 return (-1);
1131 981
1132 982 total_devices_found = 0;
1133 983 while ((dirent = readdir(dir)) != NULL) {
1134 984 if (dirent->d_name[0] == '.') {
1135 985 continue;
1136 986 }
1137 987 (void) snprintf(sdev, PATH_MAX, "/dev/rdsk/%s",
1138 988 dirent->d_name);
1139 989 #ifdef sparc
1140 990 if (!strstr(sdev, "s2")) {
1141 991 continue;
1142 992 }
1143 993 #else /* x86 */
1144 994 if (vol_running) {
1145 995 if (!(strstr(sdev, "s2") || strstr(sdev, "p0"))) {
1146 996 continue;
1147 997 }
1148 998 } else {
1149 999 if (!strstr(sdev, "p0")) {
1150 1000 continue;
1151 1001 }
1152 1002 }
↓ open down ↓ |
28 lines elided |
↑ open up ↑ |
1153 1003 #endif
1154 1004 if (!lookup_device(sdev, dev)) {
1155 1005 continue;
1156 1006 }
1157 1007 if ((t_dev = get_device(NULL, dev)) == NULL) {
1158 1008 continue;
1159 1009 }
1160 1010 total_devices_found++;
1161 1011
1162 1012 if ((!defer) && !found) {
1163 - char *sn, *tmpbuf;
1013 + char *sn, *tmpbuf = NULL;
1164 1014 /*
1165 1015 * dev_name is an optional command line input.
1166 1016 */
1167 1017 if (dev_name) {
1168 1018 if (strstr(dirent->d_name, tmpstr)) {
1169 1019 found = 1;
1170 1020 } else if (!vol_running) {
1171 1021 continue;
1172 1022 }
1173 1023 }
1174 1024 /*
1175 1025 * volmgt_symname() returns NULL if the device
1176 1026 * is not managed by volmgt.
1177 1027 */
1178 1028 sn = volmgt_symname(sdev);
1179 1029
1180 1030 if (vol_running && (sn != NULL)) {
1181 1031 if (strstr(sn, "dev") == NULL) {
1182 1032 tmpbuf = (char *)my_zalloc(PATH_MAX);
1183 1033 (void) strcpy(tmpbuf,
1184 1034 "/vol/dev/aliases/");
1185 1035 (void) strcat(tmpbuf, sn);
1186 1036 free(sn);
1187 1037 sn = tmpbuf;
1188 1038 }
1189 1039 if (dev_name && !found) {
1190 1040 if (!strstr(tmpbuf, tmpstr)) {
1191 1041 continue;
1192 1042 } else {
1193 1043 found = 1;
1194 1044 }
1195 1045 }
1196 1046 }
1197 1047
1198 1048 /*
1199 1049 * Get device type information for CD/DVD devices.
1200 1050 */
1201 1051 if (is_cd(dev)) {
1202 1052 if (check_device(t_dev,
1203 1053 CHECK_DEVICE_IS_DVD_WRITABLE)) {
1204 1054 device_type = DK_DVDR;
1205 1055 } else if (check_device(t_dev,
1206 1056 CHECK_DEVICE_IS_DVD_READABLE)) {
1207 1057 device_type = DK_DVDROM;
1208 1058 } else if (check_device(t_dev,
1209 1059 CHECK_DEVICE_IS_CD_WRITABLE)) {
1210 1060 device_type = DK_CDR;
1211 1061 } else {
1212 1062 device_type = DK_CDROM;
1213 1063 }
1214 1064 } else {
1215 1065 device_type = ioctl(t_dev->d_fd,
1216 1066 DKIOCGMEDIAINFO, &mediainfo);
1217 1067 if (device_type < 0)
1218 1068 device_type = 0;
1219 1069 else
1220 1070 device_type = mediainfo.dki_media_type;
1221 1071 }
1222 1072
1223 1073 if (!ioctl(t_dev->d_fd, DKIOCREMOVABLE, &removable) &&
1224 1074 !ioctl(t_dev->d_fd, DKIOCHOTPLUGGABLE,
1225 1075 &hotpluggable)) {
1226 1076 if (removable || hotpluggable) {
1227 1077 removable_found++;
1228 1078 pname = get_physical_name(sdev);
1229 1079 if (sn) {
1230 1080 (void) printf(" %4d. "
1231 1081 "Volmgt Node: %s\n",
1232 1082 removable_found, sn);
1233 1083 (void) printf(" "
1234 1084 "Logical Node: %s\n", sdev);
1235 1085 (void) printf(" "
1236 1086 "Physical Node: %s\n",
1237 1087 pname);
1238 1088 } else {
1239 1089 (void) printf(" %4d. "
1240 1090 "Logical Node: %s\n",
1241 1091 removable_found, sdev);
1242 1092 (void) printf(" "
1243 1093 "Physical Node: %s\n",
1244 1094 pname);
1245 1095 }
1246 1096 (void) printf(" Connected "
1247 1097 "Device: %-8.8s %-16.16s "
1248 1098 "%-4.4s\n",
1249 1099 &t_dev->d_inq[8],
1250 1100 &t_dev->d_inq[16],
1251 1101 &t_dev->d_inq[32]);
1252 1102 (void) printf(" Device "
1253 1103 "Type: ");
1254 1104 } else
1255 1105 continue;
1256 1106 } else
1257 1107 continue;
1258 1108
1259 1109 switch (device_type) {
1260 1110 case DK_CDROM:
1261 1111 (void) printf("CD Reader\n");
1262 1112 break;
1263 1113 case DK_CDR:
1264 1114 case DK_CDRW:
1265 1115 (void) printf("CD Reader/Writer\n");
1266 1116 break;
1267 1117 case DK_DVDROM:
1268 1118 (void) printf("DVD Reader\n");
1269 1119 break;
1270 1120 case DK_DVDR:
1271 1121 case DK_DVDRAM:
1272 1122 (void) printf("DVD Reader/Writer\n");
1273 1123 break;
1274 1124 case DK_FIXED_DISK:
1275 1125 if (strstr((const char *)
1276 1126 &t_dev->d_inq[16], "FD") ||
1277 1127 strstr((const char *)
1278 1128 &t_dev->d_inq[16], "LS-120"))
1279 1129 (void) printf("Floppy "
1280 1130 "drive\n");
1281 1131 else
1282 1132 (void) printf("Removable\n");
1283 1133 break;
1284 1134 case DK_FLOPPY:
1285 1135 (void) printf("Floppy drive\n");
1286 1136 break;
1287 1137 case DK_ZIP:
1288 1138 (void) printf("Zip drive\n");
1289 1139 break;
1290 1140 case DK_JAZ:
1291 1141 (void) printf("Jaz drive\n");
1292 1142 break;
1293 1143 default:
1294 1144 (void) printf("<Unknown>\n");
1295 1145 DPRINTF1("\t %d\n", device_type);
1296 1146 break;
1297 1147 }
1298 1148 get_media_info(t_dev, sdev, pname, sn);
1299 1149 }
1300 1150 fini_device(t_dev);
1301 1151 }
1302 1152
1303 1153 (void) closedir(dir);
1304 1154 return (removable_found);
1305 1155 }
1306 1156
1307 1157 /*
1308 1158 * Returns a device_t handle for a node returned by lookup_device()
1309 1159 * and takes the user supplied name and stores it inside the node.
1310 1160 */
1311 1161 static device_t *
1312 1162 get_device(char *user_supplied, char *node)
1313 1163 {
1314 1164 device_t *dev;
1315 1165 int fd;
1316 1166 char devnode[PATH_MAX];
1317 1167 int size;
1318 1168
1319 1169 /*
1320 1170 * we need to resolve any link paths to avoid fake files
1321 1171 * such as /dev/rdsk/../../export/file.
1322 1172 */
1323 1173 size = resolvepath(node, devnode, PATH_MAX);
1324 1174 if ((size <= 0) || (size >= (PATH_MAX - 1)))
1325 1175 return (NULL);
1326 1176
1327 1177 /* resolvepath may not return a null terminated string */
1328 1178 devnode[size] = '\0';
1329 1179
1330 1180
1331 1181 /* the device node must be in /devices/ or /vol/dev/rdsk */
1332 1182
1333 1183 if ((strncmp(devnode, "/devices/", 9) != 0) &&
1334 1184 (strncmp(devnode, "/vol/dev/rdsk", 13) != 0))
1335 1185 return (NULL);
1336 1186
1337 1187 /* Turn on the privileges. */
1338 1188 (void) __priv_bracket(PRIV_ON);
1339 1189
1340 1190 /*
1341 1191 * Since we are currently running with the user euid it is
1342 1192 * safe to try to open the file without checking access.
1343 1193 */
1344 1194
1345 1195 fd = open(devnode, O_RDONLY|O_NDELAY);
1346 1196
1347 1197 /* Turn off the privileges. */
1348 1198 (void) __priv_bracket(PRIV_OFF);
1349 1199
1350 1200 if (fd < 0) {
1351 1201 return (NULL);
1352 1202 }
1353 1203
1354 1204 dev = (device_t *)my_zalloc(sizeof (device_t));
1355 1205
1356 1206 dev->d_node = (char *)my_zalloc(strlen(devnode) + 1);
1357 1207 (void) strcpy(dev->d_node, devnode);
1358 1208
1359 1209 dev->d_fd = fd;
1360 1210
1361 1211 dev->d_inq = (uchar_t *)my_zalloc(INQUIRY_DATA_LENGTH);
1362 1212
1363 1213 /* Turn on privileges. */
1364 1214 (void) __priv_bracket(PRIV_ON);
1365 1215 if (!inquiry(fd, dev->d_inq)) {
1366 1216 DPRINTF1("USCSI ioctl failed %d\n",
1367 1217 uscsi_error);
1368 1218 free(dev->d_inq);
1369 1219 free(dev->d_node);
1370 1220 (void) close(dev->d_fd);
1371 1221 free(dev);
1372 1222 /* Turn off privileges. */
1373 1223 (void) __priv_bracket(PRIV_OFF);
1374 1224 return (NULL);
1375 1225 }
1376 1226 /* Turn off privileges. */
1377 1227 (void) __priv_bracket(PRIV_OFF);
1378 1228
1379 1229 if (user_supplied) {
1380 1230 dev->d_name = (char *)my_zalloc(strlen(user_supplied) + 1);
1381 1231 (void) strcpy(dev->d_name, user_supplied);
1382 1232 }
1383 1233 return (dev);
1384 1234 }
1385 1235
1386 1236 /*
1387 1237 * Check for device specific characteristics.
1388 1238 */
1389 1239 int
1390 1240 check_device(device_t *dev, int cond)
1391 1241 {
1392 1242 uchar_t page_code[4];
1393 1243
1394 1244 /* Look at the capabilities page for this information */
1395 1245 if (cond & CHECK_DEVICE_IS_CD_WRITABLE) {
1396 1246 if (get_mode_page(dev->d_fd, 0x2a, 0, 4, page_code) &&
1397 1247 (page_code[3] & 1)) {
1398 1248 return (1);
1399 1249 }
1400 1250 }
1401 1251
1402 1252 if (cond & CHECK_DEVICE_IS_DVD_WRITABLE) {
1403 1253 if (get_mode_page(dev->d_fd, 0x2a, 0, 4, page_code) &&
1404 1254 (page_code[3] & 0x10)) {
1405 1255 return (1);
1406 1256 }
1407 1257 }
1408 1258
1409 1259 if (cond & CHECK_DEVICE_IS_DVD_READABLE) {
1410 1260 if (get_mode_page(dev->d_fd, 0x2a, 0, 4, page_code) &&
1411 1261 (page_code[2] & 0x8)) {
1412 1262 return (1);
1413 1263 }
1414 1264 }
1415 1265
1416 1266 return (0);
1417 1267 }
1418 1268
1419 1269 /*
1420 1270 * Builds an open()able device path from a user supplied node which can be
1421 1271 * of the * form of /dev/[r]dsk/cxtxdx[sx] or cxtxdx[sx] or volmgt-name like
1422 1272 * cdrom[n].
1423 1273 * Returns the path found in 'found' and returns 1. Otherwise returns 0.
1424 1274 */
1425 1275 int
1426 1276 lookup_device(char *supplied, char *found)
1427 1277 {
1428 1278 struct stat statbuf;
1429 1279 int fd;
1430 1280 char tmpstr[PATH_MAX];
1431 1281
1432 1282 /* Turn on privileges */
1433 1283 (void) __priv_bracket(PRIV_ON);
1434 1284
1435 1285 /* If everything is fine and proper, no need to analyze */
1436 1286 if ((stat(supplied, &statbuf) == 0) && S_ISCHR(statbuf.st_mode) &&
1437 1287 ((fd = open(supplied, O_RDONLY|O_NDELAY)) >= 0)) {
1438 1288 (void) close(fd);
1439 1289 (void) strlcpy(found, supplied, PATH_MAX);
1440 1290 /* Turn off privilege */
1441 1291 (void) __priv_bracket(PRIV_OFF);
1442 1292 return (1);
1443 1293 }
1444 1294
1445 1295 /* Turn off privileges. */
1446 1296 (void) __priv_bracket(PRIV_OFF);
1447 1297
1448 1298 if (strncmp(supplied, "/dev/rdsk/", 10) == 0)
1449 1299 return (vol_lookup(supplied, found));
1450 1300 if (strncmp(supplied, "/dev/dsk/", 9) == 0) {
1451 1301 (void) snprintf(tmpstr, PATH_MAX, "/dev/rdsk/%s",
1452 1302 (char *)strrchr(supplied, '/'));
1453 1303
1454 1304 if ((fd = open(tmpstr, O_RDONLY|O_NDELAY)) >= 0) {
1455 1305 (void) close(fd);
1456 1306 (void) strlcpy(found, supplied, PATH_MAX);
1457 1307 return (1);
1458 1308 }
1459 1309 if ((access(tmpstr, F_OK) == 0) && vol_running)
1460 1310 return (vol_lookup(tmpstr, found));
1461 1311 else
1462 1312 return (0);
1463 1313 }
1464 1314 if ((strncmp(supplied, "cdrom", 5) != 0) &&
1465 1315 (strlen(supplied) < 32)) {
1466 1316 (void) snprintf(tmpstr, sizeof (tmpstr), "/dev/rdsk/%s",
1467 1317 supplied);
1468 1318 if (access(tmpstr, F_OK) < 0) {
1469 1319 (void) strcat(tmpstr, "s2");
1470 1320 }
1471 1321 if ((fd = open(tmpstr, O_RDONLY|O_NDELAY)) >= 0) {
1472 1322 (void) close(fd);
1473 1323 (void) strlcpy(found, tmpstr, PATH_MAX);
1474 1324 return (1);
1475 1325 }
1476 1326 if ((access(tmpstr, F_OK) == 0) && vol_running)
1477 1327 return (vol_lookup(tmpstr, found));
1478 1328 }
1479 1329 return (vol_name_to_dev_node(supplied, found));
1480 1330 }
1481 1331
1482 1332 int
1483 1333 is_cd(char *node)
1484 1334 {
1485 1335 int fd;
1486 1336 struct dk_cinfo cinfo;
1487 1337
1488 1338 fd = open(node, O_RDONLY|O_NDELAY);
1489 1339 if (fd < 0)
1490 1340 return (0);
1491 1341 if (ioctl(fd, DKIOCINFO, &cinfo) < 0) {
1492 1342 (void) close(fd);
1493 1343 return (0);
1494 1344 }
1495 1345 if (cinfo.dki_ctype != DKC_CDROM)
1496 1346 return (0);
1497 1347 return (1);
1498 1348 }
1499 1349
1500 1350 void
1501 1351 print_header(void)
1502 1352 {
1503 1353 /* l10n_NOTE : Column spacing should be kept same */
1504 1354 (void) printf(gettext(" Node "
1505 1355 "Connected Device"));
1506 1356 /* l10n_NOTE : Column spacing should be kept same */
1507 1357 (void) printf(gettext(" Device type\n"));
1508 1358 (void) printf(
1509 1359 "---------------------------+---------------------------");
1510 1360 (void) printf("-----+----------------\n");
1511 1361 }
1512 1362
1513 1363 void
1514 1364 print_divider(void)
1515 1365 {
1516 1366 (void) printf(
1517 1367 "---------------------------+---------------------------");
1518 1368 (void) printf("-----+----------------\n");
1519 1369 }
1520 1370
1521 1371 static void
1522 1372 fini_device(device_t *dev)
1523 1373 {
1524 1374 free(dev->d_inq);
1525 1375 free(dev->d_node);
1526 1376 (void) close(dev->d_fd);
1527 1377 if (dev->d_name)
1528 1378 free(dev->d_name);
1529 1379 free(dev);
1530 1380 }
1531 1381
1532 1382 void *
1533 1383 my_zalloc(size_t size)
1534 1384 {
1535 1385 void *ret;
1536 1386
1537 1387 ret = malloc(size);
1538 1388 if (ret == NULL) {
1539 1389
1540 1390 /* Lets wait a sec. and try again */
1541 1391 if (errno == EAGAIN) {
1542 1392 (void) sleep(1);
1543 1393 ret = malloc(size);
1544 1394 }
1545 1395
1546 1396 if (ret == NULL) {
1547 1397 (void) err_msg("%s\n", gettext(strerror(errno)));
1548 1398 (void) err_msg(gettext(
1549 1399 "Memory allocation failure, Exiting...\n"));
1550 1400 exit(1);
1551 1401 }
1552 1402 }
1553 1403 (void) memset(ret, 0, size);
1554 1404 return (ret);
1555 1405 }
1556 1406
1557 1407 static int
1558 1408 vol_name_to_dev_node(char *vname, char *found)
1559 1409 {
1560 1410 struct stat statbuf;
1561 1411 char *p1;
1562 1412 int i;
1563 1413
1564 1414 if (vname == NULL)
1565 1415 return (0);
1566 1416 if (vol_running)
1567 1417 (void) volmgt_check(vname);
1568 1418 p1 = media_findname(vname);
1569 1419 if (p1 == NULL)
1570 1420 return (0);
1571 1421 if (stat(p1, &statbuf) < 0) {
1572 1422 free(p1);
1573 1423 return (0);
1574 1424 }
1575 1425 if (S_ISDIR(statbuf.st_mode)) {
1576 1426 for (i = 0; i < 16; i++) {
1577 1427 (void) snprintf(found, PATH_MAX, "%s/s%d", p1, i);
1578 1428 if (access(found, F_OK) >= 0)
1579 1429 break;
1580 1430 }
1581 1431 if (i == 16) {
1582 1432 free(p1);
1583 1433 return (0);
1584 1434 }
1585 1435 } else {
1586 1436 (void) strlcpy(found, p1, PATH_MAX);
1587 1437 }
1588 1438 free(p1);
1589 1439 return (1);
1590 1440 }
1591 1441
1592 1442 /*
1593 1443 * Searches for volume manager's equivalent char device for the
1594 1444 * supplied pathname which is of the form of /dev/rdsk/cxtxdxsx
1595 1445 */
1596 1446 static int
1597 1447 vol_lookup(char *supplied, char *found)
1598 1448 {
1599 1449 char tmpstr[PATH_MAX], tmpstr1[PATH_MAX], *p;
1600 1450 int i, ret;
1601 1451
1602 1452 (void) strlcpy(tmpstr, supplied, PATH_MAX);
1603 1453 if ((p = volmgt_symname(tmpstr)) == NULL) {
1604 1454 if (strstr(tmpstr, "s2") != NULL) {
1605 1455 *((char *)(strrchr(tmpstr, 's') + 1)) = 0;
1606 1456 for (i = 0; i < 16; i++) {
1607 1457 (void) snprintf(tmpstr1, PATH_MAX, "%s%d",
1608 1458 tmpstr, i);
1609 1459 if ((p = volmgt_symname(tmpstr1)) != NULL)
1610 1460 break;
1611 1461 }
1612 1462 } else if (strstr(tmpstr, "p0") != NULL) {
1613 1463 *((char *)(strrchr(tmpstr, 'p') + 1)) = 0;
1614 1464 for (i = 0; i < 5; i++) {
1615 1465 (void) snprintf(tmpstr1, PATH_MAX, "%s%d",
1616 1466 tmpstr, i);
1617 1467 if ((p = volmgt_symname(tmpstr1)) != NULL)
1618 1468 break;
1619 1469 }
1620 1470 } else
1621 1471 return (0);
1622 1472 if (p == NULL)
1623 1473 return (0);
1624 1474 }
1625 1475
1626 1476 ret = vol_name_to_dev_node(p, found);
1627 1477 free(p);
1628 1478 return (ret);
1629 1479 }
1630 1480
1631 1481 /*PRINTFLIKE1*/
1632 1482 void
1633 1483 err_msg(char *fmt, ...)
1634 1484 {
1635 1485 va_list ap;
1636 1486
1637 1487 va_start(ap, fmt);
1638 1488 (void) vfprintf(stderr, fmt, ap);
1639 1489 va_end(ap);
1640 1490 }
1641 1491
1642 1492 int
1643 1493 inquiry(int fd, uchar_t *inq)
1644 1494 {
1645 1495 struct uscsi_cmd *scmd;
1646 1496
1647 1497 scmd = get_uscsi_cmd();
1648 1498 scmd->uscsi_flags = USCSI_READ|USCSI_SILENT;
1649 1499 scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT;
1650 1500 scmd->uscsi_cdb[0] = INQUIRY_CMD;
1651 1501 scmd->uscsi_cdb[4] = INQUIRY_DATA_LENGTH;
1652 1502 scmd->uscsi_cdblen = 6;
1653 1503 scmd->uscsi_bufaddr = (char *)inq;
1654 1504 scmd->uscsi_buflen = INQUIRY_DATA_LENGTH;
1655 1505 if ((uscsi_error = uscsi(fd, scmd)) < 0)
1656 1506 return (0);
1657 1507 return (1);
1658 1508 }
1659 1509
1660 1510 struct uscsi_cmd *
1661 1511 get_uscsi_cmd(void)
1662 1512 {
1663 1513 (void) memset(&uscmd, 0, sizeof (uscmd));
1664 1514 (void) memset(ucdb, 0, 16);
1665 1515 uscmd.uscsi_cdb = ucdb;
1666 1516 return (&uscmd);
1667 1517 }
1668 1518
1669 1519 int
1670 1520 uscsi(int fd, struct uscsi_cmd *scmd)
1671 1521 {
1672 1522 int ret, global_rqsense;
1673 1523 int retries, max_retries = 5;
1674 1524 int i;
1675 1525
1676 1526 /* set up for request sense extensions */
1677 1527 if (!(scmd->uscsi_flags & USCSI_RQENABLE)) {
1678 1528 scmd->uscsi_flags |= USCSI_RQENABLE;
1679 1529 scmd->uscsi_rqlen = RQBUFLEN;
1680 1530 scmd->uscsi_rqbuf = rqbuf;
1681 1531 global_rqsense = 1;
1682 1532 } else {
1683 1533 global_rqsense = 0;
1684 1534 }
1685 1535
1686 1536 /*
1687 1537 * The device may be busy or slow and fail with a not ready status.
1688 1538 * we'll allow a limited number of retries to give the drive time
1689 1539 * to recover.
1690 1540 */
1691 1541 for (retries = 0; retries < max_retries; retries++) {
1692 1542
1693 1543 scmd->uscsi_status = 0;
1694 1544
1695 1545 if (global_rqsense)
1696 1546 (void) memset(rqbuf, 0, RQBUFLEN);
1697 1547
1698 1548 DPRINTF("cmd:[");
1699 1549 for (i = 0; i < scmd->uscsi_cdblen; i++)
1700 1550 DPRINTF1("0x%02x ",
1701 1551 (uchar_t)scmd->uscsi_cdb[i]);
1702 1552 DPRINTF("]\n");
1703 1553
1704 1554 /*
1705 1555 * We need to have root privledges in order to use
1706 1556 * uscsi commands on the device.
1707 1557 */
1708 1558
1709 1559 ret = ioctl(fd, USCSICMD, scmd);
1710 1560
1711 1561 /* maintain consistency in case of sgen */
1712 1562 if ((ret == 0) && (scmd->uscsi_status == 2)) {
1713 1563 ret = -1;
1714 1564 errno = EIO;
1715 1565 }
1716 1566
1717 1567 /* if error and extended request sense, retrieve errors */
1718 1568 if (global_rqsense && (ret < 0) && (scmd->uscsi_status == 2)) {
1719 1569 /*
1720 1570 * The drive is not ready to recieve commands but
1721 1571 * may be in the process of becoming ready.
1722 1572 * sleep for a short time then retry command.
1723 1573 * SENSE/ASC = 2/4 : not ready
1724 1574 * ASCQ = 0 Not Reportable.
1725 1575 * ASCQ = 1 Becoming ready.
1726 1576 */
1727 1577 if ((SENSE_KEY(rqbuf) == 2) && (ASC(rqbuf) == 4) &&
1728 1578 ((ASCQ(rqbuf) == 0) || (ASCQ(rqbuf) == 1))) {
1729 1579 total_retries++;
1730 1580 (void) sleep(3);
1731 1581 continue;
1732 1582 }
1733 1583
1734 1584 /*
1735 1585 * Device is not ready to transmit or a device reset
1736 1586 * has occurred. wait for a short period of time then
1737 1587 * retry the command.
1738 1588 */
1739 1589 if ((SENSE_KEY(rqbuf) == 6) && ((ASC(rqbuf) == 0x28) ||
1740 1590 (ASC(rqbuf) == 0x29))) {
1741 1591 (void) sleep(3);
1742 1592 total_retries++;
1743 1593 continue;
1744 1594 }
1745 1595
1746 1596 DPRINTF3("cmd: 0x%02x ret:%i status:%02x ",
1747 1597 (uchar_t)scmd->uscsi_cdb[0], ret,
1748 1598 scmd->uscsi_status);
1749 1599 DPRINTF3(" sense: %02x ASC: %02x ASCQ:%02x\n",
1750 1600 (uchar_t)SENSE_KEY(rqbuf),
1751 1601 (uchar_t)ASC(rqbuf), (uchar_t)ASCQ(rqbuf));
1752 1602 }
1753 1603
1754 1604 /* no errors we'll return */
1755 1605 break;
1756 1606 }
1757 1607
1758 1608 /* store the error status for later debug printing */
1759 1609 if ((ret < 0) && (global_rqsense)) {
1760 1610 uscsi_status = scmd->uscsi_status;
1761 1611 rqstatus = scmd->uscsi_rqstatus;
1762 1612 rqresid = scmd->uscsi_rqresid;
1763 1613
1764 1614 }
1765 1615
1766 1616 DPRINTF1("total retries: %d\n", total_retries);
1767 1617
1768 1618 return (ret);
1769 1619 }
1770 1620
1771 1621 /*
1772 1622 * will get the mode page only i.e. will strip off the header.
1773 1623 */
1774 1624 int
1775 1625 get_mode_page(int fd, int page_no, int pc, int buf_len, uchar_t *buffer)
1776 1626 {
1777 1627 int ret;
1778 1628 uchar_t byte2, *buf;
1779 1629 uint_t header_len, page_len, copy_cnt;
1780 1630
1781 1631 byte2 = (uchar_t)(((pc << 6) & 0xC0) | (page_no & 0x3f));
1782 1632 buf = (uchar_t *)my_zalloc(256);
1783 1633
1784 1634 /* Ask 254 bytes only to make our IDE driver happy */
1785 1635 ret = mode_sense(fd, byte2, 1, 254, buf);
1786 1636 if (ret == 0) {
1787 1637 free(buf);
1788 1638 return (0);
1789 1639 }
1790 1640
1791 1641 header_len = 8 + read_scsi16(&buf[6]);
1792 1642 page_len = buf[header_len + 1] + 2;
1793 1643
1794 1644 copy_cnt = (page_len > buf_len) ? buf_len : page_len;
1795 1645 (void) memcpy(buffer, &buf[header_len], copy_cnt);
1796 1646 free(buf);
1797 1647
1798 1648 return (1);
1799 1649 }
1800 1650
1801 1651 int
1802 1652 mode_sense(int fd, uchar_t pc, int dbd, int page_len, uchar_t *buffer)
1803 1653 {
1804 1654 struct uscsi_cmd *scmd;
1805 1655
1806 1656 scmd = get_uscsi_cmd();
1807 1657 scmd->uscsi_flags = USCSI_READ|USCSI_SILENT;
1808 1658 scmd->uscsi_buflen = page_len;
1809 1659 scmd->uscsi_bufaddr = (char *)buffer;
1810 1660 scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT;
1811 1661 scmd->uscsi_cdblen = 0xa;
1812 1662 scmd->uscsi_cdb[0] = MODE_SENSE_10_CMD;
1813 1663 if (dbd) {
1814 1664 /* don't return any block descriptors */
1815 1665 scmd->uscsi_cdb[1] = 0x8;
1816 1666 }
1817 1667 /* the page code we want */
1818 1668 scmd->uscsi_cdb[2] = pc;
1819 1669 /* allocation length */
1820 1670 scmd->uscsi_cdb[7] = (page_len >> 8) & 0xff;
1821 1671 scmd->uscsi_cdb[8] = page_len & 0xff;
1822 1672
1823 1673 if ((uscsi_error = uscsi(fd, scmd)) < 0)
1824 1674 return (0);
1825 1675 return (1);
1826 1676 }
1827 1677
1828 1678 uint16_t
1829 1679 read_scsi16(void *addr)
1830 1680 {
1831 1681 uchar_t *ad = (uchar_t *)addr;
1832 1682 uint16_t ret;
1833 1683
1834 1684 ret = ((((uint16_t)ad[0]) << 8) | ad[1]);
1835 1685 return (ret);
1836 1686 }
1837 1687
1838 1688 /*
1839 1689 * Allocate space for and return a pointer to a string
1840 1690 * on the stack. If the string is null, create
1841 1691 * an empty string.
1842 1692 * Use destroy_data() to free when no longer used.
1843 1693 */
1844 1694 char *
1845 1695 alloc_string(s)
1846 1696 char *s;
1847 1697 {
1848 1698 char *ns;
1849 1699
1850 1700 if (s == (char *)NULL) {
1851 1701 ns = (char *)my_zalloc(1);
1852 1702 } else {
1853 1703 ns = (char *)my_zalloc(strlen(s) + 1);
1854 1704 (void) strcpy(ns, s);
1855 1705 }
1856 1706 return (ns);
1857 1707 }
1858 1708
1859 1709 /*
1860 1710 * Follow symbolic links from the logical device name to
1861 1711 * the /devfs physical device name. To be complete, we
1862 1712 * handle the case of multiple links. This function
1863 1713 * either returns NULL (no links, or some other error),
1864 1714 * or the physical device name, alloc'ed on the heap.
1865 1715 *
1866 1716 * Note that the standard /devices prefix is stripped from
1867 1717 * the final pathname, if present. The trailing options
1868 1718 * are also removed (":c, raw").
1869 1719 */
1870 1720 static char *
1871 1721 get_physical_name(char *path)
1872 1722 {
1873 1723 struct stat stbuf;
1874 1724 int i;
1875 1725 int level;
1876 1726 char *p;
1877 1727 char s[MAXPATHLEN];
1878 1728 char buf[MAXPATHLEN];
1879 1729 char dir[MAXPATHLEN];
1880 1730 char savedir[MAXPATHLEN];
1881 1731 char *result = NULL;
1882 1732
1883 1733 if (getcwd(savedir, sizeof (savedir)) == NULL) {
1884 1734 DPRINTF1("getcwd() failed - %s\n", strerror(errno));
1885 1735 return (NULL);
1886 1736 }
1887 1737
1888 1738 (void) strcpy(s, path);
1889 1739 if ((p = strrchr(s, '/')) != NULL) {
1890 1740 *p = 0;
1891 1741 }
1892 1742 if (s[0] == 0) {
1893 1743 (void) strcpy(s, "/");
1894 1744 }
1895 1745 if (chdir(s) == -1) {
1896 1746 DPRINTF2("cannot chdir() to %s - %s\n",
1897 1747 s, strerror(errno));
1898 1748 goto exit;
1899 1749 }
1900 1750
1901 1751 level = 0;
1902 1752 (void) strcpy(s, path);
1903 1753 for (;;) {
1904 1754 /*
1905 1755 * See if there's a real file out there. If not,
1906 1756 * we have a dangling link and we ignore it.
1907 1757 */
1908 1758 if (stat(s, &stbuf) == -1) {
1909 1759 goto exit;
1910 1760 }
1911 1761 if (lstat(s, &stbuf) == -1) {
1912 1762 DPRINTF2("%s: lstat() failed - %s\n",
1913 1763 s, strerror(errno));
1914 1764 goto exit;
1915 1765 }
1916 1766 /*
1917 1767 * If the file is not a link, we're done one
1918 1768 * way or the other. If there were links,
1919 1769 * return the full pathname of the resulting
1920 1770 * file.
1921 1771 */
1922 1772 if (!S_ISLNK(stbuf.st_mode)) {
1923 1773 if (level > 0) {
1924 1774 /*
1925 1775 * Strip trailing options from the
1926 1776 * physical device name
1927 1777 */
1928 1778 if ((p = strrchr(s, ':')) != NULL) {
1929 1779 *p = 0;
1930 1780 }
1931 1781 /*
1932 1782 * Get the current directory, and
1933 1783 * glue the pieces together.
1934 1784 */
1935 1785 if (getcwd(dir, sizeof (dir)) == NULL) {
1936 1786 DPRINTF1("getcwd() failed - %s\n",
1937 1787 strerror(errno));
1938 1788 goto exit;
1939 1789 }
1940 1790 (void) strcat(dir, "/");
1941 1791 (void) strcat(dir, s);
1942 1792 /*
1943 1793 * If we have the standard fixed
1944 1794 * /devices prefix, remove it.
1945 1795 */
1946 1796 p = (strstr(dir, DEVFS_PREFIX) == dir) ?
1947 1797 dir+strlen(DEVFS_PREFIX) : dir;
1948 1798 result = alloc_string(p);
1949 1799 }
1950 1800 goto exit;
1951 1801 }
1952 1802 i = readlink(s, buf, sizeof (buf));
1953 1803 if (i == -1) {
1954 1804 DPRINTF2("%s: readlink() failed - %s\n",
1955 1805 s, strerror(errno));
1956 1806 goto exit;
1957 1807 }
1958 1808 level++;
1959 1809 buf[i] = 0;
1960 1810
1961 1811 /*
1962 1812 * Break up the pathname into the directory
1963 1813 * reference, if applicable and simple filename.
1964 1814 * chdir()'ing to the directory allows us to
1965 1815 * handle links with relative pathnames correctly.
1966 1816 */
1967 1817 (void) strcpy(dir, buf);
1968 1818 if ((p = strrchr(dir, '/')) != NULL) {
1969 1819 *p = 0;
1970 1820 if (chdir(dir) == -1) {
1971 1821 DPRINTF2("cannot chdir() to %s - %s\n",
1972 1822 dir, strerror(errno));
1973 1823 goto exit;
1974 1824 }
1975 1825 (void) strcpy(s, p+1);
1976 1826 } else {
1977 1827 (void) strcpy(s, buf);
1978 1828 }
1979 1829 }
1980 1830
1981 1831 exit:
1982 1832 if (chdir(savedir) == -1) {
1983 1833 (void) printf("cannot chdir() to %s - %s\n",
1984 1834 savedir, strerror(errno));
1985 1835 }
1986 1836
1987 1837 return (result);
1988 1838 }
1989 1839
1990 1840 static void
1991 1841 get_media_info(device_t *t_dev, char *sdev, char *pname, char *sn)
1992 1842 {
1993 1843 struct dk_cinfo cinfo;
1994 1844 struct extvtoc vtocinfo;
1995 1845 float size;
1996 1846 int32_t fd;
1997 1847 smedia_handle_t handle;
1998 1848 struct dk_minfo mediainfo;
1999 1849 int device_type;
2000 1850
2001 1851 device_type = ioctl(t_dev->d_fd, DKIOCGMEDIAINFO, &mediainfo);
2002 1852
2003 1853 /*
2004 1854 * Determine bus type.
2005 1855 */
2006 1856 if (!ioctl(t_dev->d_fd, DKIOCINFO, &cinfo)) {
2007 1857 if (strstr(cinfo.dki_cname, "usb") || strstr(pname, "usb")) {
2008 1858 (void) printf("\tBus: USB\n");
2009 1859 } else if (strstr(cinfo.dki_cname, "firewire") ||
2010 1860 strstr(pname, "firewire")) {
2011 1861 (void) printf("\tBus: Firewire\n");
2012 1862 } else if (strstr(cinfo.dki_cname, "ide") ||
2013 1863 strstr(pname, "ide")) {
2014 1864 (void) printf("\tBus: IDE\n");
2015 1865 } else if (strstr(cinfo.dki_cname, "scsi") ||
2016 1866 strstr(pname, "scsi")) {
2017 1867 (void) printf("\tBus: SCSI\n");
2018 1868 } else {
2019 1869 (void) printf("\tBus: <Unknown>\n");
2020 1870 }
2021 1871 } else {
2022 1872 (void) printf("\tBus: <Unknown>\n");
2023 1873 }
2024 1874
2025 1875 /*
2026 1876 * Calculate size of media.
2027 1877 */
2028 1878 if (!device_type &&
2029 1879 (!ioctl(t_dev->d_fd, DKIOCGMEDIAINFO, &mediainfo))) {
2030 1880 size = (mediainfo.dki_lbsize*
2031 1881 mediainfo.dki_capacity)/(1024.0*1024.0);
2032 1882 if (size < 1000) {
2033 1883 (void) printf("\tSize: %.1f MB\n", size);
2034 1884 } else {
2035 1885 size = size/1000;
2036 1886 (void) printf("\tSize: %.1f GB\n", size);
2037 1887 }
2038 1888 } else {
2039 1889 (void) printf("\tSize: <Unknown>\n");
2040 1890 }
2041 1891
2042 1892 /*
2043 1893 * Print label.
2044 1894 */
2045 1895 if (!device_type && (read_extvtoc(t_dev->d_fd, &vtocinfo) >= 0)) {
2046 1896 if (*vtocinfo.v_volume) {
2047 1897 (void) printf("\tLabel: %s\n", vtocinfo.v_volume);
2048 1898 } else {
2049 1899 (void) printf("\tLabel: <None>\n");
2050 1900 }
2051 1901 } else {
2052 1902 (void) printf("\tLabel: <Unknown>\n");
2053 1903 }
2054 1904
2055 1905 /*
2056 1906 * Acess permissions.
2057 1907 */
2058 1908 if (device_type) {
2059 1909 (void) printf("\tAccess permissions: <Unknown>\n");
2060 1910 return;
2061 1911 }
2062 1912
2063 1913 (void) fprintf(stdout, gettext("\tAccess permissions: "));
2064 1914 if (sn) {
2065 1915 /*
2066 1916 * Set dev_name for process_p_flag().
2067 1917 */
2068 1918 dev_name = sn;
2069 1919 fd = my_open(sn, O_RDONLY|O_NDELAY);
2070 1920 } else {
2071 1921 dev_name = sdev;
2072 1922 fd = my_open(sdev, O_RDONLY|O_NDELAY);
2073 1923 }
2074 1924 if (fd < 0) {
2075 1925 (void) printf("<Unknown>\n");
2076 1926 DPRINTF("Could not open device.\n");
2077 1927 (void) close(fd);
2078 1928 } else {
2079 1929 /* register the fd with the libsmedia */
2080 1930 handle = smedia_get_handle(fd);
2081 1931 if (handle == NULL) {
2082 1932 (void) printf("<Unknown>\n");
2083 1933 DPRINTF("Failed to get libsmedia handle.\n");
2084 1934 (void) close(fd);
2085 1935 } else {
2086 1936 process_p_flag(handle, fd);
2087 1937 }
2088 1938 }
2089 1939 /* Clear dev_name */
2090 1940 dev_name = NULL;
2091 1941 }
↓ open down ↓ |
918 lines elided |
↑ open up ↑ |
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX