📄 scsiformat.c
字号:
/*- * Copyright (c) 1992, 1993 * The Regents of the University of California. All rights reserved. * * This software was developed by the Computer Systems Engineering group * at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and * contributed to Berkeley. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed by the University of * California, Berkeley and its contributors. * 4. Neither the name of the University nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * * @(#)scsiformat.c 5.5 (Berkeley) 4/2/94 */#ifndef lintchar copyright[] ="@(#) Copyright (c) 1992, 1993\n\ The Regents of the University of California. All rights reserved.\n";#endif /* not lint */#ifndef lintstatic char sccsid[] = "@(#)scsiformat.c 5.5 (Berkeley) 4/2/94";#endif /* not lint */#include <sys/param.h>#include <sys/ioctl.h>#include <dev/scsi/scsi.h>#include <dev/scsi/disk.h>#include <dev/scsi/disktape.h>#include <dev/scsi/scsi_ioctl.h>#define COMPAT_HPSCSI#include <errno.h>#include <fcntl.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#include <unistd.h>int fd;char *device;void scsi_str __P((char *, char *, int));void do_command __P((int, struct scsi_cdb *, void *, int));void do_format __P((void));void print_capacity __P((void));void print_inquiry __P((void));void prflags __P((int, const char *));u_char *print_mode_page __P((u_char *));void print_mode_sense __P((void));void usage __P((void));#define N2(c, d) (((c) << 8) | (d))#define N3(b, c, d) (((b) << 16) | N2(c, d))#define N4(a, b, c, d) (((a) << 24) | N3(b, c, d))int sense_pctl;intmain(argc, argv) int argc; char *argv[];{ extern char *optarg; int ch, readonly; readonly = 0; sense_pctl = SCSI_MSENSE_PCTL_CUR; while ((ch = getopt(argc, argv, "rp:")) != EOF) { switch(ch) { case 'r': readonly = 1; break; case 'p': /* mode sense page control */ switch (*optarg) { case 'c': sense_pctl = SCSI_MSENSE_PCTL_CUR; break; case 'd': sense_pctl = SCSI_MSENSE_PCTL_DFLT; break; case 's': sense_pctl = SCSI_MSENSE_PCTL_SAVED; break; case 'v': (void)printf( "*** note: for variable parameters, 1-bit means ``can write here''\n"); sense_pctl = SCSI_MSENSE_PCTL_VAR; break; } /* FALLTHROUGH */ case '?': default: usage(); } } argc -= optind; argv += optind; if (argc != 1) usage(); device = *argv; fd = open(device, readonly ? O_RDONLY : O_RDWR, 0); if (fd < 0) { (void)fprintf(stderr, "scsiformat: %s: %s\n", device, strerror(errno)); exit(1); } print_inquiry(); print_capacity(); print_mode_sense(); if (!readonly) do_format(); exit(0);}/* * Copy a counted string, trimming trailing blanks, and turning the * result into a C-style string. */voidscsi_str(src, dst, len) register char *src, *dst; register int len;{ while (src[len - 1] == ' ') { if (--len == 0) { *dst = 0; return; } } bcopy(src, dst, len); dst[len] = 0;}voidprint_inquiry(){ register struct scsi_inq_ansi *si; int ver; struct scsi_inquiry inqbuf; char vendor[10], product[17], rev[5]; static struct scsi_cdb inq = { CMD_INQUIRY, 0, 0, 0, sizeof(inqbuf), 0 }; do_command(fd, &inq, &inqbuf, sizeof(inqbuf)); (void)printf("%s: ", device); ver = (inqbuf.si_version >> VER_ANSI_SHIFT) & VER_ANSI_MASK; if (ver != 1 && ver != 2) { (void)printf("type 0x%x, qual 0x%x, ver 0x%x (ansi %d)\n", inqbuf.si_type, inqbuf.si_qual, inqbuf.si_version, ver); return; } si = (struct scsi_inq_ansi *)&inqbuf; switch (si->si_type & TYPE_TYPE_MASK) { case TYPE_DAD: (void)printf("(disk)"); break; case TYPE_WORM: (void)printf("(WORM)"); break; case TYPE_ROM: (void)printf("(CD-ROM)"); break; case TYPE_MO: (void)printf("(MO-DISK)"); break; case TYPE_JUKEBOX: (void)printf("(jukebox)"); break; default: (void)printf("(??)"); break; } scsi_str(si->si_vendor, vendor, sizeof(si->si_vendor)); scsi_str(si->si_product, product, sizeof(si->si_product)); scsi_str(si->si_rev, rev, sizeof(si->si_rev)); (void)printf(" %s %s rev %s:", vendor, product, rev);}voidprint_capacity(){ struct scsi_rc rc; /* for READ CAPACITY */ static struct scsi_cdb cap = { CMD_READ_CAPACITY }; do_command(fd, &cap, &rc, sizeof(rc)); (void)printf(" %d blocks of %d bytes each\n", N4(rc.rc_lbah, rc.rc_lbahm, rc.rc_lbalm, rc.rc_lbal) + 1, N4(rc.rc_blh, rc.rc_blhm, rc.rc_bllm, rc.rc_bll));}voidprint_mode_sense(){ register u_char *cp, *ep; register struct scsi_ms_bd *bd; register int n, i, l, len, bdlen;#ifdef TEN_BYTE_SENSE struct { struct scsi_ms10 ms; u_char p[1023 - sizeof(struct scsi_ms10)]; } msbuf; static struct scsi_cdb modesense = { CMD_MODE_SENSE10, SCSI_MSENSE_DBD, 0, 0, 0, 0, 0, sizeof(msbuf) >> 8, sizeof (msbuf), 0 }; CDB10(&modesense)->cdb_lbam = sense_pctl | SCSI_MS_PC_ALL; do_command(fd, &modesense, &msbuf, sizeof(msbuf)); len = N2(msbuf.ms.ms_lenh, msbuf.ms.ms_lenl); bdlen = N2(msbuf.ms.ms_bdlh, msbuf.ms.ms_bdll);#else struct { struct scsi_ms6 ms; u_char p[255 - sizeof(struct scsi_ms6)]; } msbuf; static struct scsi_cdb modesense = { CMD_MODE_SENSE6, 0, 0, 0, sizeof(msbuf), 0 }; CDB6(&modesense)->cdb_lbam = sense_pctl | SCSI_MS_PC_ALL; do_command(fd, &modesense, &msbuf, sizeof(msbuf)); len = msbuf.ms.ms_len; bdlen = msbuf.ms.ms_bdl;#endif (void)printf("\n%d bytes of mode sense data. ", len); (void)printf("medium type 0x%x, %swrite protected\n", msbuf.ms.ms_mt, msbuf.ms.ms_dsp & SCSI_MS_DSP_WP ? "" : "not "); if ((n = bdlen) != 0) { bd = (struct scsi_ms_bd *)msbuf.p; for (n /= sizeof(*bd); --n >= 0; bd++) { (void)printf("\tdensity code 0x%x, ", bd->bd_dc); i = N3(bd->bd_nbh, bd->bd_nbm, bd->bd_nbl); l = N3(bd->bd_blh, bd->bd_blm, bd->bd_bll); if (i) (void)printf("%d blocks of length %d\n", i, l); else (void)printf("all blocks of length %d\n", l); } } /* * Sense header lengths includes the sense header, while mode page * lengths do not ... let's hear it for consistency! */ cp = msbuf.p + bdlen; ep = msbuf.p + len - sizeof(msbuf.ms); while (cp < ep) cp = print_mode_page(cp);}voidprflags(v, cp) int v; register const char *cp;{ register const char *np; char f, sep; for (sep = '<'; (f = *cp++) != 0; cp = np) { for (np = cp; *np >= ' ';) np++; if ((v & (1 << (f - 1))) == 0) continue; printf("%c%.*s", sep, np - cp, cp); sep = ','; } if (sep != '<') putchar('>');}static char *cache_policy(x) int x;{ static char rsvd[30]; switch (x) { case SCSI_CACHE_DEFAULT: return ("default"); case SCSI_CACHE_KEEPPF: return ("toss cmd data, save prefetch"); case SCSI_CACHE_KEEPCMD: return ("toss prefetch data, save cmd"); default: (void)sprintf(rsvd, "reserved %d", x); return (rsvd); } /* NOTREACHED */}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -