xref: /aosp_15_r20/external/sg3_utils/src/sg_luns.c (revision 44704f698541f6367e81f991ef8bb54ccbf3fc18)
1*44704f69SBart Van Assche /*
2*44704f69SBart Van Assche  * Copyright (c) 2004-2021 Douglas Gilbert.
3*44704f69SBart Van Assche  * All rights reserved.
4*44704f69SBart Van Assche  * Use of this source code is governed by a BSD-style
5*44704f69SBart Van Assche  * license that can be found in the BSD_LICENSE file.
6*44704f69SBart Van Assche  *
7*44704f69SBart Van Assche  * SPDX-License-Identifier: BSD-2-Clause
8*44704f69SBart Van Assche  */
9*44704f69SBart Van Assche 
10*44704f69SBart Van Assche #include <unistd.h>
11*44704f69SBart Van Assche #include <fcntl.h>
12*44704f69SBart Van Assche #include <stdio.h>
13*44704f69SBart Van Assche #include <stdlib.h>
14*44704f69SBart Van Assche #include <stdbool.h>
15*44704f69SBart Van Assche #include <string.h>
16*44704f69SBart Van Assche #include <errno.h>
17*44704f69SBart Van Assche #include <ctype.h>
18*44704f69SBart Van Assche #include <getopt.h>
19*44704f69SBart Van Assche #define __STDC_FORMAT_MACROS 1
20*44704f69SBart Van Assche #include <inttypes.h>
21*44704f69SBart Van Assche 
22*44704f69SBart Van Assche #ifdef HAVE_CONFIG_H
23*44704f69SBart Van Assche #include "config.h"
24*44704f69SBart Van Assche #endif
25*44704f69SBart Van Assche #include "sg_lib.h"
26*44704f69SBart Van Assche #include "sg_cmds_basic.h"
27*44704f69SBart Van Assche #include "sg_unaligned.h"
28*44704f69SBart Van Assche #include "sg_pr2serr.h"
29*44704f69SBart Van Assche 
30*44704f69SBart Van Assche /* A utility program originally written for the Linux OS SCSI subsystem.
31*44704f69SBart Van Assche  *
32*44704f69SBart Van Assche  *
33*44704f69SBart Van Assche  * This program issues the SCSI REPORT LUNS command to the given SCSI device
34*44704f69SBart Van Assche  * and decodes the response.
35*44704f69SBart Van Assche  */
36*44704f69SBart Van Assche 
37*44704f69SBart Van Assche static const char * version_str = "1.48 20210804";      /* spc6r05 */
38*44704f69SBart Van Assche 
39*44704f69SBart Van Assche #define MAX_RLUNS_BUFF_LEN (1024 * 1024)
40*44704f69SBart Van Assche #define DEF_RLUNS_BUFF_LEN (1024 * 8)
41*44704f69SBart Van Assche 
42*44704f69SBart Van Assche 
43*44704f69SBart Van Assche static struct option long_options[] = {
44*44704f69SBart Van Assche         {"decode", no_argument, 0, 'd'},
45*44704f69SBart Van Assche         {"help", no_argument, 0, 'h'},
46*44704f69SBart Van Assche         {"hex", no_argument, 0, 'H'},
47*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
48*44704f69SBart Van Assche         {"linux", no_argument, 0, 'l'},
49*44704f69SBart Van Assche #endif
50*44704f69SBart Van Assche         {"lu_cong", no_argument, 0, 'L'},
51*44704f69SBart Van Assche         {"lu-cong", no_argument, 0, 'L'},
52*44704f69SBart Van Assche         {"maxlen", required_argument, 0, 'm'},
53*44704f69SBart Van Assche         {"quiet", no_argument, 0, 'q'},
54*44704f69SBart Van Assche         {"raw", no_argument, 0, 'r'},
55*44704f69SBart Van Assche         {"readonly", no_argument, 0, 'R'},
56*44704f69SBart Van Assche         {"select", required_argument, 0, 's'},
57*44704f69SBart Van Assche         {"test", required_argument, 0, 't'},
58*44704f69SBart Van Assche         {"verbose", no_argument, 0, 'v'},
59*44704f69SBart Van Assche         {"version", no_argument, 0, 'V'},
60*44704f69SBart Van Assche         {0, 0, 0, 0},
61*44704f69SBart Van Assche };
62*44704f69SBart Van Assche 
63*44704f69SBart Van Assche 
64*44704f69SBart Van Assche static void
usage()65*44704f69SBart Van Assche usage()
66*44704f69SBart Van Assche {
67*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
68*44704f69SBart Van Assche     pr2serr("Usage: sg_luns    [--decode] [--help] [--hex] [--linux] "
69*44704f69SBart Van Assche             "[--lu_cong]\n"
70*44704f69SBart Van Assche             "                  [--maxlen=LEN] [--quiet] [--raw] "
71*44704f69SBart Van Assche             "[--readonly]\n"
72*44704f69SBart Van Assche             "                  [--select=SR] [--verbose] [--version] "
73*44704f69SBart Van Assche             "DEVICE\n");
74*44704f69SBart Van Assche #else
75*44704f69SBart Van Assche     pr2serr("Usage: sg_luns    [--decode] [--help] [--hex] [--lu_cong] "
76*44704f69SBart Van Assche             "[--maxlen=LEN]\n"
77*44704f69SBart Van Assche             "                  [--quiet] [--raw] [--readonly] "
78*44704f69SBart Van Assche             "[--select=SR]\n"
79*44704f69SBart Van Assche             "                  [--verbose] [--version] DEVICE\n");
80*44704f69SBart Van Assche #endif
81*44704f69SBart Van Assche     pr2serr("     or\n"
82*44704f69SBart Van Assche             "       sg_luns    --test=ALUN [--decode] [--hex] [--lu_cong] "
83*44704f69SBart Van Assche             "[--verbose]\n"
84*44704f69SBart Van Assche             "  where:\n"
85*44704f69SBart Van Assche             "    --decode|-d        decode all luns into component parts\n"
86*44704f69SBart Van Assche             "    --help|-h          print out usage message\n"
87*44704f69SBart Van Assche             "    --hex|-H           output response in hexadecimal; used "
88*44704f69SBart Van Assche             "twice\n"
89*44704f69SBart Van Assche             "                       shows decoded values in hex\n");
90*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
91*44704f69SBart Van Assche     pr2serr("    --linux|-l         show Linux integer lun after T10 "
92*44704f69SBart Van Assche             "representation\n");
93*44704f69SBart Van Assche #endif
94*44704f69SBart Van Assche     pr2serr("    --lu_cong|-L       decode as if LU_CONG is set; used "
95*44704f69SBart Van Assche             "twice:\n"
96*44704f69SBart Van Assche             "                       decode as if LU_CONG is clear\n"
97*44704f69SBart Van Assche             "    --maxlen=LEN|-m LEN    max response length (allocation "
98*44704f69SBart Van Assche             "length in cdb)\n"
99*44704f69SBart Van Assche             "                           (def: 0 -> %d bytes)\n"
100*44704f69SBart Van Assche             "    --quiet|-q         output only ASCII hex lun values\n"
101*44704f69SBart Van Assche             "    --raw|-r           output response in binary\n"
102*44704f69SBart Van Assche             "    --readonly|-R      open DEVICE read-only (def: read-write)\n"
103*44704f69SBart Van Assche             "    --select=SR|-s SR    select report SR (def: 0)\n"
104*44704f69SBart Van Assche             "                          0 -> luns apart from 'well "
105*44704f69SBart Van Assche             "known' lus\n"
106*44704f69SBart Van Assche             "                          1 -> only 'well known' "
107*44704f69SBart Van Assche             "logical unit numbers\n"
108*44704f69SBart Van Assche             "                          2 -> all luns\n"
109*44704f69SBart Van Assche             "                          0x10 -> administrative luns\n"
110*44704f69SBart Van Assche             "                          0x11 -> admin luns + "
111*44704f69SBart Van Assche             "non-conglomerate luns\n"
112*44704f69SBart Van Assche             "                          0x12 -> admin lun + its "
113*44704f69SBart Van Assche             "subsidiary luns\n"
114*44704f69SBart Van Assche             "    --test=ALUN|-t ALUN    decode ALUN and ignore most other "
115*44704f69SBart Van Assche             "options\n"
116*44704f69SBart Van Assche             "                           and DEVICE (apart from '-H')\n"
117*44704f69SBart Van Assche             "    --verbose|-v       increase verbosity\n"
118*44704f69SBart Van Assche             "    --version|-V       print version string and exit\n\n"
119*44704f69SBart Van Assche             "Performs a SCSI REPORT LUNS command or decodes the given ALUN. "
120*44704f69SBart Van Assche             "When SR is\n0x10 or 0x11 DEVICE must be LUN 0 or REPORT LUNS "
121*44704f69SBart Van Assche             "well known logical unit;\nwhen SR is 0x12 DEVICE must be an "
122*44704f69SBart Van Assche             "administrative logical unit. When the\n--test=ALUN option is "
123*44704f69SBart Van Assche             "given, decodes ALUN rather than sending a REPORT\nLUNS "
124*44704f69SBart Van Assche             "command.\n", DEF_RLUNS_BUFF_LEN );
125*44704f69SBart Van Assche }
126*44704f69SBart Van Assche 
127*44704f69SBart Van Assche /* Decoded according to SAM-5 rev 10. Note that one draft: BCC rev 0,
128*44704f69SBart Van Assche  * defines its own "bridge addressing method" in place of the SAM-3
129*44704f69SBart Van Assche  * "logical addressing method".  */
130*44704f69SBart Van Assche static void
decode_lun(const char * leadin,const uint8_t * lunp,bool lu_cong,int do_hex,int verbose)131*44704f69SBart Van Assche decode_lun(const char * leadin, const uint8_t * lunp, bool lu_cong,
132*44704f69SBart Van Assche            int do_hex, int verbose)
133*44704f69SBart Van Assche {
134*44704f69SBart Van Assche     bool next_level, admin_lu_cong;
135*44704f69SBart Van Assche     int k, x, a_method, bus_id, target, lun, len_fld, e_a_method;
136*44704f69SBart Van Assche     uint64_t ull;
137*44704f69SBart Van Assche     char l_leadin[128];
138*44704f69SBart Van Assche     char b[256];
139*44704f69SBart Van Assche 
140*44704f69SBart Van Assche     if (0xff == lunp[0]) {
141*44704f69SBart Van Assche         printf("%sLogical unit _not_ specified\n", leadin);
142*44704f69SBart Van Assche         return;
143*44704f69SBart Van Assche     }
144*44704f69SBart Van Assche     admin_lu_cong = lu_cong;
145*44704f69SBart Van Assche     memset(l_leadin, 0, sizeof(l_leadin));
146*44704f69SBart Van Assche     for (k = 0; k < 4; ++k, lunp += 2) {
147*44704f69SBart Van Assche         next_level = false;
148*44704f69SBart Van Assche         strncpy(l_leadin, leadin, sizeof(l_leadin) - 3);
149*44704f69SBart Van Assche         if (k > 0) {
150*44704f69SBart Van Assche             if (lu_cong) {
151*44704f69SBart Van Assche                 admin_lu_cong = false;
152*44704f69SBart Van Assche                 if ((0 == lunp[0]) && (0 == lunp[1])) {
153*44704f69SBart Van Assche                     printf("%s>>>> Administrative LU\n", l_leadin);
154*44704f69SBart Van Assche                     if (do_hex || verbose)
155*44704f69SBart Van Assche                          printf("        since Subsidiary element is "
156*44704f69SBart Van Assche                                 "0x0000\n");
157*44704f69SBart Van Assche                     break;
158*44704f69SBart Van Assche                 } else
159*44704f69SBart Van Assche                     printf("%s>>Subsidiary element:\n", l_leadin);
160*44704f69SBart Van Assche             } else
161*44704f69SBart Van Assche                 printf("%s>>%s level addressing:\n", l_leadin, ((1 == k) ?
162*44704f69SBart Van Assche                          "Second" : ((2 == k) ? "Third" : "Fourth")));
163*44704f69SBart Van Assche             strcat(l_leadin, "  ");
164*44704f69SBart Van Assche         } else if (lu_cong) {
165*44704f69SBart Van Assche             printf("%s>>Administrative element:\n", l_leadin);
166*44704f69SBart Van Assche             strcat(l_leadin, "  ");
167*44704f69SBart Van Assche         }
168*44704f69SBart Van Assche         a_method = (lunp[0] >> 6) & 0x3;
169*44704f69SBart Van Assche         switch (a_method) {
170*44704f69SBart Van Assche         case 0:         /* peripheral device addressing method */
171*44704f69SBart Van Assche             if (lu_cong) {
172*44704f69SBart Van Assche                 snprintf(b, sizeof(b), "%sSimple lu addressing: ",
173*44704f69SBart Van Assche                          l_leadin);
174*44704f69SBart Van Assche                 x = 0x3fff & sg_get_unaligned_be16(lunp + 0);
175*44704f69SBart Van Assche                 if (do_hex)
176*44704f69SBart Van Assche                     printf("%s0x%04x\n", b, x);
177*44704f69SBart Van Assche                 else
178*44704f69SBart Van Assche                     printf("%s%d\n", b, x);
179*44704f69SBart Van Assche                 if (admin_lu_cong)
180*44704f69SBart Van Assche                     next_level = true;
181*44704f69SBart Van Assche             } else {
182*44704f69SBart Van Assche                 bus_id = lunp[0] & 0x3f;
183*44704f69SBart Van Assche                 snprintf(b, sizeof(b), "%sPeripheral device addressing: ",
184*44704f69SBart Van Assche                          l_leadin);
185*44704f69SBart Van Assche                 if ((0 == bus_id) && (0 == verbose)) {
186*44704f69SBart Van Assche                     if (do_hex)
187*44704f69SBart Van Assche                         printf("%slun=0x%02x\n", b, lunp[1]);
188*44704f69SBart Van Assche                     else
189*44704f69SBart Van Assche                         printf("%slun=%d\n", b, lunp[1]);
190*44704f69SBart Van Assche                 } else {
191*44704f69SBart Van Assche                     if (do_hex)
192*44704f69SBart Van Assche                         printf("%sbus_id=0x%02x, %s=0x%02x\n", b, bus_id,
193*44704f69SBart Van Assche                                (bus_id ? "target" : "lun"), lunp[1]);
194*44704f69SBart Van Assche                     else
195*44704f69SBart Van Assche                         printf("%sbus_id=%d, %s=%d\n", b, bus_id,
196*44704f69SBart Van Assche                                (bus_id ? "target" : "lun"), lunp[1]);
197*44704f69SBart Van Assche                 }
198*44704f69SBart Van Assche                 if (bus_id)
199*44704f69SBart Van Assche                     next_level = true;
200*44704f69SBart Van Assche             }
201*44704f69SBart Van Assche             break;
202*44704f69SBart Van Assche         case 1:         /* flat space addressing method */
203*44704f69SBart Van Assche             lun = 0x3fff & sg_get_unaligned_be16(lunp + 0);
204*44704f69SBart Van Assche             if (lu_cong) {
205*44704f69SBart Van Assche                 printf("%sSince LU_CONG=1, unexpected Flat space "
206*44704f69SBart Van Assche                        "addressing: lun=0x%04x\n", l_leadin, lun);
207*44704f69SBart Van Assche                 break;
208*44704f69SBart Van Assche             }
209*44704f69SBart Van Assche             if (do_hex)
210*44704f69SBart Van Assche                 printf("%sFlat space addressing: lun=0x%04x\n", l_leadin,
211*44704f69SBart Van Assche                        lun);
212*44704f69SBart Van Assche             else
213*44704f69SBart Van Assche                 printf("%sFlat space addressing: lun=%d\n", l_leadin, lun);
214*44704f69SBart Van Assche             break;
215*44704f69SBart Van Assche         case 2:         /* logical unit addressing method */
216*44704f69SBart Van Assche             target = (lunp[0] & 0x3f);
217*44704f69SBart Van Assche             bus_id = (lunp[1] >> 5) & 0x7;
218*44704f69SBart Van Assche             lun = lunp[1] & 0x1f;
219*44704f69SBart Van Assche             if (lu_cong) {
220*44704f69SBart Van Assche                 printf("%sSince LU_CONG=1, unexpected lu addressing: "
221*44704f69SBart Van Assche                        "bus_id=0x%x, target=0x%02x, lun=0x%02x\n", l_leadin,
222*44704f69SBart Van Assche                        bus_id, target, lun);
223*44704f69SBart Van Assche                 break;
224*44704f69SBart Van Assche             }
225*44704f69SBart Van Assche             if (do_hex)
226*44704f69SBart Van Assche                 printf("%sLogical unit addressing: bus_id=0x%x, "
227*44704f69SBart Van Assche                        "target=0x%02x, lun=0x%02x\n", l_leadin, bus_id,
228*44704f69SBart Van Assche                        target, lun);
229*44704f69SBart Van Assche             else
230*44704f69SBart Van Assche                 printf("%sLogical unit addressing: bus_id=%d, target=%d, "
231*44704f69SBart Van Assche                        "lun=%d\n", l_leadin, bus_id, target, lun);
232*44704f69SBart Van Assche             break;
233*44704f69SBart Van Assche         case 3:         /* extended logical unit + flat space addressing */
234*44704f69SBart Van Assche             len_fld = (lunp[0] & 0x30) >> 4;
235*44704f69SBart Van Assche             e_a_method = lunp[0] & 0xf;
236*44704f69SBart Van Assche             x = lunp[1];
237*44704f69SBart Van Assche             if ((0 == len_fld) && (1 == e_a_method)) {
238*44704f69SBart Van Assche                 snprintf(b, sizeof(b), "well known logical unit");
239*44704f69SBart Van Assche                 switch (x) {
240*44704f69SBart Van Assche                 case 1:
241*44704f69SBart Van Assche                     printf("%sREPORT LUNS %s\n", l_leadin, b);
242*44704f69SBart Van Assche                     break;
243*44704f69SBart Van Assche                 case 2:         /* obsolete in spc5r01 */
244*44704f69SBart Van Assche                     printf("%sACCESS CONTROLS %s\n", l_leadin, b);
245*44704f69SBart Van Assche                     break;
246*44704f69SBart Van Assche                 case 3:
247*44704f69SBart Van Assche                     printf("%sTARGET LOG PAGES %s\n", l_leadin, b);
248*44704f69SBart Van Assche                     break;
249*44704f69SBart Van Assche                 case 4:
250*44704f69SBart Van Assche                     printf("%sSECURITY PROTOCOL %s\n", l_leadin, b);
251*44704f69SBart Van Assche                     break;
252*44704f69SBart Van Assche                 case 5:
253*44704f69SBart Van Assche                     printf("%sMANAGEMENT PROTOCOL %s\n", l_leadin, b);
254*44704f69SBart Van Assche                     break;
255*44704f69SBart Van Assche                 case 6:
256*44704f69SBart Van Assche                     printf("%sTARGET COMMANDS %s\n", l_leadin, b);
257*44704f69SBart Van Assche                     break;
258*44704f69SBart Van Assche                 default:
259*44704f69SBart Van Assche                     if (do_hex)
260*44704f69SBart Van Assche                         printf("%s%s 0x%02x\n", l_leadin, b, x);
261*44704f69SBart Van Assche                     else
262*44704f69SBart Van Assche                         printf("%s%s %d\n", l_leadin, b, x);
263*44704f69SBart Van Assche                     break;
264*44704f69SBart Van Assche                 }
265*44704f69SBart Van Assche             } else if ((1 == len_fld) && (2 == e_a_method)) {
266*44704f69SBart Van Assche                 x = sg_get_unaligned_be24(lunp + 1);
267*44704f69SBart Van Assche                 if (do_hex)
268*44704f69SBart Van Assche                     printf("%sExtended flat space addressing: lun=0x%06x\n",
269*44704f69SBart Van Assche                            l_leadin, x);
270*44704f69SBart Van Assche                 else
271*44704f69SBart Van Assche                     printf("%sExtended flat space addressing: lun=%d\n",
272*44704f69SBart Van Assche                            l_leadin, x);
273*44704f69SBart Van Assche             } else if ((2 == len_fld) && (2 == e_a_method)) {
274*44704f69SBart Van Assche                 ull = sg_get_unaligned_be(5, lunp + 1);
275*44704f69SBart Van Assche                 if (do_hex)
276*44704f69SBart Van Assche                     printf("%sLong extended flat space addressing: "
277*44704f69SBart Van Assche                            "lun=0x%010" PRIx64 "\n", l_leadin, ull);
278*44704f69SBart Van Assche                 else
279*44704f69SBart Van Assche                     printf("%sLong extended flat space addressing: "
280*44704f69SBart Van Assche                            "lun=%" PRIu64 "\n", l_leadin, ull);
281*44704f69SBart Van Assche             } else if ((3 == len_fld) && (0xf == e_a_method))
282*44704f69SBart Van Assche                 printf("%sLogical unit _not_ specified addressing\n",
283*44704f69SBart Van Assche                        l_leadin);
284*44704f69SBart Van Assche             else {
285*44704f69SBart Van Assche                 if (len_fld < 2) {
286*44704f69SBart Van Assche                     if (1 == len_fld)
287*44704f69SBart Van Assche                         x = sg_get_unaligned_be24(lunp + 1);
288*44704f69SBart Van Assche                     if (do_hex)
289*44704f69SBart Van Assche                         printf("%sExtended logical unit addressing: "
290*44704f69SBart Van Assche                                "length=%d, e.a. method=%d, value=0x%06x\n",
291*44704f69SBart Van Assche                                l_leadin, len_fld, e_a_method, x);
292*44704f69SBart Van Assche                     else
293*44704f69SBart Van Assche                         printf("%sExtended logical unit addressing: "
294*44704f69SBart Van Assche                                "length=%d, e.a. method=%d, value=%d\n",
295*44704f69SBart Van Assche                                l_leadin, len_fld, e_a_method, x);
296*44704f69SBart Van Assche                 } else {
297*44704f69SBart Van Assche                     ull = sg_get_unaligned_be(((2 == len_fld) ? 5 : 7),
298*44704f69SBart Van Assche                                               lunp + 1);
299*44704f69SBart Van Assche                     if (do_hex) {
300*44704f69SBart Van Assche                         printf("%sExtended logical unit addressing: "
301*44704f69SBart Van Assche                                "length=%d, e. a. method=%d, ", l_leadin,
302*44704f69SBart Van Assche                                len_fld, e_a_method);
303*44704f69SBart Van Assche                         if (5 == len_fld)
304*44704f69SBart Van Assche                                 printf("value=0x%010" PRIx64 "\n", ull);
305*44704f69SBart Van Assche                         else
306*44704f69SBart Van Assche                                 printf("value=0x%014" PRIx64 "\n", ull);
307*44704f69SBart Van Assche                     } else
308*44704f69SBart Van Assche                         printf("%sExtended logical unit addressing: "
309*44704f69SBart Van Assche                                "length=%d, e. a. method=%d, value=%" PRIu64
310*44704f69SBart Van Assche                                "\n", l_leadin, len_fld, e_a_method, ull);
311*44704f69SBart Van Assche                 }
312*44704f69SBart Van Assche             }
313*44704f69SBart Van Assche             break;
314*44704f69SBart Van Assche         }
315*44704f69SBart Van Assche         if (next_level)
316*44704f69SBart Van Assche             continue;
317*44704f69SBart Van Assche         if ((2 == a_method) && (k < 3) && (lunp[2] || lunp[3]))
318*44704f69SBart Van Assche             printf("%s<<unexpected data at next level, continue>>\n",
319*44704f69SBart Van Assche                    l_leadin);
320*44704f69SBart Van Assche         break;
321*44704f69SBart Van Assche     }
322*44704f69SBart Van Assche }
323*44704f69SBart Van Assche 
324*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
325*44704f69SBart Van Assche static void
linux2t10_lun(uint64_t linux_lun,uint8_t t10_lun[])326*44704f69SBart Van Assche linux2t10_lun(uint64_t linux_lun, uint8_t t10_lun[])
327*44704f69SBart Van Assche {
328*44704f69SBart Van Assche     int k;
329*44704f69SBart Van Assche 
330*44704f69SBart Van Assche     for (k = 0; k < 8; k += 2, linux_lun >>= 16)
331*44704f69SBart Van Assche         sg_put_unaligned_be16((uint16_t)linux_lun, t10_lun + k);
332*44704f69SBart Van Assche }
333*44704f69SBart Van Assche 
334*44704f69SBart Van Assche static uint64_t
t10_2linux_lun(const uint8_t t10_lun[])335*44704f69SBart Van Assche t10_2linux_lun(const uint8_t t10_lun[])
336*44704f69SBart Van Assche {
337*44704f69SBart Van Assche     int k;
338*44704f69SBart Van Assche     const uint8_t * cp;
339*44704f69SBart Van Assche     uint64_t res;
340*44704f69SBart Van Assche 
341*44704f69SBart Van Assche     res = sg_get_unaligned_be16(t10_lun + 6);
342*44704f69SBart Van Assche     for (cp = t10_lun + 4, k = 0; k < 3; ++k, cp -= 2)
343*44704f69SBart Van Assche         res = (res << 16) + sg_get_unaligned_be16(cp);
344*44704f69SBart Van Assche     return res;
345*44704f69SBart Van Assche }
346*44704f69SBart Van Assche #endif  /* SG_LIB_LINUX */
347*44704f69SBart Van Assche 
348*44704f69SBart Van Assche 
349*44704f69SBart Van Assche static void
dStrRaw(const char * str,int len)350*44704f69SBart Van Assche dStrRaw(const char * str, int len)
351*44704f69SBart Van Assche {
352*44704f69SBart Van Assche     int k;
353*44704f69SBart Van Assche 
354*44704f69SBart Van Assche     for (k = 0; k < len; ++k)
355*44704f69SBart Van Assche         printf("%c", str[k]);
356*44704f69SBart Van Assche }
357*44704f69SBart Van Assche 
358*44704f69SBart Van Assche int
main(int argc,char * argv[])359*44704f69SBart Van Assche main(int argc, char * argv[])
360*44704f69SBart Van Assche {
361*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
362*44704f69SBart Van Assche     bool do_linux = false;
363*44704f69SBart Van Assche #endif
364*44704f69SBart Van Assche     bool do_quiet = false;
365*44704f69SBart Van Assche     bool do_raw = false;
366*44704f69SBart Van Assche     bool lu_cong_arg_given = false;
367*44704f69SBart Van Assche     bool o_readonly = false;
368*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
369*44704f69SBart Van Assche     bool test_linux_in = false;
370*44704f69SBart Van Assche     bool test_linux_out = false;
371*44704f69SBart Van Assche #endif
372*44704f69SBart Van Assche     bool trunc;
373*44704f69SBart Van Assche     bool verbose_given = false;
374*44704f69SBart Van Assche     bool version_given = false;
375*44704f69SBart Van Assche     int sg_fd, k, m, off, res, c, list_len, len_cap, luns;
376*44704f69SBart Van Assche     int decode_arg = 0;
377*44704f69SBart Van Assche     int do_hex = 0;
378*44704f69SBart Van Assche     int lu_cong_arg = 0;
379*44704f69SBart Van Assche     int maxlen = 0;
380*44704f69SBart Van Assche     int ret = 0;
381*44704f69SBart Van Assche     int select_rep = 0;
382*44704f69SBart Van Assche     int verbose = 0;
383*44704f69SBart Van Assche     unsigned int h;
384*44704f69SBart Van Assche     const char * test_arg = NULL;
385*44704f69SBart Van Assche     const char * device_name = NULL;
386*44704f69SBart Van Assche     const char * cp;
387*44704f69SBart Van Assche     uint8_t * reportLunsBuff = NULL;
388*44704f69SBart Van Assche     uint8_t * free_reportLunsBuff = NULL;
389*44704f69SBart Van Assche     uint8_t lun_arr[8];
390*44704f69SBart Van Assche     struct sg_simple_inquiry_resp sir;
391*44704f69SBart Van Assche 
392*44704f69SBart Van Assche     while (1) {
393*44704f69SBart Van Assche         int option_index = 0;
394*44704f69SBart Van Assche 
395*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
396*44704f69SBart Van Assche         c = getopt_long(argc, argv, "dhHlLm:qrRs:t:vV", long_options,
397*44704f69SBart Van Assche                         &option_index);
398*44704f69SBart Van Assche #else
399*44704f69SBart Van Assche         c = getopt_long(argc, argv, "dhHLm:qrRs:t:vV", long_options,
400*44704f69SBart Van Assche                         &option_index);
401*44704f69SBart Van Assche #endif
402*44704f69SBart Van Assche         if (c == -1)
403*44704f69SBart Van Assche             break;
404*44704f69SBart Van Assche 
405*44704f69SBart Van Assche         switch (c) {
406*44704f69SBart Van Assche         case 'd':
407*44704f69SBart Van Assche             ++decode_arg;
408*44704f69SBart Van Assche             break;
409*44704f69SBart Van Assche         case 'h':
410*44704f69SBart Van Assche         case '?':
411*44704f69SBart Van Assche             usage();
412*44704f69SBart Van Assche             return 0;
413*44704f69SBart Van Assche         case 'H':
414*44704f69SBart Van Assche             ++do_hex;
415*44704f69SBart Van Assche             break;
416*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
417*44704f69SBart Van Assche         case 'l':
418*44704f69SBart Van Assche             do_linux = false;
419*44704f69SBart Van Assche             break;
420*44704f69SBart Van Assche #endif
421*44704f69SBart Van Assche         case 'L':
422*44704f69SBart Van Assche             ++lu_cong_arg;
423*44704f69SBart Van Assche             lu_cong_arg_given = true;
424*44704f69SBart Van Assche             break;
425*44704f69SBart Van Assche         case 'm':
426*44704f69SBart Van Assche             maxlen = sg_get_num(optarg);
427*44704f69SBart Van Assche             if ((maxlen < 0) || (maxlen > MAX_RLUNS_BUFF_LEN)) {
428*44704f69SBart Van Assche                 pr2serr("argument to '--maxlen' should be %d or less\n",
429*44704f69SBart Van Assche                         MAX_RLUNS_BUFF_LEN);
430*44704f69SBart Van Assche                 return SG_LIB_SYNTAX_ERROR;
431*44704f69SBart Van Assche             } else if (maxlen < 4) {
432*44704f69SBart Van Assche                 pr2serr("Warning: setting '--maxlen' to 4\n");
433*44704f69SBart Van Assche                 maxlen = 4;
434*44704f69SBart Van Assche             }
435*44704f69SBart Van Assche             break;
436*44704f69SBart Van Assche         case 'q':
437*44704f69SBart Van Assche             do_quiet = true;
438*44704f69SBart Van Assche             break;
439*44704f69SBart Van Assche         case 'r':
440*44704f69SBart Van Assche             do_raw = true;
441*44704f69SBart Van Assche             break;
442*44704f69SBart Van Assche         case 'R':
443*44704f69SBart Van Assche             o_readonly = true;
444*44704f69SBart Van Assche             break;
445*44704f69SBart Van Assche         case 's':
446*44704f69SBart Van Assche            select_rep = sg_get_num(optarg);
447*44704f69SBart Van Assche            if ((select_rep < 0) || (select_rep > 255)) {
448*44704f69SBart Van Assche                 pr2serr("bad argument to '--select', expect 0 to 255\n");
449*44704f69SBart Van Assche                 return SG_LIB_SYNTAX_ERROR;
450*44704f69SBart Van Assche             }
451*44704f69SBart Van Assche             break;
452*44704f69SBart Van Assche         case 't':
453*44704f69SBart Van Assche             test_arg = optarg;
454*44704f69SBart Van Assche             break;
455*44704f69SBart Van Assche         case 'v':
456*44704f69SBart Van Assche             verbose_given = true;
457*44704f69SBart Van Assche             ++verbose;
458*44704f69SBart Van Assche             break;
459*44704f69SBart Van Assche         case 'V':
460*44704f69SBart Van Assche             version_given = true;
461*44704f69SBart Van Assche             break;
462*44704f69SBart Van Assche         default:
463*44704f69SBart Van Assche             pr2serr("unrecognised option code 0x%x ??\n", c);
464*44704f69SBart Van Assche             usage();
465*44704f69SBart Van Assche             return SG_LIB_SYNTAX_ERROR;
466*44704f69SBart Van Assche         }
467*44704f69SBart Van Assche     }
468*44704f69SBart Van Assche     if (optind < argc) {
469*44704f69SBart Van Assche         if (NULL == device_name) {
470*44704f69SBart Van Assche             device_name = argv[optind];
471*44704f69SBart Van Assche             ++optind;
472*44704f69SBart Van Assche         }
473*44704f69SBart Van Assche         if (optind < argc) {
474*44704f69SBart Van Assche             for (; optind < argc; ++optind)
475*44704f69SBart Van Assche                 pr2serr("Unexpected extra argument: %s\n", argv[optind]);
476*44704f69SBart Van Assche             usage();
477*44704f69SBart Van Assche             return SG_LIB_SYNTAX_ERROR;
478*44704f69SBart Van Assche         }
479*44704f69SBart Van Assche     }
480*44704f69SBart Van Assche #ifdef DEBUG
481*44704f69SBart Van Assche     pr2serr("In DEBUG mode, ");
482*44704f69SBart Van Assche     if (verbose_given && version_given) {
483*44704f69SBart Van Assche         pr2serr("but override: '-vV' given, zero verbose and continue\n");
484*44704f69SBart Van Assche         verbose_given = false;
485*44704f69SBart Van Assche         version_given = false;
486*44704f69SBart Van Assche         verbose = 0;
487*44704f69SBart Van Assche     } else if (! verbose_given) {
488*44704f69SBart Van Assche         pr2serr("set '-vv'\n");
489*44704f69SBart Van Assche         verbose = 2;
490*44704f69SBart Van Assche     } else
491*44704f69SBart Van Assche         pr2serr("keep verbose=%d\n", verbose);
492*44704f69SBart Van Assche #else
493*44704f69SBart Van Assche     if (verbose_given && version_given)
494*44704f69SBart Van Assche         pr2serr("Not in DEBUG mode, so '-vV' has no special action\n");
495*44704f69SBart Van Assche #endif
496*44704f69SBart Van Assche     if (version_given) {
497*44704f69SBart Van Assche         pr2serr("version: %s\n", version_str);
498*44704f69SBart Van Assche         return 0;
499*44704f69SBart Van Assche     }
500*44704f69SBart Van Assche 
501*44704f69SBart Van Assche     if (test_arg) {
502*44704f69SBart Van Assche         memset(lun_arr, 0, sizeof(lun_arr));
503*44704f69SBart Van Assche         cp = test_arg;
504*44704f69SBart Van Assche         /* check for leading 'L' */
505*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
506*44704f69SBart Van Assche         if ('L' == toupper(cp[0])) {
507*44704f69SBart Van Assche             uint64_t ull;
508*44704f69SBart Van Assche 
509*44704f69SBart Van Assche             if (('0' == cp[1]) && ('X' == toupper((uint8_t)cp[2])))
510*44704f69SBart Van Assche                 k = sscanf(cp + 3, " %" SCNx64, &ull);
511*44704f69SBart Van Assche             else
512*44704f69SBart Van Assche                 k = sscanf(cp + 1, " %" SCNu64, &ull);
513*44704f69SBart Van Assche             if (1 != k) {
514*44704f69SBart Van Assche                 pr2serr("Unable to read Linux style LUN integer given to "
515*44704f69SBart Van Assche                         "--test=\n");
516*44704f69SBart Van Assche                 return SG_LIB_SYNTAX_ERROR;
517*44704f69SBart Van Assche             }
518*44704f69SBart Van Assche             linux2t10_lun(ull, lun_arr);
519*44704f69SBart Van Assche             test_linux_in = true;
520*44704f69SBart Van Assche         } else
521*44704f69SBart Van Assche #endif
522*44704f69SBart Van Assche         {
523*44704f69SBart Van Assche             /* Check if trailing 'L' */
524*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
525*44704f69SBart Van Assche             m = strlen(cp);    /* must be at least 1 char in test_arg */
526*44704f69SBart Van Assche             if ('L' == toupper(cp[m - 1]))
527*44704f69SBart Van Assche                 test_linux_out = true;
528*44704f69SBart Van Assche #endif
529*44704f69SBart Van Assche             if (('0' == cp[0]) && ('X' == toupper(cp[1])))
530*44704f69SBart Van Assche                 cp += 2;
531*44704f69SBart Van Assche             if (strchr(cp, ' ') || strchr(cp, '\t') || strchr(cp, '-')) {
532*44704f69SBart Van Assche                 for (k = 0; k < 8; ++k, cp += 2) {
533*44704f69SBart Van Assche                     c = *cp;
534*44704f69SBart Van Assche                     if ('\0' == c)
535*44704f69SBart Van Assche                         break;
536*44704f69SBart Van Assche                     else if (! isxdigit(c))
537*44704f69SBart Van Assche                         ++cp;
538*44704f69SBart Van Assche                     if (1 != sscanf(cp, "%2x", &h))
539*44704f69SBart Van Assche                         break;
540*44704f69SBart Van Assche                     lun_arr[k] = h & 0xff;
541*44704f69SBart Van Assche                 }
542*44704f69SBart Van Assche             } else {
543*44704f69SBart Van Assche                 for (k = 0; k < 8; ++k, cp += 2) {
544*44704f69SBart Van Assche                 if (1 != sscanf(cp, "%2x", &h))
545*44704f69SBart Van Assche                         break;
546*44704f69SBart Van Assche                     lun_arr[k] = h & 0xff;
547*44704f69SBart Van Assche                 }
548*44704f69SBart Van Assche             }
549*44704f69SBart Van Assche             if (0 == k) {
550*44704f69SBart Van Assche                 pr2serr("expected a hex number, optionally prefixed by "
551*44704f69SBart Van Assche                         "'0x'\n");
552*44704f69SBart Van Assche                 return SG_LIB_SYNTAX_ERROR;
553*44704f69SBart Van Assche             }
554*44704f69SBart Van Assche         }
555*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
556*44704f69SBart Van Assche         if (verbose || test_linux_in || decode_arg)
557*44704f69SBart Van Assche #else
558*44704f69SBart Van Assche         if (verbose || decode_arg)
559*44704f69SBart Van Assche #endif
560*44704f69SBart Van Assche         {
561*44704f69SBart Van Assche             if (decode_arg > 1) {
562*44704f69SBart Van Assche                 printf("64 bit LUN in T10 (hex, dashed) format: ");
563*44704f69SBart Van Assche                 for (k = 0; k < 8; k += 2)
564*44704f69SBart Van Assche                     printf("%c%02x%02x", (k ? '-' : ' '), lun_arr[k],
565*44704f69SBart Van Assche                            lun_arr[k + 1]);
566*44704f69SBart Van Assche             } else {
567*44704f69SBart Van Assche                 printf("64 bit LUN in T10 preferred (hex) format: ");
568*44704f69SBart Van Assche                 for (k = 0; k < 8; ++k)
569*44704f69SBart Van Assche                     printf(" %02x", lun_arr[k]);
570*44704f69SBart Van Assche             }
571*44704f69SBart Van Assche             printf("\n");
572*44704f69SBart Van Assche         }
573*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
574*44704f69SBart Van Assche         if (test_linux_out) {
575*44704f69SBart Van Assche             if (do_hex > 1)
576*44704f69SBart Van Assche                 printf("Linux 'word flipped' integer LUN representation: "
577*44704f69SBart Van Assche                        "0x%016" PRIx64 "\n", t10_2linux_lun(lun_arr));
578*44704f69SBart Van Assche             else if (do_hex)
579*44704f69SBart Van Assche                 printf("Linux 'word flipped' integer LUN representation: 0x%"
580*44704f69SBart Van Assche                        PRIx64 "\n", t10_2linux_lun(lun_arr));
581*44704f69SBart Van Assche             else
582*44704f69SBart Van Assche                 printf("Linux 'word flipped' integer LUN representation: %"
583*44704f69SBart Van Assche                        PRIu64 "\n", t10_2linux_lun(lun_arr));
584*44704f69SBart Van Assche         }
585*44704f69SBart Van Assche #endif
586*44704f69SBart Van Assche         printf("Decoded LUN:\n");
587*44704f69SBart Van Assche         decode_lun("  ", lun_arr, (lu_cong_arg % 2), do_hex, verbose);
588*44704f69SBart Van Assche         return 0;
589*44704f69SBart Van Assche     }
590*44704f69SBart Van Assche     if (NULL == device_name) {
591*44704f69SBart Van Assche         pr2serr("missing device name!\n");
592*44704f69SBart Van Assche         usage();
593*44704f69SBart Van Assche         return SG_LIB_SYNTAX_ERROR;
594*44704f69SBart Van Assche     }
595*44704f69SBart Van Assche 
596*44704f69SBart Van Assche     if (do_raw) {
597*44704f69SBart Van Assche         if (sg_set_binary_mode(STDOUT_FILENO) < 0) {
598*44704f69SBart Van Assche             perror("sg_set_binary_mode");
599*44704f69SBart Van Assche             return SG_LIB_FILE_ERROR;
600*44704f69SBart Van Assche         }
601*44704f69SBart Van Assche     }
602*44704f69SBart Van Assche 
603*44704f69SBart Van Assche     sg_fd = sg_cmds_open_device(device_name, o_readonly, verbose);
604*44704f69SBart Van Assche     if (sg_fd < 0) {
605*44704f69SBart Van Assche         int err = -sg_fd;
606*44704f69SBart Van Assche 
607*44704f69SBart Van Assche         pr2serr("open error: %s: %s\n", device_name, safe_strerror(err));
608*44704f69SBart Van Assche         if ((! o_readonly) && ((err == EACCES) || (err == EROFS)))
609*44704f69SBart Van Assche             pr2serr("Perhaps try again with --readonly option or with root "
610*44704f69SBart Van Assche                     "permissions\n");
611*44704f69SBart Van Assche         return sg_convert_errno(-sg_fd);
612*44704f69SBart Van Assche     }
613*44704f69SBart Van Assche     if (decode_arg && (! lu_cong_arg_given)) {
614*44704f69SBart Van Assche         if (verbose > 1)
615*44704f69SBart Van Assche             pr2serr("in order to decode LUN and since --lu_cong not given, "
616*44704f69SBart Van Assche                     "do standard\nINQUIRY to find LU_CONG bit\n");
617*44704f69SBart Van Assche         /* check if LU_CONG set in standard INQUIRY response */
618*44704f69SBart Van Assche         res = sg_simple_inquiry(sg_fd, &sir, false, verbose);
619*44704f69SBart Van Assche         ret = res;
620*44704f69SBart Van Assche         if (res) {
621*44704f69SBart Van Assche             pr2serr("fetching standard INQUIRY response failed\n");
622*44704f69SBart Van Assche             goto the_end;
623*44704f69SBart Van Assche         }
624*44704f69SBart Van Assche         lu_cong_arg = !!(0x40 & sir.byte_1);
625*44704f69SBart Van Assche         if (verbose && lu_cong_arg)
626*44704f69SBart Van Assche             pr2serr("LU_CONG bit set in standard INQUIRY response\n");
627*44704f69SBart Van Assche     }
628*44704f69SBart Van Assche 
629*44704f69SBart Van Assche     if (0 == maxlen)
630*44704f69SBart Van Assche         maxlen = DEF_RLUNS_BUFF_LEN;
631*44704f69SBart Van Assche     reportLunsBuff = (uint8_t *)sg_memalign(maxlen, 0, &free_reportLunsBuff,
632*44704f69SBart Van Assche                                             verbose > 3);
633*44704f69SBart Van Assche     if (NULL == reportLunsBuff) {
634*44704f69SBart Van Assche         pr2serr("unable to sg_memalign %d bytes\n", maxlen);
635*44704f69SBart Van Assche         return sg_convert_errno(ENOMEM);
636*44704f69SBart Van Assche     }
637*44704f69SBart Van Assche     trunc = false;
638*44704f69SBart Van Assche 
639*44704f69SBart Van Assche     res = sg_ll_report_luns(sg_fd, select_rep, reportLunsBuff, maxlen, true,
640*44704f69SBart Van Assche                             verbose);
641*44704f69SBart Van Assche     ret = res;
642*44704f69SBart Van Assche     if (0 == res) {
643*44704f69SBart Van Assche         list_len = sg_get_unaligned_be32(reportLunsBuff + 0);
644*44704f69SBart Van Assche         len_cap = list_len + 8;
645*44704f69SBart Van Assche         if (len_cap > maxlen)
646*44704f69SBart Van Assche             len_cap = maxlen;
647*44704f69SBart Van Assche         if (do_raw) {
648*44704f69SBart Van Assche             dStrRaw((const char *)reportLunsBuff, len_cap);
649*44704f69SBart Van Assche             goto the_end;
650*44704f69SBart Van Assche         }
651*44704f69SBart Van Assche         if (1 == do_hex) {
652*44704f69SBart Van Assche             hex2stdout(reportLunsBuff, len_cap, 1);
653*44704f69SBart Van Assche             goto the_end;
654*44704f69SBart Van Assche         }
655*44704f69SBart Van Assche         luns = (list_len / 8);
656*44704f69SBart Van Assche         if (! do_quiet)
657*44704f69SBart Van Assche             printf("Lun list length = %d which imples %d lun entr%s\n",
658*44704f69SBart Van Assche                    list_len, luns, ((1 == luns) ? "y" : "ies"));
659*44704f69SBart Van Assche         if ((list_len + 8) > maxlen) {
660*44704f69SBart Van Assche             luns = ((maxlen - 8) / 8);
661*44704f69SBart Van Assche             trunc = true;
662*44704f69SBart Van Assche             pr2serr("  <<too many luns for internal buffer, will show %d "
663*44704f69SBart Van Assche                     "lun%s>>\n", luns, ((1 == luns) ? "" : "s"));
664*44704f69SBart Van Assche         }
665*44704f69SBart Van Assche         if (verbose > 1) {
666*44704f69SBart Van Assche             pr2serr("\nOutput response in hex\n");
667*44704f69SBart Van Assche             hex2stderr(reportLunsBuff, (trunc ? maxlen : list_len + 8), 1);
668*44704f69SBart Van Assche         }
669*44704f69SBart Van Assche         for (k = 0, off = 8; k < luns; ++k, off += 8) {
670*44704f69SBart Van Assche             if (! do_quiet) {
671*44704f69SBart Van Assche                 if (0 == k)
672*44704f69SBart Van Assche                     printf("Report luns [select_report=0x%x]:\n", select_rep);
673*44704f69SBart Van Assche                 printf("    ");
674*44704f69SBart Van Assche             }
675*44704f69SBart Van Assche             for (m = 0; m < 8; ++m)
676*44704f69SBart Van Assche                 printf("%02x", reportLunsBuff[off + m]);
677*44704f69SBart Van Assche #ifdef SG_LIB_LINUX
678*44704f69SBart Van Assche             if (do_linux) {
679*44704f69SBart Van Assche                 uint64_t lin_lun;
680*44704f69SBart Van Assche 
681*44704f69SBart Van Assche                 lin_lun = t10_2linux_lun(reportLunsBuff + off);
682*44704f69SBart Van Assche                 if (do_hex > 1)
683*44704f69SBart Van Assche                     printf("    [0x%" PRIx64 "]", lin_lun);
684*44704f69SBart Van Assche                 else
685*44704f69SBart Van Assche                     printf("    [%" PRIu64 "]", lin_lun);
686*44704f69SBart Van Assche             }
687*44704f69SBart Van Assche #endif
688*44704f69SBart Van Assche             printf("\n");
689*44704f69SBart Van Assche             if (decode_arg)
690*44704f69SBart Van Assche                 decode_lun("      ", reportLunsBuff + off,
691*44704f69SBart Van Assche                            (bool)(lu_cong_arg % 2), do_hex, verbose);
692*44704f69SBart Van Assche         }
693*44704f69SBart Van Assche     } else if (SG_LIB_CAT_INVALID_OP == res)
694*44704f69SBart Van Assche         pr2serr("Report Luns command not supported (support mandatory in "
695*44704f69SBart Van Assche                 "SPC-3)\n");
696*44704f69SBart Van Assche     else if (SG_LIB_CAT_ABORTED_COMMAND == res)
697*44704f69SBart Van Assche         pr2serr("Report Luns, aborted command\n");
698*44704f69SBart Van Assche     else if (SG_LIB_CAT_ILLEGAL_REQ == res)
699*44704f69SBart Van Assche         pr2serr("Report Luns command has bad field in cdb\n");
700*44704f69SBart Van Assche     else {
701*44704f69SBart Van Assche         char b[80];
702*44704f69SBart Van Assche 
703*44704f69SBart Van Assche         sg_get_category_sense_str(res, sizeof(b), b, verbose);
704*44704f69SBart Van Assche         pr2serr("Report Luns command: %s\n", b);
705*44704f69SBart Van Assche     }
706*44704f69SBart Van Assche 
707*44704f69SBart Van Assche the_end:
708*44704f69SBart Van Assche     if (free_reportLunsBuff)
709*44704f69SBart Van Assche         free(free_reportLunsBuff);
710*44704f69SBart Van Assche     res = sg_cmds_close_device(sg_fd);
711*44704f69SBart Van Assche     if (res < 0) {
712*44704f69SBart Van Assche         pr2serr("close error: %s\n", safe_strerror(-res));
713*44704f69SBart Van Assche         if (0 == ret)
714*44704f69SBart Van Assche             return sg_convert_errno(-res);
715*44704f69SBart Van Assche     }
716*44704f69SBart Van Assche     if (0 == verbose) {
717*44704f69SBart Van Assche         if (! sg_if_can2stderr("sg_luns failed: ", ret))
718*44704f69SBart Van Assche             pr2serr("Some error occurred, try again with '-v' or '-vv' for "
719*44704f69SBart Van Assche                     "more information\n");
720*44704f69SBart Van Assche     }
721*44704f69SBart Van Assche     return (ret >= 0) ? ret : SG_LIB_CAT_OTHER;
722*44704f69SBart Van Assche }
723