• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 #ifndef SG_LIB_H
2 #define SG_LIB_H
3 
4 /*
5  * Copyright (c) 2004-2018 Douglas Gilbert.
6  * All rights reserved.
7  * Use of this source code is governed by a BSD-style
8  * license that can be found in the BSD_LICENSE file.
9  */
10 
11 /*
12  *
13  * On 5th October 2004 a FreeBSD license was added to this file.
14  * The intention is to keep this file and the related sg_lib.c file
15  * as open source and encourage their unencumbered use.
16  *
17  * Current version number is in the sg_lib.c file and can be accessed
18  * with the sg_lib_version() function.
19  */
20 
21 
22 /*
23  * This header file contains defines and function declarations that may
24  * be useful to applications that communicate with devices that use a
25  * SCSI command set. These command sets have names like SPC-4, SBC-3,
26  * SSC-3, SES-2 and draft standards defining them can be found at
27  * http://www.t10.org . Virtually all devices in the Linux SCSI subsystem
28  * utilize SCSI command sets. Many devices in other Linux device subsystems
29  * utilize SCSI command sets either natively or via emulation (e.g. a
30  * parallel ATA disk in a USB enclosure).
31  */
32 
33 #include <stdio.h>
34 #include <stdint.h>
35 #include <stdbool.h>
36 
37 #ifdef __cplusplus
38 extern "C" {
39 #endif
40 
41 /* SCSI Peripheral Device Types (PDT) [5 bit field] */
42 #define PDT_DISK 0x0    /* direct access block device (disk) */
43 #define PDT_TAPE 0x1    /* sequential access device (magnetic tape) */
44 #define PDT_PRINTER 0x2 /* printer device (see SSC-1) */
45 #define PDT_PROCESSOR 0x3       /* processor device (e.g. SAFTE device) */
46 #define PDT_WO 0x4      /* write once device (some optical disks) */
47 #define PDT_MMC 0x5     /* CD/DVD/BD (multi-media) */
48 #define PDT_SCANNER 0x6 /* obsolete */
49 #define PDT_OPTICAL 0x7 /* optical memory device (some optical disks) */
50 #define PDT_MCHANGER 0x8        /* media changer device (e.g. tape robot) */
51 #define PDT_COMMS 0x9   /* communications device (obsolete) */
52 #define PDT_SAC 0xc     /* storage array controller device */
53 #define PDT_SES 0xd     /* SCSI Enclosure Services (SES) device */
54 #define PDT_RBC 0xe     /* Reduced Block Commands (simplified PDT_DISK) */
55 #define PDT_OCRW 0xf    /* optical card read/write device */
56 #define PDT_BCC 0x10    /* bridge controller commands */
57 #define PDT_OSD 0x11    /* Object Storage Device (OSD) */
58 #define PDT_ADC 0x12    /* Automation/drive commands (ADC) */
59 #define PDT_SMD 0x13    /* Security Manager Device (SMD) */
60 #define PDT_ZBC 0x14    /* Zoned Block Commands (ZBC) */
61 #define PDT_WLUN 0x1e   /* Well known logical unit (WLUN) */
62 #define PDT_UNKNOWN 0x1f        /* Unknown or no device type */
63 
64 #ifndef SAM_STAT_GOOD
65 /* The SCSI status codes as found in SAM-4 at www.t10.org */
66 #define SAM_STAT_GOOD 0x0
67 #define SAM_STAT_CHECK_CONDITION 0x2
68 #define SAM_STAT_CONDITION_MET 0x4
69 #define SAM_STAT_BUSY 0x8
70 #define SAM_STAT_INTERMEDIATE 0x10              /* obsolete in SAM-4 */
71 #define SAM_STAT_INTERMEDIATE_CONDITION_MET 0x14  /* obsolete in SAM-4 */
72 #define SAM_STAT_RESERVATION_CONFLICT 0x18
73 #define SAM_STAT_COMMAND_TERMINATED 0x22        /* obsolete in SAM-3 */
74 #define SAM_STAT_TASK_SET_FULL 0x28
75 #define SAM_STAT_ACA_ACTIVE 0x30
76 #define SAM_STAT_TASK_ABORTED 0x40
77 #endif
78 
79 /* The SCSI sense key codes as found in SPC-4 at www.t10.org */
80 #define SPC_SK_NO_SENSE 0x0
81 #define SPC_SK_RECOVERED_ERROR 0x1
82 #define SPC_SK_NOT_READY 0x2
83 #define SPC_SK_MEDIUM_ERROR 0x3
84 #define SPC_SK_HARDWARE_ERROR 0x4
85 #define SPC_SK_ILLEGAL_REQUEST 0x5
86 #define SPC_SK_UNIT_ATTENTION 0x6
87 #define SPC_SK_DATA_PROTECT 0x7
88 #define SPC_SK_BLANK_CHECK 0x8
89 #define SPC_SK_VENDOR_SPECIFIC 0x9
90 #define SPC_SK_COPY_ABORTED 0xa
91 #define SPC_SK_ABORTED_COMMAND 0xb
92 #define SPC_SK_RESERVED 0xc
93 #define SPC_SK_VOLUME_OVERFLOW 0xd
94 #define SPC_SK_MISCOMPARE 0xe
95 #define SPC_SK_COMPLETED 0xf
96 
97 /* Transport protocol identifiers or just Protocol identifiers */
98 #define TPROTO_FCP 0
99 #define TPROTO_SPI 1
100 #define TPROTO_SSA 2
101 #define TPROTO_1394 3
102 #define TPROTO_SRP 4            /* SCSI over RDMA */
103 #define TPROTO_ISCSI 5
104 #define TPROTO_SAS 6
105 #define TPROTO_ADT 7
106 #define TPROTO_ATA 8
107 #define TPROTO_UAS 9            /* USB attached SCSI */
108 #define TPROTO_SOP 0xa          /* SCSI over PCIe */
109 #define TPROTO_PCIE 0xb         /* includes NVMe */
110 #define TPROTO_NONE 0xf
111 
112 /* SCSI Feature Sets (sfs) */
113 #define SCSI_FS_SPC_DISCOVERY_2016 0x1
114 #define SCSI_FS_SBC_BASE_2010 0x102
115 #define SCSI_FS_SBC_BASE_2016 0x101
116 #define SCSI_FS_SBC_BASIC_PROV_2016 0x103
117 #define SCSI_FS_SBC_DRIVE_MAINT_2016 0x104
118 
119 /* Often SCSI responses use the highest integer that can fit in a field
120  * to indicate "unbounded" or limit does not apply. Sometimes represented
121  * in output as "-1" for brevity */
122 #define SG_LIB_UNBOUNDED_16BIT 0xffff
123 #define SG_LIB_UNBOUNDED_32BIT 0xffffffffU
124 #define SG_LIB_UNBOUNDED_64BIT 0xffffffffffffffffULL
125 
126 #if (__STDC_VERSION__ >= 199901L)  /* C99 or later */
127     typedef uintptr_t sg_uintptr_t;
128 #else
129     typedef unsigned long sg_uintptr_t;
130 #endif
131 
132 
133 /* The format of the version string is like this: "2.26 20170906" */
134 const char * sg_lib_version();
135 
136 /* Returns length of SCSI command given the opcode (first byte).
137  * Yields the wrong answer for variable length commands (opcode=0x7f)
138  * and potentially some vendor specific commands. */
139 int sg_get_command_size(unsigned char cdb_byte0);
140 
141 /* Command name given pointer to the cdb. Certain command names
142  * depend on peripheral type (give 0 or -1 if unknown). Places command
143  * name into buff and will write no more than buff_len bytes. */
144 void sg_get_command_name(const unsigned char * cdbp, int peri_type,
145                          int buff_len, char * buff);
146 
147 /* Command name given only the first byte (byte 0) of a cdb and
148  * peripheral type (give 0 or -1 if unknown). */
149 void sg_get_opcode_name(unsigned char cdb_byte0, int peri_type, int buff_len,
150                         char * buff);
151 
152 /* Command name given opcode (byte 0), service action and peripheral type.
153  * If no service action give 0, if unknown peripheral type give 0 or -1 . */
154 void sg_get_opcode_sa_name(unsigned char cdb_byte0, int service_action,
155                            int peri_type, int buff_len, char * buff);
156 
157 /* Fetch scsi status string. */
158 void sg_get_scsi_status_str(int scsi_status, int buff_len, char * buff);
159 
160 /* This is a slightly stretched SCSI sense "descriptor" format header.
161  * The addition is to allow the 0x70 and 0x71 response codes. The idea
162  * is to place the salient data of both "fixed" and "descriptor" sense
163  * format into one structure to ease application processing.
164  * The original sense buffer should be kept around for those cases
165  * in which more information is required (e.g. the LBA of a MEDIUM ERROR). */
166 struct sg_scsi_sense_hdr {
167     unsigned char response_code; /* permit: 0x0, 0x70, 0x71, 0x72, 0x73 */
168     unsigned char sense_key;
169     unsigned char asc;
170     unsigned char ascq;
171     unsigned char byte4;
172     unsigned char byte5;
173     unsigned char byte6;
174     unsigned char additional_length;
175 };
176 
177 /* Maps the salient data from a sense buffer which is in either fixed or
178  * descriptor format into a structure mimicking a descriptor format
179  * header (i.e. the first 8 bytes of sense descriptor format).
180  * If zero response code returns false. Otherwise returns true and if 'sshp'
181  * is non-NULL then zero all fields and then set the appropriate fields in
182  * that structure. sshp::additional_length is always 0 for response
183  * codes 0x70 and 0x71 (fixed format). */
184 bool sg_scsi_normalize_sense(const unsigned char * sensep, int sense_len,
185                              struct sg_scsi_sense_hdr * sshp);
186 
187 /* Attempt to find the first SCSI sense data descriptor that matches the
188  * given 'desc_type'. If found return pointer to start of sense data
189  * descriptor; otherwise (including fixed format sense data) returns NULL. */
190 const unsigned char * sg_scsi_sense_desc_find(const unsigned char * sensep,
191                                               int sense_len, int desc_type);
192 
193 /* Get sense key from sense buffer. If successful returns a sense key value
194  * between 0 and 15. If sense buffer cannot be decode, returns -1 . */
195 int sg_get_sense_key(const unsigned char * sensep, int sense_len);
196 
197 /* Yield string associated with sense_key value. Returns 'buff'. */
198 char * sg_get_sense_key_str(int sense_key, int buff_len, char * buff);
199 
200 /* Yield string associated with ASC/ASCQ values. Returns 'buff'. */
201 char * sg_get_asc_ascq_str(int asc, int ascq, int buff_len, char * buff);
202 
203 /* Returns true if valid bit set, false if valid bit clear. Irrespective the
204  * information field is written out via 'info_outp' (except when it is
205  * NULL). Handles both fixed and descriptor sense formats. */
206 bool sg_get_sense_info_fld(const unsigned char * sensep, int sb_len,
207                            uint64_t * info_outp);
208 
209 /* Returns true if fixed format or command specific information descriptor
210  * is found in the descriptor sense; else false. If available the command
211  * specific information field (4 byte integer in fixed format, 8 byte
212  * integer in descriptor format) is written out via 'cmd_spec_outp'.
213  * Handles both fixed and descriptor sense formats. */
214 bool sg_get_sense_cmd_spec_fld(const unsigned char * sensep, int sb_len,
215                                uint64_t * cmd_spec_outp);
216 
217 /* Returns true if any of the 3 bits (i.e. FILEMARK, EOM or ILI) are set.
218  * In descriptor format if the stream commands descriptor not found
219  * then returns false. Writes true or false corresponding to these bits to
220  * the last three arguments if they are non-NULL. */
221 bool sg_get_sense_filemark_eom_ili(const unsigned char * sensep, int sb_len,
222                                    bool * filemark_p, bool * eom_p,
223                                    bool * ili_p);
224 
225 /* Returns true if SKSV is set and sense key is NO_SENSE or NOT_READY. Also
226  * returns true if progress indication sense data descriptor found. Places
227  * progress field from sense data where progress_outp points. If progress
228  * field is not available returns false. Handles both fixed and descriptor
229  * sense formats. N.B. App should multiply by 100 and divide by 65536
230  * to get percentage completion from given value. */
231 bool sg_get_sense_progress_fld(const unsigned char * sensep, int sb_len,
232                                int * progress_outp);
233 
234 /* Closely related to sg_print_sense(). Puts decoded sense data in 'buff'.
235  * Usually multiline with multiple '\n' including one trailing. If
236  * 'raw_sinfo' set appends sense buffer in hex. 'leadin' is string prepended
237  * to each line written to 'buff', NULL treated as "". Returns the number of
238  * bytes written to 'buff' excluding the trailing '\0'.
239  * N.B. prior to sg3_utils v 1.42 'leadin' was only prepended to the first
240  * line output. Also this function returned type void. */
241 int sg_get_sense_str(const char * leadin, const unsigned char * sense_buffer,
242                      int sb_len, bool raw_sinfo, int buff_len, char * buff);
243 
244 /* Decode descriptor format sense descriptors (assumes sense buffer is
245  * in descriptor format). 'leadin' is string prepended to each line written
246  * to 'b', NULL treated as "". Returns the number of bytes written to 'b'
247  * excluding the trailing '\0'. */
248 int sg_get_sense_descriptors_str(const char * leadin,
249                                  const unsigned char * sense_buffer,
250                                  int sb_len, int blen, char * b);
251 
252 /* Decodes a designation descriptor (e.g. as found in the Device
253  * Identification VPD page (0x83)) into string 'b' whose maximum length is
254  * blen. 'leadin' is string prepended to each line written to 'b', NULL
255  * treated as "". Returns the number of bytes written to 'b' excluding the
256  * trailing '\0'. */
257 int sg_get_designation_descriptor_str(const char * leadin,
258                                       const unsigned char * ddp, int dd_len,
259                                       bool print_assoc, bool do_long,
260                                       int blen, char * b);
261 
262 /* Yield string associated with peripheral device type (pdt). Returns
263  * 'buff'. If 'pdt' out of range yields "bad pdt" string. */
264 char * sg_get_pdt_str(int pdt, int buff_len, char * buff);
265 
266 /* Some lesser used PDTs share a lot in common with a more used PDT.
267  * Examples are PDT_ADC decaying to PDT_TAPE and PDT_ZBC to PDT_DISK.
268  * If such a lesser used 'pdt' is given to this function, then it will
269  * return the more used PDT (i.e. "decays to"); otherwise 'pdt' is returned.
270  * Valid for 'pdt' 0 to 31, for other values returns 0. */
271 int sg_lib_pdt_decay(int pdt);
272 
273 /* Yield string associated with transport protocol identifier (tpi). Returns
274  * 'buff'. If 'tpi' out of range yields "bad tpi" string. */
275 char * sg_get_trans_proto_str(int tpi, int buff_len, char * buff);
276 
277 /* Decode TransportID pointed to by 'bp' of length 'bplen'. Place decoded
278  * string output in 'buff' which is also the return value. Each new line
279  * is prefixed by 'leadin'. If leadin NULL treat as "". */
280 char * sg_decode_transportid_str(const char * leadin, unsigned char * bp,
281                                  int bplen, bool only_one, int buff_len,
282                                  char * buff);
283 
284 /* Returns a designator's type string given 'val' (0 to 15 inclusive),
285  * otherwise returns NULL. */
286 const char * sg_get_desig_type_str(int val);
287 
288 /* Returns a designator's code_set string given 'val' (0 to 15 inclusive),
289  * otherwise returns NULL. */
290 const char * sg_get_desig_code_set_str(int val);
291 
292 /* Returns a designator's association string given 'val' (0 to 3 inclusive),
293  * otherwise returns NULL. */
294 const char * sg_get_desig_assoc_str(int val);
295 
296 /* Yield SCSI Feature Set (sfs) string. When 'peri_type' is < -1 (or > 31)
297  * returns pointer to string (same as 'buff') associated with 'sfs_code'.
298  * When 'peri_type' is between -1 (for SPC) and 31 (inclusive) then a match
299  * on both 'sfs_code' and 'peri_type' is required. If 'foundp' is not NULL
300  * then where it points is set to true if a match is found else it is set to
301  * false. If 'buff' is not NULL then in the case of a match a descriptive
302  * string is written to 'buff' while if there is not a not then a string
303  * ending in "Reserved" is written (and may be prefixed with SPC, SBC, SSC
304  * or ZBC). Returns 'buff' (i.e. a pointer value) even if it is NULL.
305  * Example:
306  *    char b[64];
307  *    ...
308  *    printf("%s\n", sg_get_sfs_str(sfs_code, -2, sizeof(b), b, NULL, 0));
309  */
310 const char * sg_get_sfs_str(uint16_t sfs_code, int peri_type, int buff_len,
311                             char * buff, bool * foundp, int verbose);
312 
313 /* This is a heuristic that takes into account the command bytes and length
314  * to decide whether the presented unstructured sequence of bytes could be
315  * a SCSI command. If so it returns true otherwise false. Vendor specific
316  * SCSI commands (i.e. opcodes from 0xc0 to 0xff), if presented, are assumed
317  * to follow SCSI conventions (i.e. length of 6, 10, 12 or 16 bytes). The
318  * only SCSI commands considered above 16 bytes of length are the Variable
319  * Length Commands (opcode 0x7f) and the XCDB wrapped commands (opcode 0x7e).
320  * Both have an inbuilt length field which can be cross checked with clen.
321  * No NVMe commands (64 bytes long plus some extra added by some OSes) have
322  * opcodes 0x7e or 0x7f yet. ATA is register based but SATA has FIS
323  * structures that are sent across the wire. The 'FIS register' structure is
324  * used to move a command from a SATA host to device, but the ATA 'command'
325  * is not the first byte. So it is harder to say what will happen if a
326  * FIS structure is presented as a SCSI command, hopfully there is a low
327  * probability this function will yield true in that case. */
328 bool sg_is_scsi_cdb(const uint8_t * cdbp, int clen);
329 
330 /* Yield string associated with NVMe command status value in sct_sc. It
331  * expects to decode DW3 bits 27:17 from the completion queue. Bits 27:25
332  * are the Status Code Type (SCT) and bits 24:17 are the Status Code (SC).
333  * Bit 17 in DW3 should be bit 0 in sct_sc. If no status string is found
334  * a string of the form "Reserved [0x<sct_sc_in_hex>]" is generated.
335  * Returns 'buff'. Does nothing if buff_len<=0 or if buff is NULL.*/
336 char * sg_get_nvme_cmd_status_str(uint16_t sct_sc, int buff_len, char * buff);
337 
338 /* Attempts to map NVMe status value ((SCT << 8) | SC) n sct_sc to a SCSI
339  * status, sense_key, asc and ascq tuple. If successful returns true and
340  * writes to non-NULL pointer arguments; otherwise returns false. */
341 bool sg_nvme_status2scsi(uint16_t sct_sc, uint8_t * status_p, uint8_t * sk_p,
342                          uint8_t * asc_p, uint8_t * ascq_p);
343 
344 extern FILE * sg_warnings_strm;
345 
346 void sg_set_warnings_strm(FILE * warnings_strm);
347 
348 /* The following "print" functions send ACSII to 'sg_warnings_strm' file
349  * descriptor (default value is stderr). 'leadin' is string prepended to
350  * each line printed out, NULL treated as "". */
351 void sg_print_command(const unsigned char * command);
352 void sg_print_scsi_status(int scsi_status);
353 
354 /* 'leadin' is string prepended to each line printed out, NULL treated as
355  * "". N.B. prior to sg3_utils v 1.42 'leadin' was only prepended to the
356  * first line printed. */
357 void sg_print_sense(const char * leadin, const unsigned char * sense_buffer,
358                     int sb_len, bool raw_info);
359 
360 /* Following examines exit_status and outputs a clear error message to
361  * warnings_strm (usually stderr) if one is known and returns true.
362  * Otherwise it doesn't print anything and returns false. Note that if
363  * exit_status==0 then returns true but prints nothing and if
364  * exit_status<0 ("some error occurred") false is returned. If leadin is
365  * non-NULL is will be printed before error message. */
366 bool sg_if_can2stderr(const char * leadin, int exit_status);
367 
368 /* Utilities can use these exit status values for syntax errors and
369  * file (device node) problems (e.g. not found or permissions). */
370 #define SG_LIB_SYNTAX_ERROR 1   /* command line syntax problem */
371 #define SG_LIB_FILE_ERROR 15    /* device or other file problem */
372 
373 /* The sg_err_category_sense() function returns one of the following.
374  * These may be used as exit status values (from a process). Notice that
375  * some of the lower values correspond to SCSI sense key values. */
376 #define SG_LIB_CAT_CLEAN 0      /* No errors or other information */
377 /* Value 1 left unused for utilities to use SG_LIB_SYNTAX_ERROR */
378 #define SG_LIB_CAT_NOT_READY 2  /* sense key, unit stopped? */
379                                 /*       [sk,asc,ascq: 0x2,*,*] */
380 #define SG_LIB_CAT_MEDIUM_HARD 3 /* medium or hardware error, blank check */
381                                 /*       [sk,asc,ascq: 0x3/0x4/0x8,*,*] */
382 #define SG_LIB_CAT_ILLEGAL_REQ 5 /* Illegal request (other than invalid */
383                                 /* opcode):   [sk,asc,ascq: 0x5,*,*] */
384 #define SG_LIB_CAT_UNIT_ATTENTION 6 /* sense key, device state changed */
385                                 /*       [sk,asc,ascq: 0x6,*,*] */
386         /* was SG_LIB_CAT_MEDIA_CHANGED earlier [sk,asc,ascq: 0x6,0x28,*] */
387 #define SG_LIB_CAT_DATA_PROTECT 7 /* sense key, media write protected? */
388                                 /*       [sk,asc,ascq: 0x7,*,*] */
389 #define SG_LIB_CAT_INVALID_OP 9 /* (Illegal request,) Invalid opcode: */
390                                 /*       [sk,asc,ascq: 0x5,0x20,0x0] */
391 #define SG_LIB_CAT_COPY_ABORTED 10 /* sense key, some data transferred */
392                                 /*       [sk,asc,ascq: 0xa,*,*] */
393 #define SG_LIB_CAT_ABORTED_COMMAND 11 /* interpreted from sense buffer */
394                                 /*       [sk,asc,ascq: 0xb,! 0x10,*] */
395 #define SG_LIB_CAT_MISCOMPARE 14 /* sense key, probably verify */
396                                 /*       [sk,asc,ascq: 0xe,*,*] */
397 #define SG_LIB_CAT_NO_SENSE 20  /* sense data with key of "no sense" */
398                                 /*       [sk,asc,ascq: 0x0,*,*] */
399 #define SG_LIB_CAT_RECOVERED 21 /* Successful command after recovered err */
400                                 /*       [sk,asc,ascq: 0x1,*,*] */
401 #define SG_LIB_CAT_RES_CONFLICT SAM_STAT_RESERVATION_CONFLICT
402                                 /* 24: this is a SCSI status, not sense. */
403                                 /* It indicates reservation by another */
404                                 /* machine blocks this command */
405 #define SG_LIB_CAT_CONDITION_MET 25 /* SCSI status, not sense key. */
406                                     /* Only from PRE-FETCH (SBC-4) */
407 #define SG_LIB_CAT_BUSY       26 /* SCSI status, not sense. Invites retry */
408 #define SG_LIB_CAT_TS_FULL    27 /* SCSI status, not sense. Wait then retry */
409 #define SG_LIB_CAT_ACA_ACTIVE 28 /* SCSI status; ACA seldom used */
410 #define SG_LIB_CAT_TASK_ABORTED 29 /* SCSI status, this command aborted by? */
411 #define SG_LIB_CAT_PROTECTION 40 /* subset of aborted command (for PI, DIF) */
412                                 /*       [sk,asc,ascq: 0xb,0x10,*] */
413 #define SG_LIB_NVME_STATUS 48   /* NVMe Status Field (SF) other than 0 */
414 #define SG_LIB_WILD_RESID 49    /* Residual value for data-in transfer of a */
415                                 /* SCSI command is nonsensical */
416 #define SG_LIB_OS_BASE_ERR 50   /* in Linux: values found in: */
417                                 /* include/uapi/asm-generic/errno-base.h */
418                                 /* Example: ENOMEM reported as 62 (=50+12) */
419 #define SG_LIB_CAT_MALFORMED 97 /* Response to SCSI command malformed */
420 #define SG_LIB_CAT_SENSE 98     /* Something else is in the sense buffer */
421 #define SG_LIB_CAT_OTHER 99     /* Some other error/warning has occurred */
422                                 /* (e.g. a transport or driver error) */
423 
424 /* Returns a SG_LIB_CAT_* value. If cannot decode sense_buffer or a less
425  * common sense key then return SG_LIB_CAT_SENSE .*/
426 int sg_err_category_sense(const unsigned char * sense_buffer, int sb_len);
427 
428 /* Here are some additional sense data categories that are not returned
429  * by sg_err_category_sense() but are returned by some related functions. */
430 #define SG_LIB_CAT_ILLEGAL_REQ_WITH_INFO 17 /* Illegal request (other than */
431                                 /* invalid opcode) plus 'info' field: */
432                                 /*  [sk,asc,ascq: 0x5,*,*] */
433 #define SG_LIB_CAT_MEDIUM_HARD_WITH_INFO 18 /* medium or hardware error */
434                                 /* sense key plus 'info' field: */
435                                 /*       [sk,asc,ascq: 0x3/0x4,*,*] */
436 #define SG_LIB_CAT_PROTECTION_WITH_INFO 41 /* aborted command sense key, */
437                                 /* protection plus 'info' field: */
438                                 /*  [sk,asc,ascq: 0xb,0x10,*] */
439 #define SG_LIB_CAT_TIMEOUT 33
440 
441 /* Yield string associated with sense category. Returns 'buff' (or pointer
442  * to "Bad sense category" if 'buff' is NULL). If sense_cat unknown then
443  * yield "Sense category: <sense_cat>" string. */
444 const char * sg_get_category_sense_str(int sense_cat, int buff_len,
445                                        char * buff, int verbose);
446 
447 
448 /* Iterates to next designation descriptor in the device identification
449  * VPD page. The 'initial_desig_desc' should point to start of first
450  * descriptor with 'page_len' being the number of valid bytes in that
451  * and following descriptors. To start, 'off' should point to a negative
452  * value, thereafter it should point to the value yielded by the previous
453  * call. If 0 returned then 'initial_desig_desc + *off' should be a valid
454  * descriptor; returns -1 if normal end condition and -2 for an abnormal
455  * termination. Matches association, designator_type and/or code_set when
456  * any of those values are greater than or equal to zero. */
457 int sg_vpd_dev_id_iter(const unsigned char * initial_desig_desc, int page_len,
458                        int * off, int m_assoc, int m_desig_type,
459                        int m_code_set);
460 
461 
462 /* <<< General purpose (i.e. not SCSI specific) utility functions >>> */
463 
464 /* Always returns valid string even if errnum is wild (or library problem).
465  * If errnum is negative, flip its sign. */
466 char * safe_strerror(int errnum);
467 
468 
469 /* Print (to stdout) 'str' of bytes in hex, 16 bytes per line optionally
470  * followed at the right hand side of the line with an ASCII interpretation.
471  * Each line is prefixed with an address, starting at 0 for str[0]..str[15].
472  * All output numbers are in hex. 'no_ascii' allows for 3 output types:
473  *     > 0     each line has address then up to 16 ASCII-hex bytes
474  *     = 0     in addition, the bytes are listed in ASCII to the right
475  *     < 0     only the ASCII-hex bytes are listed (i.e. without address)
476 */
477 void dStrHex(const char * str, int len, int no_ascii);
478 
479 /* Print (to sg_warnings_strm (stderr)) 'str' of bytes in hex, 16 bytes per
480  * line optionally followed at right by its ASCII interpretation. Same
481  * logic as dStrHex() with different output stream (i.e. stderr). */
482 void dStrHexErr(const char * str, int len, int no_ascii);
483 
484 /* Read 'len' bytes from 'str' and output as ASCII-Hex bytes (space
485  * separated) to 'b' not to exceed 'b_len' characters. Each line
486  * starts with 'leadin' (NULL for no leadin) and there are 16 bytes
487  * per line with an extra space between the 8th and 9th bytes. 'format'
488  * is 0 for repeat in printable ASCII ('.' for non printable chars) to
489  * right of each line; 1 don't (so just output ASCII hex). Returns
490  * number of bytes written to 'b' excluding the trailing '\0'. */
491 int dStrHexStr(const char * str, int len, const char * leadin, int format,
492                int cb_len, char * cbp);
493 
494 /* The following 3 functions are equivalent to dStrHex(), dStrHexErr() and
495  * dStrHexStr() respectively. The difference is the type of the first of
496  * argument: uint8_t instead of char. The name of the argument is changed
497  * to b_str to stress it is a pointer to the start of a binary string. */
498 void hex2stdout(const uint8_t * b_str, int len, int no_ascii);
499 void hex2stderr(const uint8_t * b_str, int len, int no_ascii);
500 int hex2str(const uint8_t * b_str, int len, const char * leadin, int format,
501             int cb_len, char * cbp);
502 
503 /* Returns true when executed on big endian machine; else returns false.
504  * Useful for displaying ATA identify words (which need swapping on a
505  * big endian machine). */
506 bool sg_is_big_endian();
507 
508 /* Returns true if byte sequence starting at bp with a length of b_len is
509  * all zeros (for sg_all_zeros()) or all 0xff_s (for sg_all_ffs());
510  * otherwise returns false. If bp is NULL ir b_len <= 0 returns false. */
511 bool sg_all_zeros(const uint8_t * bp, int b_len);
512 bool sg_all_ffs(const uint8_t * bp, int b_len);
513 
514 /* Extract character sequence from ATA words as in the model string
515  * in a IDENTIFY DEVICE response. Returns number of characters
516  * written to 'ochars' before 0 character is found or 'num' words
517  * are processed. */
518 int sg_ata_get_chars(const uint16_t * word_arr, int start_word,
519                      int num_words, bool is_big_endian, char * ochars);
520 
521 /* Print (to stdout) 16 bit 'words' in hex, 8 words per line optionally
522  * followed at the right hand side of the line with an ASCII interpretation
523  * (pairs of ASCII characters in big endian order (upper first)).
524  * Each line is prefixed with an address, starting at 0.
525  * All output numbers are in hex. 'no_ascii' allows for 3 output types:
526  *     > 0     each line has address then up to 8 ASCII-hex words
527  *     = 0     in addition, the words are listed in ASCII pairs to the right
528  *     = -1    only the ASCII-hex words are listed (i.e. without address)
529  *     = -2    only the ASCII-hex words, formatted for "hdparm --Istdin"
530  *     < -2    same as -1
531  * If 'swapb' is true then bytes in each word swapped. Needs to be set
532  * for ATA IDENTIFY DEVICE response on big-endian machines.
533 */
534 void dWordHex(const uint16_t * words, int num, int no_ascii, bool swapb);
535 
536 /* If the number in 'buf' can not be decoded or the multiplier is unknown
537  * then -1 is returned. Accepts a hex prefix (0x or 0X) or a 'h' (or 'H')
538  * suffix. Otherwise a decimal multiplier suffix may be given. Recognised
539  * multipliers: c C  *1;  w W  *2; b  B *512;  k K KiB  *1,024;
540  * KB  *1,000;  m M MiB  *1,048,576; MB *1,000,000; g G GiB *1,073,741,824;
541  * GB *1,000,000,000 and <n>x<m> which multiplies <n> by <m> . Ignore leading
542  * spaces and tabs; accept comma, hyphen, space, tab and hash as terminator.
543  */
544 int sg_get_num(const char * buf);
545 
546 /* If the number in 'buf' can not be decoded then -1 is returned. Accepts a
547  * hex prefix (0x or 0X) or a 'h' (or 'H') suffix; otherwise decimal is
548  * assumed. Does not accept multipliers. Accept a comma (","), hyphen ("-"),
549  * a whitespace or newline as terminator. Only decimal numbers can represent
550  * negative numbers and '-1' must be treated separately. */
551 int sg_get_num_nomult(const char * buf);
552 
553 /* If the number in 'buf' can not be decoded or the multiplier is unknown
554  * then -1LL is returned. Accepts a hex prefix (0x or 0X) or a 'h' (or 'H')
555  * suffix. Otherwise a decimal multiplier suffix may be given. In addition
556  * to supporting the multipliers of sg_get_num(), this function supports:
557  * t T TiB  *(2**40); TB *(10**12); p P PiB  *(2**50); PB  *(10**15) .
558  * Ignore leading spaces and tabs; accept comma, hyphen, space, tab and hash
559  * as terminator. */
560 int64_t sg_get_llnum(const char * buf);
561 
562 /* If the number in 'buf' can not be decoded then -1 is returned. Accepts a
563  * hex prefix (0x or 0X) or a 'h' (or 'H') suffix; otherwise decimal is
564  * assumed. Does not accept multipliers. Accept a comma (","), hyphen ("-"),
565  * a whitespace or newline as terminator. Only decimal numbers can represent
566  * negative numbers and '-1' must be treated separately. */
567 int64_t sg_get_llnum_nomult(const char * buf);
568 
569 /* Returns pointer to heap (or NULL) that is aligned to a align_to byte
570  * boundary. Sends back *buff_to_free pointer in third argument that may be
571  * different from the return value. If it is different then the *buff_to_free
572  * pointer should be freed (rather than the returned value) when the heap is
573  * no longer needed. If align_to is 0 then aligns to OS's page size. Sets all
574  * returned heap to zeros. If num_bytes is 0 then set to page size. */
575 uint8_t * sg_memalign(uint32_t num_bytes, uint32_t align_to,
576                       uint8_t ** buff_to_free, bool vb);
577 
578 /* Returns OS page size in bytes. If uncertain returns 4096. */
579 uint32_t sg_get_page_size(void);
580 
581 /* If os_err_num is within bounds then the returned value is 'os_err_num +
582  * SG_LIB_OS_BASE_ERR' otherwise -1 is returned. If os_err_num is 0 then 0
583  * is returned. */
584 int sg_convert_errno(int os_err_num);
585 
586 
587 /* <<< Architectural support functions [is there a better place?] >>> */
588 
589 /* Non Unix OSes distinguish between text and binary files.
590  * Set text mode on fd. Does nothing in Unix. Returns negative number on
591  * failure. */
592 int sg_set_text_mode(int fd);
593 
594 /* Set binary mode on fd. Does nothing in Unix. Returns negative number on
595  * failure. */
596 int sg_set_binary_mode(int fd);
597 
598 #ifdef __cplusplus
599 }
600 #endif
601 
602 #endif          /* SG_LIB_H */
603