1 /* $NetBSD: chio.c,v 1.6 1998/01/04 23:53:58 thorpej Exp $ */ 2 /* 3 * Copyright (c) 1996 Jason R. Thorpe <thorpej@and.com> 4 * All rights reserved. 5 * 6 * Redistribution and use in source and binary forms, with or without 7 * modification, are permitted provided that the following conditions 8 * are met: 9 * 1. Redistributions of source code must retain the above copyright 10 * notice, this list of conditions and the following disclaimer. 11 * 2. Redistributions in binary form must reproduce the above copyright 12 * notice, this list of conditions and the following disclaimer in the 13 * documentation and/or other materials provided with the distribution. 14 * 3. All advertising materials mentioning features or use of this software 15 * must display the following acknowledgements: 16 * This product includes software developed by Jason R. Thorpe 17 * for And Communications, http://www.and.com/ 18 * 4. The name of the author may not be used to endorse or promote products 19 * derived from this software without specific prior written permission. 20 * 21 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 22 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 23 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 24 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 25 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 26 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 27 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 28 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 29 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 30 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31 * SUCH DAMAGE. 32 */ 33 /* 34 * Additional Copyright (c) 1997, by Matthew Jacob, for NASA/Ames Research Ctr. 35 * Addidional Copyright (c) 2000, by C. Stephen Gunn, Waterspout Communications 36 */ 37 38 #ifndef lint 39 static const char copyright[] = 40 "@(#) Copyright (c) 1996 Jason R. Thorpe. All rights reserved."; 41 static const char rcsid[] = 42 "$FreeBSD$"; 43 #endif /* not lint */ 44 45 #include <sys/param.h> 46 #include <sys/chio.h> 47 #include <err.h> 48 #include <fcntl.h> 49 #include <stdio.h> 50 #include <stdlib.h> 51 #include <string.h> 52 #include <unistd.h> 53 54 #include "defs.h" 55 #include "pathnames.h" 56 57 extern char *__progname; /* from crt0.o */ 58 59 static void usage __P((void)); 60 static void cleanup __P((void)); 61 static int parse_element_type __P((char *)); 62 static int parse_element_unit __P((char *)); 63 static const char * element_type_name __P((int et)); 64 static int parse_special __P((char *)); 65 static int is_special __P((char *)); 66 static const char *bits_to_string __P((int, const char *)); 67 68 static void find_element __P((char *, u_int16_t *, u_int16_t *)); 69 static struct changer_element_status *get_element_status __P((u_int16_t, u_int16_t)); 70 71 static int do_move __P((char *, int, char **)); 72 static int do_exchange __P((char *, int, char **)); 73 static int do_position __P((char *, int, char **)); 74 static int do_params __P((char *, int, char **)); 75 static int do_getpicker __P((char *, int, char **)); 76 static int do_setpicker __P((char *, int, char **)); 77 static int do_status __P((char *, int, char **)); 78 static int do_ielem __P((char *, int, char **)); 79 static int do_return __P((char *, int, char **)); 80 static int do_voltag __P((char *, int, char **)); 81 82 #ifndef CHET_VT 83 #define CHET_VT 10 /* Completely Arbitrary */ 84 #endif 85 86 /* Valid changer element types. */ 87 const struct element_type elements[] = { 88 { "drive", CHET_DT }, 89 { "picker", CHET_MT }, 90 { "portal", CHET_IE }, 91 { "slot", CHET_ST }, 92 { "voltag", CHET_VT }, /* Select tapes by barcode */ 93 { NULL, 0 }, 94 }; 95 96 /* Valid commands. */ 97 const struct changer_command commands[] = { 98 { "exchange", do_exchange }, 99 { "getpicker", do_getpicker }, 100 { "ielem", do_ielem }, 101 { "move", do_move }, 102 { "params", do_params }, 103 { "position", do_position }, 104 { "setpicker", do_setpicker }, 105 { "status", do_status }, 106 { "return", do_return }, 107 { "voltag", do_voltag }, 108 { NULL, 0 }, 109 }; 110 111 /* Valid special words. */ 112 const struct special_word specials[] = { 113 { "inv", SW_INVERT }, 114 { "inv1", SW_INVERT1 }, 115 { "inv2", SW_INVERT2 }, 116 { NULL, 0 }, 117 }; 118 119 static int changer_fd; 120 static const char *changer_name; 121 122 int 123 main(argc, argv) 124 int argc; 125 char **argv; 126 { 127 int ch, i; 128 129 while ((ch = getopt(argc, argv, "f:")) != -1) { 130 switch (ch) { 131 case 'f': 132 changer_name = optarg; 133 break; 134 135 default: 136 usage(); 137 } 138 } 139 argc -= optind; 140 argv += optind; 141 142 if (argc == 0) 143 usage(); 144 145 /* Get the default changer if not already specified. */ 146 if (changer_name == NULL) 147 if ((changer_name = getenv(CHANGER_ENV_VAR)) == NULL) 148 changer_name = _PATH_CH; 149 150 /* Open the changer device. */ 151 if ((changer_fd = open(changer_name, O_RDWR, 0600)) == -1) 152 err(1, "%s: open", changer_name); 153 154 /* Register cleanup function. */ 155 if (atexit(cleanup)) 156 err(1, "can't register cleanup function"); 157 158 /* Find the specified command. */ 159 for (i = 0; commands[i].cc_name != NULL; ++i) 160 if (strcmp(*argv, commands[i].cc_name) == 0) 161 break; 162 if (commands[i].cc_name == NULL) { 163 /* look for abbreviation */ 164 for (i = 0; commands[i].cc_name != NULL; ++i) 165 if (strncmp(*argv, commands[i].cc_name, 166 strlen(*argv)) == 0) 167 break; 168 } 169 170 if (commands[i].cc_name == NULL) 171 errx(1, "unknown command: %s", *argv); 172 173 exit ((*commands[i].cc_handler)(commands[i].cc_name, argc, argv)); 174 /* NOTREACHED */ 175 } 176 177 static int 178 do_move(cname, argc, argv) 179 char *cname; 180 int argc; 181 char **argv; 182 { 183 struct changer_move cmd; 184 int val; 185 186 /* 187 * On a move command, we expect the following: 188 * 189 * <from ET> <from EU> <to ET> <to EU> [inv] 190 * 191 * where ET == element type and EU == element unit. 192 */ 193 194 ++argv; --argc; 195 196 if (argc < 4) { 197 warnx("%s: too few arguments", cname); 198 goto usage; 199 } else if (argc > 5) { 200 warnx("%s: too many arguments", cname); 201 goto usage; 202 } 203 (void) memset(&cmd, 0, sizeof(cmd)); 204 205 /* <from ET> */ 206 cmd.cm_fromtype = parse_element_type(*argv); 207 ++argv; --argc; 208 209 /* Check for voltag virtual type */ 210 if (CHET_VT == cmd.cm_fromtype) { 211 find_element(*argv, &cmd.cm_fromtype, &cmd.cm_fromunit); 212 } else { 213 /* <from EU> */ 214 cmd.cm_fromunit = parse_element_unit(*argv); 215 } 216 ++argv; --argc; 217 218 /* <to ET> */ 219 cmd.cm_totype = parse_element_type(*argv); 220 ++argv; --argc; 221 222 /* Check for voltag virtual type, and report error */ 223 if (CHET_VT == cmd.cm_totype) 224 errx(1,"%s: voltag only makes sense as an element source", 225 cname); 226 227 /* <to EU> */ 228 cmd.cm_tounit = parse_element_unit(*argv); 229 ++argv; --argc; 230 231 /* Deal with optional command modifier. */ 232 if (argc) { 233 val = parse_special(*argv); 234 switch (val) { 235 case SW_INVERT: 236 cmd.cm_flags |= CM_INVERT; 237 break; 238 239 default: 240 errx(1, "%s: inappropriate modifier `%s'", 241 cname, *argv); 242 /* NOTREACHED */ 243 } 244 } 245 246 /* Send command to changer. */ 247 if (ioctl(changer_fd, CHIOMOVE, &cmd)) 248 err(1, "%s: CHIOMOVE", changer_name); 249 250 return (0); 251 252 usage: 253 (void) fprintf(stderr, "usage: %s %s " 254 "<from ET> <from EU> <to ET> <to EU> [inv]\n", __progname, cname); 255 return (1); 256 } 257 258 static int 259 do_exchange(cname, argc, argv) 260 char *cname; 261 int argc; 262 char **argv; 263 { 264 struct changer_exchange cmd; 265 int val; 266 267 /* 268 * On an exchange command, we expect the following: 269 * 270 * <src ET> <src EU> <dst1 ET> <dst1 EU> [<dst2 ET> <dst2 EU>] [inv1] [inv2] 271 * 272 * where ET == element type and EU == element unit. 273 */ 274 275 ++argv; --argc; 276 277 if (argc < 4) { 278 warnx("%s: too few arguments", cname); 279 goto usage; 280 } else if (argc > 8) { 281 warnx("%s: too many arguments", cname); 282 goto usage; 283 } 284 (void) memset(&cmd, 0, sizeof(cmd)); 285 286 /* <src ET> */ 287 cmd.ce_srctype = parse_element_type(*argv); 288 ++argv; --argc; 289 290 /* Check for voltag virtual type */ 291 if (CHET_VT == cmd.ce_srctype) { 292 find_element(*argv, &cmd.ce_srctype, &cmd.ce_srcunit); 293 } else { 294 /* <from EU> */ 295 cmd.ce_srcunit = parse_element_unit(*argv); 296 } 297 ++argv; --argc; 298 299 /* <dst1 ET> */ 300 cmd.ce_fdsttype = parse_element_type(*argv); 301 ++argv; --argc; 302 303 /* Check for voltag virtual type */ 304 if (CHET_VT == cmd.ce_fdsttype) { 305 find_element(*argv, &cmd.ce_fdsttype, &cmd.ce_fdstunit); 306 } else { 307 /* <from EU> */ 308 cmd.ce_fdstunit = parse_element_unit(*argv); 309 } 310 ++argv; --argc; 311 312 /* 313 * If the next token is a special word or there are no more 314 * arguments, then this is a case of simple exchange. 315 * dst2 == src. 316 */ 317 if ((argc == 0) || is_special(*argv)) { 318 cmd.ce_sdsttype = cmd.ce_srctype; 319 cmd.ce_sdstunit = cmd.ce_srcunit; 320 goto do_special; 321 } 322 323 /* <dst2 ET> */ 324 cmd.ce_sdsttype = parse_element_type(*argv); 325 ++argv; --argc; 326 327 if (CHET_VT == cmd.ce_sdsttype) 328 errx(1,"%s %s: voltag only makes sense as an element source", 329 cname, *argv); 330 331 /* <dst2 EU> */ 332 cmd.ce_sdstunit = parse_element_unit(*argv); 333 ++argv; --argc; 334 335 do_special: 336 /* Deal with optional command modifiers. */ 337 while (argc) { 338 val = parse_special(*argv); 339 ++argv; --argc; 340 switch (val) { 341 case SW_INVERT1: 342 cmd.ce_flags |= CE_INVERT1; 343 break; 344 345 case SW_INVERT2: 346 cmd.ce_flags |= CE_INVERT2; 347 break; 348 349 default: 350 errx(1, "%s: inappropriate modifier `%s'", 351 cname, *argv); 352 /* NOTREACHED */ 353 } 354 } 355 356 /* Send command to changer. */ 357 if (ioctl(changer_fd, CHIOEXCHANGE, &cmd)) 358 err(1, "%s: CHIOEXCHANGE", changer_name); 359 360 return (0); 361 362 usage: 363 (void) fprintf(stderr, 364 "usage: %s %s <src ET> <src EU> <dst1 ET> <dst1 EU>\n" 365 " [<dst2 ET> <dst2 EU>] [inv1] [inv2]\n", 366 __progname, cname); 367 return (1); 368 } 369 370 static int 371 do_position(cname, argc, argv) 372 char *cname; 373 int argc; 374 char **argv; 375 { 376 struct changer_position cmd; 377 int val; 378 379 /* 380 * On a position command, we expect the following: 381 * 382 * <to ET> <to EU> [inv] 383 * 384 * where ET == element type and EU == element unit. 385 */ 386 387 ++argv; --argc; 388 389 if (argc < 2) { 390 warnx("%s: too few arguments", cname); 391 goto usage; 392 } else if (argc > 3) { 393 warnx("%s: too many arguments", cname); 394 goto usage; 395 } 396 (void) memset(&cmd, 0, sizeof(cmd)); 397 398 /* <to ET> */ 399 cmd.cp_type = parse_element_type(*argv); 400 ++argv; --argc; 401 402 /* <to EU> */ 403 cmd.cp_unit = parse_element_unit(*argv); 404 ++argv; --argc; 405 406 /* Deal with optional command modifier. */ 407 if (argc) { 408 val = parse_special(*argv); 409 switch (val) { 410 case SW_INVERT: 411 cmd.cp_flags |= CP_INVERT; 412 break; 413 414 default: 415 errx(1, "%s: inappropriate modifier `%s'", 416 cname, *argv); 417 /* NOTREACHED */ 418 } 419 } 420 421 /* Send command to changer. */ 422 if (ioctl(changer_fd, CHIOPOSITION, &cmd)) 423 err(1, "%s: CHIOPOSITION", changer_name); 424 425 return (0); 426 427 usage: 428 (void) fprintf(stderr, "usage: %s %s <to ET> <to EU> [inv]\n", 429 __progname, cname); 430 return (1); 431 } 432 433 /* ARGSUSED */ 434 static int 435 do_params(cname, argc, argv) 436 char *cname; 437 int argc; 438 char **argv; 439 { 440 struct changer_params data; 441 int picker; 442 443 /* No arguments to this command. */ 444 445 ++argv; --argc; 446 447 if (argc) { 448 warnx("%s: no arguments expected", cname); 449 goto usage; 450 } 451 452 /* Get params from changer and display them. */ 453 (void) memset(&data, 0, sizeof(data)); 454 if (ioctl(changer_fd, CHIOGPARAMS, &data)) 455 err(1, "%s: CHIOGPARAMS", changer_name); 456 457 (void) printf("%s: %d slot%s, %d drive%s, %d picker%s", 458 changer_name, 459 data.cp_nslots, (data.cp_nslots > 1) ? "s" : "", 460 data.cp_ndrives, (data.cp_ndrives > 1) ? "s" : "", 461 data.cp_npickers, (data.cp_npickers > 1) ? "s" : ""); 462 if (data.cp_nportals) 463 (void) printf(", %d portal%s", data.cp_nportals, 464 (data.cp_nportals > 1) ? "s" : ""); 465 466 /* Get current picker from changer and display it. */ 467 if (ioctl(changer_fd, CHIOGPICKER, &picker)) 468 err(1, "%s: CHIOGPICKER", changer_name); 469 470 (void) printf("\n%s: current picker: %d\n", changer_name, picker); 471 472 return (0); 473 474 usage: 475 (void) fprintf(stderr, "usage: %s %s\n", __progname, cname); 476 return (1); 477 } 478 479 /* ARGSUSED */ 480 static int 481 do_getpicker(cname, argc, argv) 482 char *cname; 483 int argc; 484 char **argv; 485 { 486 int picker; 487 488 /* No arguments to this command. */ 489 490 ++argv; --argc; 491 492 if (argc) { 493 warnx("%s: no arguments expected", cname); 494 goto usage; 495 } 496 497 /* Get current picker from changer and display it. */ 498 if (ioctl(changer_fd, CHIOGPICKER, &picker)) 499 err(1, "%s: CHIOGPICKER", changer_name); 500 501 (void) printf("%s: current picker: %d\n", changer_name, picker); 502 503 return (0); 504 505 usage: 506 (void) fprintf(stderr, "usage: %s %s\n", __progname, cname); 507 return (1); 508 } 509 510 static int 511 do_setpicker(cname, argc, argv) 512 char *cname; 513 int argc; 514 char **argv; 515 { 516 int picker; 517 518 ++argv; --argc; 519 520 if (argc < 1) { 521 warnx("%s: too few arguments", cname); 522 goto usage; 523 } else if (argc > 1) { 524 warnx("%s: too many arguments", cname); 525 goto usage; 526 } 527 528 picker = parse_element_unit(*argv); 529 530 /* Set the changer picker. */ 531 if (ioctl(changer_fd, CHIOSPICKER, &picker)) 532 err(1, "%s: CHIOSPICKER", changer_name); 533 534 return (0); 535 536 usage: 537 (void) fprintf(stderr, "usage: %s %s <picker>\n", __progname, cname); 538 return (1); 539 } 540 541 static int 542 do_status(cname, argc, argv) 543 char *cname; 544 int argc; 545 char **argv; 546 { 547 struct changer_params cp; 548 struct changer_element_status_request cesr; 549 int i, count, base, chet, schet, echet; 550 char *description; 551 int pvoltag = 0; 552 int avoltag = 0; 553 int sense = 0; 554 int scsi = 0; 555 int source = 0; 556 int intaddr = 0; 557 int c; 558 559 count = 0; 560 base = 0; 561 description = NULL; 562 563 optind = optreset = 1; 564 while ((c = getopt(argc, argv, "vVsSbaI")) != -1) { 565 switch (c) { 566 case 'v': 567 pvoltag = 1; 568 break; 569 case 'V': 570 avoltag = 1; 571 break; 572 case 's': 573 sense = 1; 574 break; 575 case 'S': 576 source = 1; 577 break; 578 case 'b': 579 scsi = 1; 580 break; 581 case 'I': 582 intaddr = 1; 583 break; 584 case 'a': 585 pvoltag = avoltag = source = sense = scsi = intaddr = 1; 586 break; 587 default: 588 warnx("%s: bad option", cname); 589 goto usage; 590 } 591 } 592 593 argc -= optind; 594 argv += optind; 595 596 /* 597 * On a status command, we expect the following: 598 * 599 * [<ET> [<start> [<end>] ] ] 600 * 601 * where ET == element type, start == first element to report, 602 * end == number of elements to report 603 * 604 * If we get no arguments, we get the status of all 605 * known element types. 606 */ 607 if (argc > 3) { 608 warnx("%s: too many arguments", cname); 609 goto usage; 610 } 611 612 /* 613 * Get params from changer. Specifically, we need the element 614 * counts. 615 */ 616 if (ioctl(changer_fd, CHIOGPARAMS, (char *)&cp)) 617 err(1, "%s: CHIOGPARAMS", changer_name); 618 619 if (argc > 0) 620 schet = echet = parse_element_type(argv[0]); 621 else { 622 schet = CHET_MT; 623 echet = CHET_DT; 624 } 625 if (argc > 1) { 626 base = atol(argv[1]); 627 count = 1; 628 } 629 if (argc > 2) 630 count = atol(argv[2]) - base + 1; 631 632 if (base < 0 || count < 0) 633 errx(1, "bad arguments"); 634 635 for (chet = schet; chet <= echet; ++chet) { 636 switch (chet) { 637 case CHET_MT: 638 if (count == 0) 639 count = cp.cp_npickers; 640 else if (count > cp.cp_npickers) 641 errx(1, "not that many pickers in device"); 642 description = "picker"; 643 break; 644 645 case CHET_ST: 646 if (count == 0) 647 count = cp.cp_nslots; 648 else if (count > cp.cp_nslots) 649 errx(1, "not that many slots in device"); 650 description = "slot"; 651 break; 652 653 case CHET_IE: 654 if (count == 0) 655 count = cp.cp_nportals; 656 else if (count > cp.cp_nportals) 657 errx(1, "not that many portals in device"); 658 description = "portal"; 659 break; 660 661 case CHET_DT: 662 if (count == 0) 663 count = cp.cp_ndrives; 664 else if (count > cp.cp_ndrives) 665 errx(1, "not that many drives in device"); 666 description = "drive"; 667 break; 668 669 default: 670 /* To appease gcc -Wuninitialized. */ 671 count = 0; 672 description = NULL; 673 } 674 675 if (count == 0) { 676 if (argc == 0) 677 continue; 678 else { 679 printf("%s: no %s elements\n", 680 changer_name, description); 681 return (0); 682 } 683 } 684 685 bzero(&cesr, sizeof(cesr)); 686 cesr.cesr_element_type = chet; 687 cesr.cesr_element_base = base; 688 cesr.cesr_element_count = count; 689 /* Allocate storage for the status structures. */ 690 cesr.cesr_element_status = 691 (struct changer_element_status *) 692 calloc(count, sizeof(struct changer_element_status)); 693 694 if (!cesr.cesr_element_status) 695 errx(1, "can't allocate status storage"); 696 697 if (avoltag || pvoltag) 698 cesr.cesr_flags |= CESR_VOLTAGS; 699 700 if (ioctl(changer_fd, CHIOGSTATUS, (char *)&cesr)) { 701 free(cesr.cesr_element_status); 702 err(1, "%s: CHIOGSTATUS", changer_name); 703 } 704 705 /* Dump the status for each reported element. */ 706 for (i = 0; i < count; ++i) { 707 struct changer_element_status *ces = 708 &(cesr.cesr_element_status[i]); 709 printf("%s %d: %s", description, ces->ces_addr, 710 bits_to_string(ces->ces_flags, 711 CESTATUS_BITS)); 712 if (sense) 713 printf(" sense: <0x%02x/0x%02x>", 714 ces->ces_sensecode, 715 ces->ces_sensequal); 716 if (pvoltag) 717 printf(" voltag: <%s:%d>", 718 ces->ces_pvoltag.cv_volid, 719 ces->ces_pvoltag.cv_serial); 720 if (avoltag) 721 printf(" avoltag: <%s:%d>", 722 ces->ces_avoltag.cv_volid, 723 ces->ces_avoltag.cv_serial); 724 if (source) { 725 if (ces->ces_flags & CES_SOURCE_VALID) 726 printf(" source: <%s %d>", 727 element_type_name( 728 ces->ces_source_type), 729 ces->ces_source_addr); 730 else 731 printf(" source: <>"); 732 } 733 if (intaddr) 734 printf(" intaddr: <%d>", ces->ces_int_addr); 735 if (scsi) { 736 printf(" scsi: <"); 737 if (ces->ces_flags & CES_SCSIID_VALID) 738 printf("%d", ces->ces_scsi_id); 739 else 740 putchar('?'); 741 putchar(':'); 742 if (ces->ces_flags & CES_LUN_VALID) 743 printf("%d", ces->ces_scsi_lun); 744 else 745 putchar('?'); 746 putchar('>'); 747 } 748 putchar('\n'); 749 } 750 751 free(cesr.cesr_element_status); 752 count = 0; 753 } 754 755 return (0); 756 757 usage: 758 (void) fprintf(stderr, "usage: %s %s [-vVsSbaA] [<element type> [<start-addr> [<end-addr>] ] ]\n", 759 __progname, cname); 760 return (1); 761 } 762 763 static int 764 do_ielem(cname, argc, argv) 765 char *cname; 766 int argc; 767 char **argv; 768 { 769 int timeout = 0; 770 771 if (argc == 2) { 772 timeout = atol(argv[1]); 773 } else if (argc > 1) { 774 warnx("%s: too many arguments", cname); 775 goto usage; 776 } 777 778 if (ioctl(changer_fd, CHIOIELEM, &timeout)) 779 err(1, "%s: CHIOIELEM", changer_name); 780 781 return (0); 782 783 usage: 784 (void) fprintf(stderr, "usage: %s %s [<timeout>]\n", 785 __progname, cname); 786 return (1); 787 } 788 789 static int 790 do_voltag(cname, argc, argv) 791 char *cname; 792 int argc; 793 char **argv; 794 { 795 int force = 0; 796 int clear = 0; 797 int alternate = 0; 798 int c; 799 struct changer_set_voltag_request csvr; 800 801 bzero(&csvr, sizeof(csvr)); 802 803 optind = optreset = 1; 804 while ((c = getopt(argc, argv, "fca")) != -1) { 805 switch (c) { 806 case 'f': 807 force = 1; 808 break; 809 case 'c': 810 clear = 1; 811 break; 812 case 'a': 813 alternate = 1; 814 break; 815 default: 816 warnx("%s: bad option", cname); 817 goto usage; 818 } 819 } 820 821 argc -= optind; 822 argv += optind; 823 824 if (argc < 2) { 825 warnx("%s: missing element specification", cname); 826 goto usage; 827 } 828 829 csvr.csvr_type = parse_element_type(argv[0]); 830 csvr.csvr_addr = atol(argv[1]); 831 832 if (!clear) { 833 if (argc < 3 || argc > 4) { 834 warnx("%s: missing argument", cname); 835 goto usage; 836 } 837 838 if (force) 839 csvr.csvr_flags = CSVR_MODE_REPLACE; 840 else 841 csvr.csvr_flags = CSVR_MODE_SET; 842 843 if (strlen(argv[2]) > sizeof(csvr.csvr_voltag.cv_volid)) { 844 warnx("%s: volume label too long", cname); 845 goto usage; 846 } 847 848 strncpy(csvr.csvr_voltag.cv_volid, argv[2], 849 sizeof(csvr.csvr_voltag.cv_volid)); 850 851 if (argc == 4) { 852 csvr.csvr_voltag.cv_serial = atol(argv[3]); 853 } 854 } else { 855 if (argc != 2) { 856 warnx("%s: unexpected argument", cname); 857 goto usage; 858 } 859 csvr.csvr_flags = CSVR_MODE_CLEAR; 860 } 861 862 if (alternate) { 863 csvr.csvr_flags |= CSVR_ALTERNATE; 864 } 865 866 if (ioctl(changer_fd, CHIOSETVOLTAG, &csvr)) 867 err(1, "%s: CHIOSETVOLTAG", changer_name); 868 869 return 0; 870 usage: 871 (void) fprintf(stderr, 872 "usage: %s %s [-fca] <element> [<voltag> [<vsn>] ]\n", 873 __progname, cname); 874 return 1; 875 } 876 877 static int 878 parse_element_type(cp) 879 char *cp; 880 { 881 int i; 882 883 for (i = 0; elements[i].et_name != NULL; ++i) 884 if (strcmp(elements[i].et_name, cp) == 0) 885 return (elements[i].et_type); 886 887 errx(1, "invalid element type `%s'", cp); 888 /* NOTREACHED */ 889 } 890 891 static const char * 892 element_type_name(et) 893 int et; 894 { 895 int i; 896 897 for (i = 0; elements[i].et_name != NULL; i++) 898 if (elements[i].et_type == et) 899 return elements[i].et_name; 900 901 return "unknown"; 902 } 903 904 static int 905 parse_element_unit(cp) 906 char *cp; 907 { 908 int i; 909 char *p; 910 911 i = (int)strtol(cp, &p, 10); 912 if ((i < 0) || (*p != '\0')) 913 errx(1, "invalid unit number `%s'", cp); 914 915 return (i); 916 } 917 918 static int 919 parse_special(cp) 920 char *cp; 921 { 922 int val; 923 924 val = is_special(cp); 925 if (val) 926 return (val); 927 928 errx(1, "invalid modifier `%s'", cp); 929 /* NOTREACHED */ 930 } 931 932 static int 933 is_special(cp) 934 char *cp; 935 { 936 int i; 937 938 for (i = 0; specials[i].sw_name != NULL; ++i) 939 if (strcmp(specials[i].sw_name, cp) == 0) 940 return (specials[i].sw_value); 941 942 return (0); 943 } 944 945 static const char * 946 bits_to_string(v, cp) 947 int v; 948 const char *cp; 949 { 950 const char *np; 951 char f, sep, *bp; 952 static char buf[128]; 953 954 bp = buf; 955 (void) memset(buf, 0, sizeof(buf)); 956 957 for (sep = '<'; (f = *cp++) != 0; cp = np) { 958 for (np = cp; *np >= ' ';) 959 np++; 960 if ((v & (1 << (f - 1))) == 0) 961 continue; 962 (void) snprintf(bp, sizeof(buf) - (bp - &buf[0]), 963 "%c%.*s", sep, (int)(long)(np - cp), cp); 964 bp += strlen(bp); 965 sep = ','; 966 } 967 if (sep != '<') 968 *bp = '>'; 969 970 return (buf); 971 } 972 /* 973 * do_return() 974 * 975 * Given an element reference, ask the changer/picker to move that 976 * element back to its source slot. 977 */ 978 static int 979 do_return(cname, argc, argv) 980 char *cname; 981 int argc; 982 char **argv; 983 { 984 struct changer_element_status *ces; 985 struct changer_move cmd; 986 u_int16_t type, element; 987 988 ++argv; --argc; 989 990 if (argc < 2) { 991 warnx("%s: too few arguments", cname); 992 goto usage; 993 } else if (argc > 3) { 994 warnx("%s: too many arguments", cname); 995 goto usage; 996 } 997 998 type = parse_element_type(*argv); 999 ++argv; --argc; 1000 1001 /* Handle voltag virtual Changer Element Type */ 1002 if (CHET_VT == type) { 1003 find_element(*argv, &type, &element); 1004 } else { 1005 element = parse_element_unit(*argv); 1006 } 1007 ++argv; --argc; 1008 1009 ces = get_element_status(type, element); /* Get the status */ 1010 1011 if (NULL == ces) 1012 errx(1, "%s: null element status pointer", cname); 1013 1014 if (!(ces->ces_flags & CES_SOURCE_VALID)) 1015 errx(1, "%s: no source information", cname); 1016 1017 (void) memset(&cmd, 0, sizeof(cmd)); 1018 1019 cmd.cm_fromtype = type; 1020 cmd.cm_fromunit = element; 1021 cmd.cm_totype = ces->ces_source_type; 1022 cmd.cm_tounit = ces->ces_source_addr; 1023 1024 if (ioctl(changer_fd, CHIOMOVE, &cmd) == -1) 1025 err(1, "%s: CHIOMOVE", changer_name); 1026 free(ces); 1027 1028 return(0); 1029 1030 usage: 1031 (void) fprintf(stderr, "usage: %s %s " 1032 "<from ET> <from EU>\n", __progname, cname); 1033 return(1); 1034 } 1035 1036 /* 1037 * get_element_status() 1038 * 1039 * return a *cesr for the specified changer element. This 1040 * routing will malloc()/calloc() the memory. The caller 1041 * should free() it when done. 1042 */ 1043 static struct changer_element_status * 1044 get_element_status(type, element) 1045 u_int16_t type; 1046 u_int16_t element; 1047 { 1048 struct changer_element_status_request cesr; 1049 struct changer_element_status *ces; 1050 1051 ces = (struct changer_element_status *) 1052 calloc(1, sizeof(struct changer_element_status)); 1053 1054 if (NULL == ces) 1055 errx(1, "can't allocate status storage"); 1056 1057 (void)memset(&cesr, 0, sizeof(cesr)); 1058 1059 cesr.cesr_element_type = type; 1060 cesr.cesr_element_base = element; 1061 cesr.cesr_element_count = 1; /* Only this one element */ 1062 cesr.cesr_flags |= CESR_VOLTAGS; /* Grab voltags as well */ 1063 cesr.cesr_element_status = ces; 1064 1065 if (ioctl(changer_fd, CHIOGSTATUS, (char *)&cesr) == -1) { 1066 free(ces); 1067 err(1, "%s: CHIOGSTATUS", changer_name); 1068 /* NOTREACHED */ 1069 } 1070 1071 return ces; 1072 } 1073 1074 1075 /* 1076 * find_element() 1077 * 1078 * Given a <voltag> find the chager element and unit, or exit 1079 * with an error if it isn't found. We grab the changer status 1080 * and iterate until we find a match, or crap out. 1081 */ 1082 static void 1083 find_element(voltag, et, eu) 1084 char *voltag; 1085 u_int16_t *et; 1086 u_int16_t *eu; 1087 { 1088 struct changer_params cp; 1089 struct changer_element_status_request cesr; 1090 struct changer_element_status *ch_ces, *ces; 1091 int elem, total_elem, found = 0; 1092 1093 /* 1094 * Get the changer parameters, we're interested in the counts 1095 * for all types of elements to perform our search. 1096 */ 1097 if (ioctl(changer_fd, CHIOGPARAMS, (char *)&cp)) 1098 err(1, "%s: CHIOGPARAMS", changer_name); 1099 1100 /* Allocate some memory for the results */ 1101 total_elem = (cp.cp_nslots + cp.cp_ndrives 1102 + cp.cp_npickers + cp.cp_nportals); 1103 1104 ch_ces = (struct changer_element_status *) 1105 calloc(total_elem, sizeof(struct changer_element_status)); 1106 1107 if (NULL == ch_ces) 1108 errx(1, "can't allocate status storage"); 1109 1110 ces = ch_ces; 1111 1112 /* Read in the changer slots */ 1113 if (cp.cp_nslots > 0) { 1114 cesr.cesr_element_type = CHET_ST; 1115 cesr.cesr_element_base = 0; 1116 cesr.cesr_element_count = cp.cp_nslots; 1117 cesr.cesr_flags |= CESR_VOLTAGS; 1118 cesr.cesr_element_status = ces; 1119 1120 if (ioctl(changer_fd, CHIOGSTATUS, (char *)&cesr) == -1) { 1121 free(ch_ces); 1122 err(1, "%s: CHIOGSTATUS", changer_name); 1123 } 1124 ces += cp.cp_nslots; 1125 } 1126 1127 /* Read in the drive information */ 1128 if (cp.cp_ndrives > 0 ) { 1129 1130 (void) memset(&cesr, 0, sizeof(cesr)); 1131 cesr.cesr_element_type = CHET_DT; 1132 cesr.cesr_element_base = 0; 1133 cesr.cesr_element_count = cp.cp_ndrives; 1134 cesr.cesr_flags |= CESR_VOLTAGS; 1135 cesr.cesr_element_status = ces; 1136 1137 if (ioctl(changer_fd, CHIOGSTATUS, (char *)&cesr) == -1) { 1138 free(ch_ces); 1139 err(1, "%s: CHIOGSTATUS", changer_name); 1140 } 1141 ces += cp.cp_ndrives; 1142 } 1143 1144 /* Read in the portal information */ 1145 if (cp.cp_nportals > 0 ) { 1146 (void) memset(&cesr, 0, sizeof(cesr)); 1147 cesr.cesr_element_type = CHET_IE; 1148 cesr.cesr_element_base = 0; 1149 cesr.cesr_element_count = cp.cp_nportals; 1150 cesr.cesr_flags |= CESR_VOLTAGS; 1151 cesr.cesr_element_status = ces; 1152 1153 if (ioctl(changer_fd, CHIOGSTATUS, (char *)&cesr) == -1) { 1154 free(ch_ces); 1155 err(1, "%s: CHIOGSTATUS", changer_name); 1156 } 1157 ces += cp.cp_nportals; 1158 } 1159 1160 /* Read in the picker information */ 1161 if (cp.cp_npickers > 0) { 1162 (void) memset(&cesr, 0, sizeof(cesr)); 1163 cesr.cesr_element_type = CHET_MT; 1164 cesr.cesr_element_base = 0; 1165 cesr.cesr_element_count = cp.cp_npickers; 1166 cesr.cesr_flags |= CESR_VOLTAGS; 1167 cesr.cesr_element_status = ces; 1168 1169 if (ioctl(changer_fd, CHIOGSTATUS, (char *)&cesr) == -1) { 1170 free(ch_ces); 1171 err(1, "%s: CHIOGSTATUS", changer_name); 1172 } 1173 } 1174 1175 /* 1176 * Now search the list the specified <voltag> 1177 */ 1178 for (elem = 0; elem <= total_elem; ++elem) { 1179 1180 ces = &ch_ces[elem]; 1181 1182 /* Make sure we have a tape in this element */ 1183 if ((ces->ces_flags & (CES_STATUS_ACCESS|CES_STATUS_FULL)) 1184 != (CES_STATUS_ACCESS|CES_STATUS_FULL)) 1185 continue; 1186 1187 /* Check to see if it is our target */ 1188 if (strcasecmp(voltag, ces->ces_pvoltag.cv_volid) == 0) { 1189 *et = ces->ces_type; 1190 *eu = ces->ces_addr; 1191 ++found; 1192 break; 1193 } 1194 } 1195 if (!found) { 1196 errx(1, "%s: unable to locate voltag: %s", changer_name, 1197 voltag); 1198 } 1199 free(ch_ces); 1200 return; 1201 } 1202 1203 static void 1204 cleanup() 1205 { 1206 /* Simple enough... */ 1207 (void)close(changer_fd); 1208 } 1209 1210 static void 1211 usage() 1212 { 1213 (void) fprintf(stderr, "usage: %s [-f changer] command [-<flags>] " 1214 "arg1 arg2 [arg3 [...]]\n", __progname); 1215 exit(1); 1216 } 1217