xref: /freebsd/usr.sbin/ctladm/ctladm.c (revision 76afb20c58adb296f09857aed214b91464242264)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3  *
4  * Copyright (c) 2003, 2004 Silicon Graphics International Corp.
5  * Copyright (c) 1997-2007 Kenneth D. Merry
6  * Copyright (c) 2012 The FreeBSD Foundation
7  * Copyright (c) 2018 Marcelo Araujo <araujo@FreeBSD.org>
8  * All rights reserved.
9  *
10  * Portions of this software were developed by Edward Tomasz Napierala
11  * under sponsorship from the FreeBSD Foundation.
12  *
13  * Redistribution and use in source and binary forms, with or without
14  * modification, are permitted provided that the following conditions
15  * are met:
16  * 1. Redistributions of source code must retain the above copyright
17  *    notice, this list of conditions, and the following disclaimer,
18  *    without modification.
19  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
20  *    substantially similar to the "NO WARRANTY" disclaimer below
21  *    ("Disclaimer") and any redistribution must be conditioned upon
22  *    including a substantially similar Disclaimer requirement for further
23  *    binary redistribution.
24  *
25  * NO WARRANTY
26  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
27  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
28  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
29  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
30  * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
31  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
32  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
33  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
34  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
35  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
36  * POSSIBILITY OF SUCH DAMAGES.
37  *
38  * $Id: //depot/users/kenm/FreeBSD-test2/usr.sbin/ctladm/ctladm.c#4 $
39  */
40 /*
41  * CAM Target Layer exercise program.
42  *
43  * Author: Ken Merry <ken@FreeBSD.org>
44  */
45 
46 #include <sys/cdefs.h>
47 __FBSDID("$FreeBSD$");
48 
49 #include <sys/param.h>
50 #include <sys/callout.h>
51 #include <sys/ioctl.h>
52 #include <sys/linker.h>
53 #include <sys/module.h>
54 #include <sys/queue.h>
55 #include <sys/sbuf.h>
56 #include <sys/nv.h>
57 #include <sys/stat.h>
58 #include <bsdxml.h>
59 #include <ctype.h>
60 #include <err.h>
61 #include <errno.h>
62 #include <fcntl.h>
63 #include <getopt.h>
64 #include <stdlib.h>
65 #include <stdint.h>
66 #include <stdio.h>
67 #include <string.h>
68 #include <unistd.h>
69 #include <cam/scsi/scsi_all.h>
70 #include <cam/scsi/scsi_message.h>
71 #include <cam/ctl/ctl.h>
72 #include <cam/ctl/ctl_io.h>
73 #include <cam/ctl/ctl_backend.h>
74 #include <cam/ctl/ctl_ioctl.h>
75 #include <cam/ctl/ctl_util.h>
76 #include <cam/ctl/ctl_scsi_all.h>
77 #include <camlib.h>
78 #include <libutil.h>
79 #include "ctladm.h"
80 
81 #ifdef min
82 #undef min
83 #endif
84 #define min(x,y) (x < y) ? x : y
85 
86 typedef enum {
87 	CTLADM_CMD_TUR,
88 	CTLADM_CMD_INQUIRY,
89 	CTLADM_CMD_REQ_SENSE,
90 	CTLADM_CMD_ARRAYLIST,
91 	CTLADM_CMD_REPORT_LUNS,
92 	CTLADM_CMD_HELP,
93 	CTLADM_CMD_DEVLIST,
94 	CTLADM_CMD_ADDDEV,
95 	CTLADM_CMD_RM,
96 	CTLADM_CMD_CREATE,
97 	CTLADM_CMD_READ,
98 	CTLADM_CMD_WRITE,
99 	CTLADM_CMD_PORT,
100 	CTLADM_CMD_PORTLIST,
101 	CTLADM_CMD_READCAPACITY,
102 	CTLADM_CMD_MODESENSE,
103 	CTLADM_CMD_DUMPOOA,
104 	CTLADM_CMD_DUMPSTRUCTS,
105 	CTLADM_CMD_START,
106 	CTLADM_CMD_STOP,
107 	CTLADM_CMD_SYNC_CACHE,
108 	CTLADM_CMD_LUNLIST,
109 	CTLADM_CMD_DELAY,
110 	CTLADM_CMD_ERR_INJECT,
111 	CTLADM_CMD_PRES_IN,
112 	CTLADM_CMD_PRES_OUT,
113 	CTLADM_CMD_INQ_VPD_DEVID,
114 	CTLADM_CMD_RTPG,
115 	CTLADM_CMD_MODIFY,
116 	CTLADM_CMD_ISLIST,
117 	CTLADM_CMD_ISLOGOUT,
118 	CTLADM_CMD_ISTERMINATE,
119 	CTLADM_CMD_LUNMAP
120 } ctladm_cmdfunction;
121 
122 typedef enum {
123 	CTLADM_ARG_NONE		= 0x0000000,
124 	CTLADM_ARG_AUTOSENSE	= 0x0000001,
125 	CTLADM_ARG_DEVICE	= 0x0000002,
126 	CTLADM_ARG_ARRAYSIZE	= 0x0000004,
127 	CTLADM_ARG_BACKEND	= 0x0000008,
128 	CTLADM_ARG_CDBSIZE	= 0x0000010,
129 	CTLADM_ARG_DATALEN	= 0x0000020,
130 	CTLADM_ARG_FILENAME	= 0x0000040,
131 	CTLADM_ARG_LBA		= 0x0000080,
132 	CTLADM_ARG_PC		= 0x0000100,
133 	CTLADM_ARG_PAGE_CODE	= 0x0000200,
134 	CTLADM_ARG_PAGE_LIST	= 0x0000400,
135 	CTLADM_ARG_SUBPAGE	= 0x0000800,
136 	CTLADM_ARG_PAGELIST	= 0x0001000,
137 	CTLADM_ARG_DBD		= 0x0002000,
138 	CTLADM_ARG_TARG_LUN	= 0x0004000,
139 	CTLADM_ARG_BLOCKSIZE	= 0x0008000,
140 	CTLADM_ARG_IMMED	= 0x0010000,
141 	CTLADM_ARG_RELADR	= 0x0020000,
142 	CTLADM_ARG_RETRIES	= 0x0040000,
143 	CTLADM_ARG_ONOFFLINE	= 0x0080000,
144 	CTLADM_ARG_ONESHOT	= 0x0100000,
145 	CTLADM_ARG_TIMEOUT	= 0x0200000,
146 	CTLADM_ARG_INITIATOR	= 0x0400000,
147 	CTLADM_ARG_NOCOPY	= 0x0800000,
148 	CTLADM_ARG_NEED_TL	= 0x1000000
149 } ctladm_cmdargs;
150 
151 struct ctladm_opts {
152 	const char	*optname;
153 	uint32_t	cmdnum;
154 	ctladm_cmdargs	argnum;
155 	const char	*subopt;
156 };
157 
158 typedef enum {
159 	CC_OR_NOT_FOUND,
160 	CC_OR_AMBIGUOUS,
161 	CC_OR_FOUND
162 } ctladm_optret;
163 
164 static const char rw_opts[] = "Nb:c:d:f:l:";
165 static const char startstop_opts[] = "i";
166 
167 static struct ctladm_opts option_table[] = {
168 	{"adddev", CTLADM_CMD_ADDDEV, CTLADM_ARG_NONE, NULL},
169 	{"create", CTLADM_CMD_CREATE, CTLADM_ARG_NONE, "b:B:d:l:o:s:S:t:"},
170 	{"delay", CTLADM_CMD_DELAY, CTLADM_ARG_NEED_TL, "T:l:t:"},
171 	{"devid", CTLADM_CMD_INQ_VPD_DEVID, CTLADM_ARG_NEED_TL, NULL},
172 	{"devlist", CTLADM_CMD_DEVLIST, CTLADM_ARG_NONE, "b:vx"},
173 	{"dumpooa", CTLADM_CMD_DUMPOOA, CTLADM_ARG_NONE, NULL},
174 	{"dumpstructs", CTLADM_CMD_DUMPSTRUCTS, CTLADM_ARG_NONE, NULL},
175 	{"help", CTLADM_CMD_HELP, CTLADM_ARG_NONE, NULL},
176 	{"inject", CTLADM_CMD_ERR_INJECT, CTLADM_ARG_NEED_TL, "cd:i:p:r:s:"},
177 	{"inquiry", CTLADM_CMD_INQUIRY, CTLADM_ARG_NEED_TL, NULL},
178 	{"islist", CTLADM_CMD_ISLIST, CTLADM_ARG_NONE, "vx"},
179 	{"islogout", CTLADM_CMD_ISLOGOUT, CTLADM_ARG_NONE, "ac:i:p:"},
180 	{"isterminate", CTLADM_CMD_ISTERMINATE, CTLADM_ARG_NONE, "ac:i:p:"},
181 	{"lunlist", CTLADM_CMD_LUNLIST, CTLADM_ARG_NONE, NULL},
182 	{"lunmap", CTLADM_CMD_LUNMAP, CTLADM_ARG_NONE, "p:l:L:"},
183 	{"modesense", CTLADM_CMD_MODESENSE, CTLADM_ARG_NEED_TL, "P:S:dlm:c:"},
184 	{"modify", CTLADM_CMD_MODIFY, CTLADM_ARG_NONE, "b:l:o:s:"},
185 	{"port", CTLADM_CMD_PORT, CTLADM_ARG_NONE, "lo:O:d:crp:qt:w:W:x"},
186 	{"portlist", CTLADM_CMD_PORTLIST, CTLADM_ARG_NONE, "f:ilp:qvx"},
187 	{"prin", CTLADM_CMD_PRES_IN, CTLADM_ARG_NEED_TL, "a:"},
188 	{"prout", CTLADM_CMD_PRES_OUT, CTLADM_ARG_NEED_TL, "a:k:r:s:"},
189 	{"read", CTLADM_CMD_READ, CTLADM_ARG_NEED_TL, rw_opts},
190 	{"readcapacity", CTLADM_CMD_READCAPACITY, CTLADM_ARG_NEED_TL, "c:"},
191 	{"remove", CTLADM_CMD_RM, CTLADM_ARG_NONE, "b:l:o:"},
192 	{"reportluns", CTLADM_CMD_REPORT_LUNS, CTLADM_ARG_NEED_TL, NULL},
193 	{"reqsense", CTLADM_CMD_REQ_SENSE, CTLADM_ARG_NEED_TL, NULL},
194 	{"rtpg", CTLADM_CMD_RTPG, CTLADM_ARG_NEED_TL, NULL},
195 	{"start", CTLADM_CMD_START, CTLADM_ARG_NEED_TL, startstop_opts},
196 	{"stop", CTLADM_CMD_STOP, CTLADM_ARG_NEED_TL, startstop_opts},
197 	{"synccache", CTLADM_CMD_SYNC_CACHE, CTLADM_ARG_NEED_TL, "b:c:il:r"},
198 	{"tur", CTLADM_CMD_TUR, CTLADM_ARG_NEED_TL, NULL},
199 	{"write", CTLADM_CMD_WRITE, CTLADM_ARG_NEED_TL, rw_opts},
200 	{"-?", CTLADM_CMD_HELP, CTLADM_ARG_NONE, NULL},
201 	{"-h", CTLADM_CMD_HELP, CTLADM_ARG_NONE, NULL},
202 	{NULL, 0, 0, NULL}
203 };
204 
205 
206 ctladm_optret getoption(struct ctladm_opts *table, char *arg, uint32_t *cmdnum,
207 			ctladm_cmdargs *argnum, const char **subopt);
208 static int cctl_dump_ooa(int fd, int argc, char **argv);
209 static int cctl_port(int fd, int argc, char **argv, char *combinedopt);
210 static int cctl_do_io(int fd, int retries, union ctl_io *io, const char *func);
211 static int cctl_delay(int fd, int lun, int argc, char **argv,
212 		      char *combinedopt);
213 static int cctl_lunlist(int fd);
214 static int cctl_sync_cache(int fd, int lun, int iid, int retries,
215 			   int argc, char **argv, char *combinedopt);
216 static int cctl_start_stop(int fd, int lun, int iid, int retries,
217 			   int start, int argc, char **argv, char *combinedopt);
218 static int cctl_mode_sense(int fd, int lun, int iid, int retries,
219 			   int argc, char **argv, char *combinedopt);
220 static int cctl_read_capacity(int fd, int lun, int iid,
221 			      int retries, int argc, char **argv,
222 			      char *combinedopt);
223 static int cctl_read_write(int fd, int lun, int iid, int retries,
224 			   int argc, char **argv, char *combinedopt,
225 			   ctladm_cmdfunction command);
226 static int cctl_get_luns(int fd, int lun, int iid, int retries,
227 			 struct scsi_report_luns_data **lun_data,
228 			 uint32_t *num_luns);
229 static int cctl_report_luns(int fd, int lun, int iid, int retries);
230 static int cctl_tur(int fd, int lun, int iid, int retries);
231 static int cctl_get_inquiry(int fd, int lun, int iid, int retries,
232 			    char *path_str, int path_len,
233 			    struct scsi_inquiry_data *inq_data);
234 static int cctl_inquiry(int fd, int lun, int iid, int retries);
235 static int cctl_req_sense(int fd, int lun, int iid, int retries);
236 static int cctl_persistent_reserve_in(int fd, int lun,
237 				      int initiator, int argc, char **argv,
238 				      char *combinedopt, int retry_count);
239 static int cctl_persistent_reserve_out(int fd, int lun,
240 				       int initiator, int argc, char **argv,
241 				       char *combinedopt, int retry_count);
242 static int cctl_create_lun(int fd, int argc, char **argv, char *combinedopt);
243 static int cctl_inquiry_vpd_devid(int fd, int lun, int initiator);
244 static int cctl_report_target_port_group(int fd, int lun, int initiator);
245 static int cctl_modify_lun(int fd, int argc, char **argv, char *combinedopt);
246 static int cctl_portlist(int fd, int argc, char **argv, char *combinedopt);
247 
248 ctladm_optret
249 getoption(struct ctladm_opts *table, char *arg, uint32_t *cmdnum,
250 	  ctladm_cmdargs *argnum, const char **subopt)
251 {
252 	struct ctladm_opts *opts;
253 	int num_matches = 0;
254 
255 	for (opts = table; (opts != NULL) && (opts->optname != NULL);
256 	     opts++) {
257 		if (strncmp(opts->optname, arg, strlen(arg)) == 0) {
258 			*cmdnum = opts->cmdnum;
259 			*argnum = opts->argnum;
260 			*subopt = opts->subopt;
261 
262 			if (strcmp(opts->optname, arg) == 0)
263 				return (CC_OR_FOUND);
264 
265 			if (++num_matches > 1)
266 				return(CC_OR_AMBIGUOUS);
267 		}
268 	}
269 
270 	if (num_matches > 0)
271 		return(CC_OR_FOUND);
272 	else
273 		return(CC_OR_NOT_FOUND);
274 }
275 
276 static int
277 cctl_dump_ooa(int fd, int argc, char **argv)
278 {
279 	struct ctl_ooa ooa;
280 	long double cmd_latency;
281 	int num_entries, len, lun = -1, retval = 0;
282 	unsigned int i;
283 
284 	num_entries = 104;
285 
286 	if ((argc > 2) && (isdigit(argv[2][0])))
287 		lun = strtol(argv[2], NULL, 0);
288 retry:
289 
290 	len = num_entries * sizeof(struct ctl_ooa_entry);
291 	bzero(&ooa, sizeof(ooa));
292 	ooa.entries = malloc(len);
293 	if (ooa.entries == NULL) {
294 		warn("%s: error mallocing %d bytes", __func__, len);
295 		return (1);
296 	}
297 	if (lun >= 0) {
298 		ooa.lun_num = lun;
299 	} else
300 		ooa.flags |= CTL_OOA_FLAG_ALL_LUNS;
301 	ooa.alloc_len = len;
302 	ooa.alloc_num = num_entries;
303 	if (ioctl(fd, CTL_GET_OOA, &ooa) == -1) {
304 		warn("%s: CTL_GET_OOA ioctl failed", __func__);
305 		retval = 1;
306 		goto bailout;
307 	}
308 
309 	if (ooa.status == CTL_OOA_NEED_MORE_SPACE) {
310 		num_entries = num_entries * 2;
311 		free(ooa.entries);
312 		ooa.entries = NULL;
313 		goto retry;
314 	}
315 
316 	if (ooa.status != CTL_OOA_OK) {
317 		warnx("%s: CTL_GET_OOA ioctl returned error %d", __func__,
318 		      ooa.status);
319 		retval = 1;
320 		goto bailout;
321 	}
322 
323 	fprintf(stdout, "Dumping OOA queues\n");
324 	for (i = 0; i < ooa.fill_num; i++) {
325 		struct ctl_ooa_entry *entry;
326 		char cdb_str[(SCSI_MAX_CDBLEN * 3) +1];
327 		struct bintime delta_bt;
328 		struct timespec ts;
329 
330 		entry = &ooa.entries[i];
331 
332 		delta_bt = ooa.cur_bt;
333 		bintime_sub(&delta_bt, &entry->start_bt);
334 		bintime2timespec(&delta_bt, &ts);
335 		cmd_latency = ts.tv_sec * 1000;
336 		if (ts.tv_nsec > 0)
337 			cmd_latency += ts.tv_nsec / 1000000;
338 
339 		fprintf(stdout, "LUN %jd tag 0x%04x%s%s%s%s%s: %s. CDB: %s "
340 			"(%0.0Lf ms)\n",
341 			(intmax_t)entry->lun_num, entry->tag_num,
342 			(entry->cmd_flags & CTL_OOACMD_FLAG_BLOCKED) ?
343 			 " BLOCKED" : "",
344 			(entry->cmd_flags & CTL_OOACMD_FLAG_DMA) ? " DMA" : "",
345 			(entry->cmd_flags & CTL_OOACMD_FLAG_DMA_QUEUED) ?
346 			 " DMAQUEUED" : "",
347 			(entry->cmd_flags & CTL_OOACMD_FLAG_ABORT) ?
348 			 " ABORT" : "",
349 			(entry->cmd_flags & CTL_OOACMD_FLAG_RTR) ? " RTR" :"",
350 			scsi_op_desc(entry->cdb[0], NULL),
351 			scsi_cdb_string(entry->cdb, cdb_str, sizeof(cdb_str)),
352 			cmd_latency);
353 	}
354 	fprintf(stdout, "OOA queues dump done\n");
355 
356 bailout:
357 	free(ooa.entries);
358 	return (retval);
359 }
360 
361 static int
362 cctl_dump_structs(int fd, ctladm_cmdargs cmdargs __unused)
363 {
364 	if (ioctl(fd, CTL_DUMP_STRUCTS) == -1) {
365 		warn(__func__);
366 		return (1);
367 	}
368 	return (0);
369 }
370 
371 typedef enum {
372 	CCTL_PORT_MODE_NONE,
373 	CCTL_PORT_MODE_LIST,
374 	CCTL_PORT_MODE_SET,
375 	CCTL_PORT_MODE_ON,
376 	CCTL_PORT_MODE_OFF,
377 	CCTL_PORT_MODE_CREATE,
378 	CCTL_PORT_MODE_REMOVE
379 } cctl_port_mode;
380 
381 static struct ctladm_opts cctl_fe_table[] = {
382 	{"fc", CTL_PORT_FC, CTLADM_ARG_NONE, NULL},
383 	{"scsi", CTL_PORT_SCSI, CTLADM_ARG_NONE, NULL},
384 	{"internal", CTL_PORT_INTERNAL, CTLADM_ARG_NONE, NULL},
385 	{"iscsi", CTL_PORT_ISCSI, CTLADM_ARG_NONE, NULL},
386 	{"sas", CTL_PORT_SAS, CTLADM_ARG_NONE, NULL},
387 	{"all", CTL_PORT_ALL, CTLADM_ARG_NONE, NULL},
388 	{NULL, 0, 0, NULL}
389 };
390 
391 static int
392 cctl_port(int fd, int argc, char **argv, char *combinedopt)
393 {
394 	int c;
395 	int32_t targ_port = -1;
396 	int retval = 0;
397 	int wwnn_set = 0, wwpn_set = 0;
398 	uint64_t wwnn = 0, wwpn = 0;
399 	cctl_port_mode port_mode = CCTL_PORT_MODE_NONE;
400 	struct ctl_port_entry entry;
401 	struct ctl_req req;
402 	char *driver = NULL;
403 	nvlist_t *option_list;
404 	ctl_port_type port_type = CTL_PORT_NONE;
405 	int quiet = 0, xml = 0;
406 
407 	option_list = nvlist_create(0);
408 	if (option_list == NULL)
409 		err(1, "%s: unable to allocate nvlist", __func__);
410 
411 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
412 		switch (c) {
413 		case 'l':
414 			if (port_mode != CCTL_PORT_MODE_NONE)
415 				goto bailout_badarg;
416 
417 			port_mode = CCTL_PORT_MODE_LIST;
418 			break;
419 		case 'c':
420 			port_mode = CCTL_PORT_MODE_CREATE;
421 			break;
422 		case 'r':
423 			port_mode = CCTL_PORT_MODE_REMOVE;
424 			break;
425 		case 'o':
426 			if (port_mode != CCTL_PORT_MODE_NONE)
427 				goto bailout_badarg;
428 
429 			if (strcasecmp(optarg, "on") == 0)
430 				port_mode = CCTL_PORT_MODE_ON;
431 			else if (strcasecmp(optarg, "off") == 0)
432 				port_mode = CCTL_PORT_MODE_OFF;
433 			else {
434 				warnx("Invalid -o argument %s, \"on\" or "
435 				      "\"off\" are the only valid args",
436 				      optarg);
437 				retval = 1;
438 				goto bailout;
439 			}
440 			break;
441 		case 'O': {
442 			char *tmpstr;
443 			char *name, *value;
444 
445 			tmpstr = strdup(optarg);
446 			name = strsep(&tmpstr, "=");
447 			if (name == NULL) {
448 				warnx("%s: option -O takes \"name=value\""
449 				      "argument", __func__);
450 				retval = 1;
451 				goto bailout;
452 			}
453 			value = strsep(&tmpstr, "=");
454 			if (value == NULL) {
455 				warnx("%s: option -O takes \"name=value\""
456 				      "argument", __func__);
457 				retval = 1;
458 				goto bailout;
459 			}
460 
461 			free(tmpstr);
462 			nvlist_add_string(option_list, name, value);
463 			break;
464 		}
465 		case 'd':
466 			if (driver != NULL) {
467 				warnx("%s: option -d cannot be specified twice",
468 				    __func__);
469 				retval = 1;
470 				goto bailout;
471 			}
472 
473 			driver = strdup(optarg);
474 			break;
475 		case 'p':
476 			targ_port = strtol(optarg, NULL, 0);
477 			break;
478 		case 'q':
479 			quiet = 1;
480 			break;
481 		case 't': {
482 			ctladm_optret optret;
483 			ctladm_cmdargs argnum;
484 			const char *subopt;
485 			ctl_port_type tmp_port_type;
486 
487 			optret = getoption(cctl_fe_table, optarg, &tmp_port_type,
488 					   &argnum, &subopt);
489 			if (optret == CC_OR_AMBIGUOUS) {
490 				warnx("%s: ambiguous frontend type %s",
491 				      __func__, optarg);
492 				retval = 1;
493 				goto bailout;
494 			} else if (optret == CC_OR_NOT_FOUND) {
495 				warnx("%s: invalid frontend type %s",
496 				      __func__, optarg);
497 				retval = 1;
498 				goto bailout;
499 			}
500 
501 			port_type |= tmp_port_type;
502 			break;
503 		}
504 		case 'w':
505 			if ((port_mode != CCTL_PORT_MODE_NONE)
506 			 && (port_mode != CCTL_PORT_MODE_SET))
507 				goto bailout_badarg;
508 
509 			port_mode = CCTL_PORT_MODE_SET;
510 
511 			wwnn = strtoull(optarg, NULL, 0);
512 			wwnn_set = 1;
513 			break;
514 		case 'W':
515 			if ((port_mode != CCTL_PORT_MODE_NONE)
516 			 && (port_mode != CCTL_PORT_MODE_SET))
517 				goto bailout_badarg;
518 
519 			port_mode = CCTL_PORT_MODE_SET;
520 
521 			wwpn = strtoull(optarg, NULL, 0);
522 			wwpn_set = 1;
523 			break;
524 		case 'x':
525 			xml = 1;
526 			break;
527 		}
528 	}
529 
530 	if (driver == NULL)
531 		driver = strdup("ioctl");
532 
533 	/*
534 	 * The user can specify either one or more frontend types (-t), or
535 	 * a specific frontend, but not both.
536 	 *
537 	 * If the user didn't specify a frontend type or number, set it to
538 	 * all.  This is primarily needed for the enable/disable ioctls.
539 	 * This will be a no-op for the listing code.  For the set ioctl,
540 	 * we'll throw an error, since that only works on one port at a time.
541 	 */
542 	if ((port_type != CTL_PORT_NONE) && (targ_port != -1)) {
543 		warnx("%s: can only specify one of -t or -n", __func__);
544 		retval = 1;
545 		goto bailout;
546 	} else if ((targ_port == -1) && (port_type == CTL_PORT_NONE))
547 		port_type = CTL_PORT_ALL;
548 
549 	bzero(&entry, sizeof(entry));
550 
551 	/*
552 	 * These are needed for all but list/dump mode.
553 	 */
554 	entry.port_type = port_type;
555 	entry.targ_port = targ_port;
556 
557 	switch (port_mode) {
558 	case CCTL_PORT_MODE_LIST: {
559 		char opts[] = "xq";
560 		char argx[] = "-x";
561 		char argq[] = "-q";
562 		char *argvx[2];
563 		int argcx = 0;
564 
565 		optind = 0;
566 		optreset = 1;
567 		if (xml)
568 			argvx[argcx++] = argx;
569 		if (quiet)
570 			argvx[argcx++] = argq;
571 		cctl_portlist(fd, argcx, argvx, opts);
572 		break;
573 	}
574 	case CCTL_PORT_MODE_REMOVE:
575 		if (targ_port == -1) {
576 			warnx("%s: -r require -p", __func__);
577 			retval = 1;
578 			goto bailout;
579 		}
580 	case CCTL_PORT_MODE_CREATE: {
581 		bzero(&req, sizeof(req));
582 		strlcpy(req.driver, driver, sizeof(req.driver));
583 
584 		if (port_mode == CCTL_PORT_MODE_REMOVE) {
585 			req.reqtype = CTL_REQ_REMOVE;
586 			nvlist_add_stringf(option_list, "port_id", "%d",
587 			    targ_port);
588 		} else
589 			req.reqtype = CTL_REQ_CREATE;
590 
591 		req.args = nvlist_pack(option_list, &req.args_len);
592 		if (req.args == NULL) {
593 			warn("%s: error packing nvlist", __func__);
594 			retval = 1;
595 			goto bailout;
596 		}
597 
598 		retval = ioctl(fd, CTL_PORT_REQ, &req);
599 		free(req.args);
600 		if (retval == -1) {
601 			warn("%s: CTL_PORT_REQ ioctl failed", __func__);
602 			retval = 1;
603 			goto bailout;
604 		}
605 
606 		switch (req.status) {
607 		case CTL_LUN_ERROR:
608 			warnx("error: %s", req.error_str);
609 			retval = 1;
610 			goto bailout;
611 		case CTL_LUN_WARNING:
612 			warnx("warning: %s", req.error_str);
613 			break;
614 		case CTL_LUN_OK:
615 			break;
616 		default:
617 			warnx("unknown status: %d", req.status);
618 			retval = 1;
619 			goto bailout;
620 		}
621 
622 		break;
623 	}
624 	case CCTL_PORT_MODE_SET:
625 		if (targ_port == -1) {
626 			warnx("%s: -w and -W require -n", __func__);
627 			retval = 1;
628 			goto bailout;
629 		}
630 
631 		if (wwnn_set) {
632 			entry.flags |= CTL_PORT_WWNN_VALID;
633 			entry.wwnn = wwnn;
634 		}
635 		if (wwpn_set) {
636 			entry.flags |= CTL_PORT_WWPN_VALID;
637 			entry.wwpn = wwpn;
638 		}
639 
640 		if (ioctl(fd, CTL_SET_PORT_WWNS, &entry) == -1) {
641 			warn("%s: CTL_SET_PORT_WWNS ioctl failed", __func__);
642 			retval = 1;
643 			goto bailout;
644 		}
645 		break;
646 	case CCTL_PORT_MODE_ON:
647 		if (ioctl(fd, CTL_ENABLE_PORT, &entry) == -1) {
648 			warn("%s: CTL_ENABLE_PORT ioctl failed", __func__);
649 			retval = 1;
650 			goto bailout;
651 		}
652 		fprintf(stdout, "Front End Ports enabled\n");
653 		break;
654 	case CCTL_PORT_MODE_OFF:
655 		if (ioctl(fd, CTL_DISABLE_PORT, &entry) == -1) {
656 			warn("%s: CTL_DISABLE_PORT ioctl failed", __func__);
657 			retval = 1;
658 			goto bailout;
659 		}
660 		fprintf(stdout, "Front End Ports disabled\n");
661 		break;
662 	default:
663 		warnx("%s: one of -l, -o or -w/-W must be specified", __func__);
664 		retval = 1;
665 		goto bailout;
666 		break;
667 	}
668 
669 bailout:
670 	nvlist_destroy(option_list);
671 	free(driver);
672 	return (retval);
673 
674 bailout_badarg:
675 	warnx("%s: only one of -l, -o or -w/-W may be specified", __func__);
676 	return (1);
677 }
678 
679 static int
680 cctl_do_io(int fd, int retries, union ctl_io *io, const char *func)
681 {
682 	do {
683 		if (ioctl(fd, CTL_IO, io) == -1) {
684 			warn("%s: error sending CTL_IO ioctl", func);
685 			return (-1);
686 		}
687 	} while (((io->io_hdr.status & CTL_STATUS_MASK) != CTL_SUCCESS)
688 	      && (retries-- > 0));
689 
690 	return (0);
691 }
692 
693 static int
694 cctl_delay(int fd, int lun, int argc, char **argv,
695 	   char *combinedopt)
696 {
697 	struct ctl_io_delay_info delay_info;
698 	char *delayloc = NULL;
699 	char *delaytype = NULL;
700 	int delaytime = -1;
701 	int retval;
702 	int c;
703 
704 	retval = 0;
705 
706 	memset(&delay_info, 0, sizeof(delay_info));
707 
708 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
709 		switch (c) {
710 		case 'T':
711 			delaytype = strdup(optarg);
712 			break;
713 		case 'l':
714 			delayloc = strdup(optarg);
715 			break;
716 		case 't':
717 			delaytime = strtoul(optarg, NULL, 0);
718 			break;
719 		}
720 	}
721 
722 	if (delaytime == -1) {
723 		warnx("%s: you must specify the delaytime with -t", __func__);
724 		retval = 1;
725 		goto bailout;
726 	}
727 
728 	if (strcasecmp(delayloc, "datamove") == 0)
729 		delay_info.delay_loc = CTL_DELAY_LOC_DATAMOVE;
730 	else if (strcasecmp(delayloc, "done") == 0)
731 		delay_info.delay_loc = CTL_DELAY_LOC_DONE;
732 	else {
733 		warnx("%s: invalid delay location %s", __func__, delayloc);
734 		retval = 1;
735 		goto bailout;
736 	}
737 
738 	if ((delaytype == NULL)
739 	 || (strcmp(delaytype, "oneshot") == 0))
740 		delay_info.delay_type = CTL_DELAY_TYPE_ONESHOT;
741 	else if (strcmp(delaytype, "cont") == 0)
742 		delay_info.delay_type = CTL_DELAY_TYPE_CONT;
743 	else {
744 		warnx("%s: invalid delay type %s", __func__, delaytype);
745 		retval = 1;
746 		goto bailout;
747 	}
748 
749 	delay_info.lun_id = lun;
750 	delay_info.delay_secs = delaytime;
751 
752 	if (ioctl(fd, CTL_DELAY_IO, &delay_info) == -1) {
753 		warn("%s: CTL_DELAY_IO ioctl failed", __func__);
754 		retval = 1;
755 		goto bailout;
756 	}
757 	switch (delay_info.status) {
758 	case CTL_DELAY_STATUS_NONE:
759 		warnx("%s: no delay status??", __func__);
760 		retval = 1;
761 		break;
762 	case CTL_DELAY_STATUS_OK:
763 		break;
764 	case CTL_DELAY_STATUS_INVALID_LUN:
765 		warnx("%s: invalid lun %d", __func__, lun);
766 		retval = 1;
767 		break;
768 	case CTL_DELAY_STATUS_INVALID_TYPE:
769 		warnx("%s: invalid delay type %d", __func__,
770 		      delay_info.delay_type);
771 		retval = 1;
772 		break;
773 	case CTL_DELAY_STATUS_INVALID_LOC:
774 		warnx("%s: delay location %s not implemented?", __func__,
775 		      delayloc);
776 		retval = 1;
777 		break;
778 	case CTL_DELAY_STATUS_NOT_IMPLEMENTED:
779 		warnx("%s: delay not implemented in the kernel", __func__);
780 		warnx("%s: recompile with the CTL_IO_DELAY flag set", __func__);
781 		retval = 1;
782 		break;
783 	default:
784 		warnx("%s: unknown delay return status %d", __func__,
785 		      delay_info.status);
786 		retval = 1;
787 		break;
788 	}
789 
790 bailout:
791 	free(delayloc);
792 	free(delaytype);
793 	return (retval);
794 }
795 
796 static struct ctladm_opts cctl_err_types[] = {
797 	{"aborted", CTL_LUN_INJ_ABORTED, CTLADM_ARG_NONE, NULL},
798 	{"mediumerr", CTL_LUN_INJ_MEDIUM_ERR, CTLADM_ARG_NONE, NULL},
799 	{"ua", CTL_LUN_INJ_UA, CTLADM_ARG_NONE, NULL},
800 	{"custom", CTL_LUN_INJ_CUSTOM, CTLADM_ARG_NONE, NULL},
801 	{NULL, 0, 0, NULL}
802 
803 };
804 
805 static struct ctladm_opts cctl_err_patterns[] = {
806 	{"read", CTL_LUN_PAT_READ, CTLADM_ARG_NONE, NULL},
807 	{"write", CTL_LUN_PAT_WRITE, CTLADM_ARG_NONE, NULL},
808 	{"rw", CTL_LUN_PAT_READWRITE, CTLADM_ARG_NONE, NULL},
809 	{"readwrite", CTL_LUN_PAT_READWRITE, CTLADM_ARG_NONE, NULL},
810 	{"readcap", CTL_LUN_PAT_READCAP, CTLADM_ARG_NONE, NULL},
811 	{"tur", CTL_LUN_PAT_TUR, CTLADM_ARG_NONE, NULL},
812 	{"any", CTL_LUN_PAT_ANY, CTLADM_ARG_NONE, NULL},
813 #if 0
814 	{"cmd", CTL_LUN_PAT_CMD,  CTLADM_ARG_NONE, NULL},
815 #endif
816 	{NULL, 0, 0, NULL}
817 };
818 
819 static int
820 cctl_error_inject(int fd, uint32_t lun, int argc, char **argv,
821 		  char *combinedopt)
822 {
823 	int retval = 0;
824 	struct ctl_error_desc err_desc;
825 	uint64_t lba = 0;
826 	uint32_t len = 0;
827 	uint64_t delete_id = 0;
828 	int delete_id_set = 0;
829 	int continuous = 0;
830 	int sense_len = 0;
831 	int fd_sense = 0;
832 	int c;
833 
834 	bzero(&err_desc, sizeof(err_desc));
835 	err_desc.lun_id = lun;
836 
837 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
838 		switch (c) {
839 		case 'c':
840 			continuous = 1;
841 			break;
842 		case 'd':
843 			delete_id = strtoull(optarg, NULL, 0);
844 			delete_id_set = 1;
845 			break;
846 		case 'i':
847 		case 'p': {
848 			ctladm_optret optret;
849 			ctladm_cmdargs argnum;
850 			const char *subopt;
851 
852 			if (c == 'i') {
853 				ctl_lun_error err_type;
854 
855 				if (err_desc.lun_error != CTL_LUN_INJ_NONE) {
856 					warnx("%s: can't specify multiple -i "
857 					      "arguments", __func__);
858 					retval = 1;
859 					goto bailout;
860 				}
861 				optret = getoption(cctl_err_types, optarg,
862 						   &err_type, &argnum, &subopt);
863 				err_desc.lun_error = err_type;
864 			} else {
865 				ctl_lun_error_pattern pattern;
866 
867 				optret = getoption(cctl_err_patterns, optarg,
868 						   &pattern, &argnum, &subopt);
869 				err_desc.error_pattern |= pattern;
870 			}
871 
872 			if (optret == CC_OR_AMBIGUOUS) {
873 				warnx("%s: ambiguous argument %s", __func__,
874 				      optarg);
875 				retval = 1;
876 				goto bailout;
877 			} else if (optret == CC_OR_NOT_FOUND) {
878 				warnx("%s: argument %s not found", __func__,
879 				      optarg);
880 				retval = 1;
881 				goto bailout;
882 			}
883 			break;
884 		}
885 		case 'r': {
886 			char *tmpstr, *tmpstr2;
887 
888 			tmpstr = strdup(optarg);
889 			if (tmpstr == NULL) {
890 				warn("%s: error duplicating string %s",
891 				     __func__, optarg);
892 				retval = 1;
893 				goto bailout;
894 			}
895 
896 			tmpstr2 = strsep(&tmpstr, ",");
897 			if (tmpstr2 == NULL) {
898 				warnx("%s: invalid -r argument %s", __func__,
899 				      optarg);
900 				retval = 1;
901 				free(tmpstr);
902 				goto bailout;
903 			}
904 			lba = strtoull(tmpstr2, NULL, 0);
905 			tmpstr2 = strsep(&tmpstr, ",");
906 			if (tmpstr2 == NULL) {
907 				warnx("%s: no len argument for -r lba,len, got"
908 				      " %s", __func__, optarg);
909 				retval = 1;
910 				free(tmpstr);
911 				goto bailout;
912 			}
913 			len = strtoul(tmpstr2, NULL, 0);
914 			free(tmpstr);
915 			break;
916 		}
917 		case 's': {
918 			struct get_hook hook;
919 			char *sensestr;
920 
921 			sense_len = strtol(optarg, NULL, 0);
922 			if (sense_len <= 0) {
923 				warnx("invalid number of sense bytes %d",
924 				      sense_len);
925 				retval = 1;
926 				goto bailout;
927 			}
928 
929 			sense_len = MIN(sense_len, SSD_FULL_SIZE);
930 
931 			hook.argc = argc - optind;
932 			hook.argv = argv + optind;
933 			hook.got = 0;
934 
935 			sensestr = cget(&hook, NULL);
936 			if ((sensestr != NULL)
937 			 && (sensestr[0] == '-')) {
938 				fd_sense = 1;
939 			} else {
940 				buff_encode_visit(
941 				    (uint8_t *)&err_desc.custom_sense,
942 				    sense_len, sensestr, iget, &hook);
943 			}
944 			optind += hook.got;
945 			break;
946 		}
947 		default:
948 			break;
949 		}
950 	}
951 
952 	if (delete_id_set != 0) {
953 		err_desc.serial = delete_id;
954 		if (ioctl(fd, CTL_ERROR_INJECT_DELETE, &err_desc) == -1) {
955 			warn("%s: error issuing CTL_ERROR_INJECT_DELETE ioctl",
956 			     __func__);
957 			retval = 1;
958 		}
959 		goto bailout;
960 	}
961 
962 	if (err_desc.lun_error == CTL_LUN_INJ_NONE) {
963 		warnx("%s: error injection command (-i) needed",
964 		      __func__);
965 		retval = 1;
966 		goto bailout;
967 	} else if ((err_desc.lun_error == CTL_LUN_INJ_CUSTOM)
968 		&& (sense_len == 0)) {
969 		warnx("%s: custom error requires -s", __func__);
970 		retval = 1;
971 		goto bailout;
972 	}
973 
974 	if (continuous != 0)
975 		err_desc.lun_error |= CTL_LUN_INJ_CONTINUOUS;
976 
977 	/*
978 	 * If fd_sense is set, we need to read the sense data the user
979 	 * wants returned from stdin.
980 	 */
981         if (fd_sense == 1) {
982 		ssize_t amt_read;
983 		int amt_to_read = sense_len;
984 		u_int8_t *buf_ptr = (uint8_t *)&err_desc.custom_sense;
985 
986 		for (amt_read = 0; amt_to_read > 0;
987 		     amt_read = read(STDIN_FILENO, buf_ptr, amt_to_read)) {
988 			if (amt_read == -1) {
989 				warn("error reading sense data from stdin");
990 				retval = 1;
991 				goto bailout;
992 			}
993 			amt_to_read -= amt_read;
994 			buf_ptr += amt_read;
995 		}
996 	}
997 
998 	if (err_desc.error_pattern == CTL_LUN_PAT_NONE) {
999 		warnx("%s: command pattern (-p) needed", __func__);
1000 		retval = 1;
1001 		goto bailout;
1002 	}
1003 
1004 	if (len != 0) {
1005 		err_desc.error_pattern |= CTL_LUN_PAT_RANGE;
1006 		/*
1007 		 * We could check here to see whether it's a read/write
1008 		 * command, but that will be pointless once we allow
1009 		 * custom patterns.  At that point, the user could specify
1010 		 * a READ(6) CDB type, and we wouldn't have an easy way here
1011 		 * to verify whether range checking is possible there.  The
1012 		 * user will just figure it out when his error never gets
1013 		 * executed.
1014 		 */
1015 #if 0
1016 		if ((err_desc.pattern & CTL_LUN_PAT_READWRITE) == 0) {
1017 			warnx("%s: need read and/or write pattern if range "
1018 			      "is specified", __func__);
1019 			retval = 1;
1020 			goto bailout;
1021 		}
1022 #endif
1023 		err_desc.lba_range.lba = lba;
1024 		err_desc.lba_range.len = len;
1025 	}
1026 
1027 	if (ioctl(fd, CTL_ERROR_INJECT, &err_desc) == -1) {
1028 		warn("%s: error issuing CTL_ERROR_INJECT ioctl", __func__);
1029 		retval = 1;
1030 	} else {
1031 		printf("Error injection succeeded, serial number is %ju\n",
1032 		       (uintmax_t)err_desc.serial);
1033 	}
1034 bailout:
1035 
1036 	return (retval);
1037 }
1038 
1039 static int
1040 cctl_lunlist(int fd)
1041 {
1042 	struct scsi_report_luns_data *lun_data;
1043 	struct scsi_inquiry_data *inq_data;
1044 	uint32_t num_luns;
1045 	int initid;
1046 	unsigned int i;
1047 	int retval;
1048 
1049 	inq_data = NULL;
1050 	initid = 7;
1051 
1052 	/*
1053 	 * XXX KDM assuming LUN 0 is fine, but we may need to change this
1054 	 * if we ever acquire the ability to have multiple targets.
1055 	 */
1056 	if ((retval = cctl_get_luns(fd, /*lun*/ 0, initid,
1057 				    /*retries*/ 2, &lun_data, &num_luns)) != 0)
1058 		goto bailout;
1059 
1060 	inq_data = malloc(sizeof(*inq_data));
1061 	if (inq_data == NULL) {
1062 		warn("%s: couldn't allocate memory for inquiry data\n",
1063 		     __func__);
1064 		retval = 1;
1065 		goto bailout;
1066 	}
1067 	for (i = 0; i < num_luns; i++) {
1068 		char scsi_path[40];
1069 		int lun_val;
1070 
1071 		switch (lun_data->luns[i].lundata[0] & RPL_LUNDATA_ATYP_MASK) {
1072 		case RPL_LUNDATA_ATYP_PERIPH:
1073 			lun_val = lun_data->luns[i].lundata[1];
1074 			break;
1075 		case RPL_LUNDATA_ATYP_FLAT:
1076 			lun_val = (lun_data->luns[i].lundata[0] &
1077 				RPL_LUNDATA_FLAT_LUN_MASK) |
1078 				(lun_data->luns[i].lundata[1] <<
1079 				RPL_LUNDATA_FLAT_LUN_BITS);
1080 			break;
1081 		case RPL_LUNDATA_ATYP_LUN:
1082 		case RPL_LUNDATA_ATYP_EXTLUN:
1083 		default:
1084 			fprintf(stdout, "Unsupported LUN format %d\n",
1085 				lun_data->luns[i].lundata[0] &
1086 				RPL_LUNDATA_ATYP_MASK);
1087 			lun_val = -1;
1088 			break;
1089 		}
1090 		if (lun_val == -1)
1091 			continue;
1092 
1093 		if ((retval = cctl_get_inquiry(fd, lun_val, initid,
1094 					       /*retries*/ 2, scsi_path,
1095 					       sizeof(scsi_path),
1096 					       inq_data)) != 0) {
1097 			goto bailout;
1098 		}
1099 		printf("%s", scsi_path);
1100 		scsi_print_inquiry(inq_data);
1101 	}
1102 bailout:
1103 
1104 	if (lun_data != NULL)
1105 		free(lun_data);
1106 
1107 	if (inq_data != NULL)
1108 		free(inq_data);
1109 
1110 	return (retval);
1111 }
1112 
1113 static int
1114 cctl_sync_cache(int fd, int lun, int iid, int retries,
1115 		int argc, char **argv, char *combinedopt)
1116 {
1117 	union ctl_io *io;
1118 	int cdb_size = -1;
1119 	int retval;
1120 	uint64_t our_lba = 0;
1121 	uint32_t our_block_count = 0;
1122 	int reladr = 0, immed = 0;
1123 	int c;
1124 
1125 	retval = 0;
1126 
1127 	io = ctl_scsi_alloc_io(iid);
1128 	if (io == NULL) {
1129 		warnx("%s: can't allocate memory", __func__);
1130 		return (1);
1131 	}
1132 
1133 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
1134 		switch (c) {
1135 		case 'b':
1136 			our_block_count = strtoul(optarg, NULL, 0);
1137 			break;
1138 		case 'c':
1139 			cdb_size = strtol(optarg, NULL, 0);
1140 			break;
1141 		case 'i':
1142 			immed = 1;
1143 			break;
1144 		case 'l':
1145 			our_lba = strtoull(optarg, NULL, 0);
1146 			break;
1147 		case 'r':
1148 			reladr = 1;
1149 			break;
1150 		default:
1151 			break;
1152 		}
1153 	}
1154 
1155 	if (cdb_size != -1) {
1156 		switch (cdb_size) {
1157 		case 10:
1158 		case 16:
1159 			break;
1160 		default:
1161 			warnx("%s: invalid cdbsize %d, valid sizes are 10 "
1162 			      "and 16", __func__, cdb_size);
1163 			retval = 1;
1164 			goto bailout;
1165 			break; /* NOTREACHED */
1166 		}
1167 	} else
1168 		cdb_size = 10;
1169 
1170 	ctl_scsi_sync_cache(/*io*/ io,
1171 			    /*immed*/ immed,
1172 			    /*reladr*/ reladr,
1173 			    /*minimum_cdb_size*/ cdb_size,
1174 			    /*starting_lba*/ our_lba,
1175 			    /*block_count*/ our_block_count,
1176 			    /*tag_type*/ CTL_TAG_SIMPLE,
1177 			    /*control*/ 0);
1178 
1179 	io->io_hdr.nexus.targ_lun = lun;
1180 	io->io_hdr.nexus.initid = iid;
1181 
1182 	if (cctl_do_io(fd, retries, io, __func__) != 0) {
1183 		retval = 1;
1184 		goto bailout;
1185 	}
1186 
1187 	if ((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS) {
1188 		fprintf(stdout, "Cache synchronized successfully\n");
1189 	} else
1190 		ctl_io_error_print(io, NULL, stderr);
1191 bailout:
1192 	ctl_scsi_free_io(io);
1193 
1194 	return (retval);
1195 }
1196 
1197 static int
1198 cctl_start_stop(int fd, int lun, int iid, int retries, int start,
1199 		int argc, char **argv, char *combinedopt)
1200 {
1201 	union ctl_io *io;
1202 	char scsi_path[40];
1203 	int immed = 0;
1204 	int retval, c;
1205 
1206 	retval = 0;
1207 
1208 	io = ctl_scsi_alloc_io(iid);
1209 	if (io == NULL) {
1210 		warnx("%s: can't allocate memory", __func__);
1211 		return (1);
1212 	}
1213 
1214 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
1215 		switch (c) {
1216 		case 'i':
1217 			immed = 1;
1218 			break;
1219 		default:
1220 			break;
1221 		}
1222 	}
1223 	/*
1224 	 * Use an ordered tag for the stop command, to guarantee that any
1225 	 * pending I/O will finish before the stop command executes.  This
1226 	 * would normally be the case anyway, since CTL will basically
1227 	 * treat the start/stop command as an ordered command with respect
1228 	 * to any other command except an INQUIRY.  (See ctl_ser_table.c.)
1229 	 */
1230 	ctl_scsi_start_stop(/*io*/ io,
1231 			    /*start*/ start,
1232 			    /*load_eject*/ 0,
1233 			    /*immediate*/ immed,
1234 			    /*power_conditions*/ SSS_PC_START_VALID,
1235 			    /*ctl_tag_type*/ start ? CTL_TAG_SIMPLE :
1236 						     CTL_TAG_ORDERED,
1237 			    /*control*/ 0);
1238 
1239 	io->io_hdr.nexus.targ_lun = lun;
1240 	io->io_hdr.nexus.initid = iid;
1241 
1242 	if (cctl_do_io(fd, retries, io, __func__) != 0) {
1243 		retval = 1;
1244 		goto bailout;
1245 	}
1246 
1247 	ctl_scsi_path_string(io, scsi_path, sizeof(scsi_path));
1248 	if ((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS) {
1249 		fprintf(stdout, "%s LUN %s successfully\n", scsi_path,
1250 			(start) ?  "started" : "stopped");
1251 	} else
1252 		ctl_io_error_print(io, NULL, stderr);
1253 
1254 bailout:
1255 	ctl_scsi_free_io(io);
1256 
1257 	return (retval);
1258 }
1259 
1260 static int
1261 cctl_mode_sense(int fd, int lun, int iid, int retries,
1262 		int argc, char **argv, char *combinedopt)
1263 {
1264 	union ctl_io *io;
1265 	uint32_t datalen;
1266 	uint8_t *dataptr;
1267 	int pc = -1, cdbsize, retval, dbd = 0, subpage = -1;
1268 	int list = 0;
1269 	int page_code = -1;
1270 	int c;
1271 
1272 	cdbsize = 0;
1273 	retval = 0;
1274 	dataptr = NULL;
1275 
1276 	io = ctl_scsi_alloc_io(iid);
1277 	if (io == NULL) {
1278 		warn("%s: can't allocate memory", __func__);
1279 		return (1);
1280 	}
1281 
1282 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
1283 		switch (c) {
1284 		case 'P':
1285 			pc = strtoul(optarg, NULL, 0);
1286 			break;
1287 		case 'S':
1288 			subpage = strtoul(optarg, NULL, 0);
1289 			break;
1290 		case 'd':
1291 			dbd = 1;
1292 			break;
1293 		case 'l':
1294 			list = 1;
1295 			break;
1296 		case 'm':
1297 			page_code = strtoul(optarg, NULL, 0);
1298 			break;
1299 		case 'c':
1300 			cdbsize = strtol(optarg, NULL, 0);
1301 			break;
1302 		default:
1303 			break;
1304 		}
1305 	}
1306 
1307 	if (((list == 0) && (page_code == -1))
1308 	 || ((list != 0) && (page_code != -1))) {
1309 		warnx("%s: you must specify either a page code (-m) or -l",
1310 		      __func__);
1311 		retval = 1;
1312 		goto bailout;
1313 	}
1314 
1315 	if ((page_code != -1)
1316 	 && ((page_code > SMS_ALL_PAGES_PAGE)
1317 	  || (page_code < 0))) {
1318 		warnx("%s: page code %d is out of range", __func__,
1319 		      page_code);
1320 		retval = 1;
1321 		goto bailout;
1322 	}
1323 
1324 	if (list == 1) {
1325 		page_code = SMS_ALL_PAGES_PAGE;
1326 		if (pc != -1) {
1327 			warnx("%s: arg -P makes no sense with -l",
1328 			      __func__);
1329 			retval = 1;
1330 			goto bailout;
1331 		}
1332 		if (subpage != -1) {
1333 			warnx("%s: arg -S makes no sense with -l", __func__);
1334 			retval = 1;
1335 			goto bailout;
1336 		}
1337 	}
1338 
1339 	if (pc == -1)
1340 		pc = SMS_PAGE_CTRL_CURRENT;
1341 	else {
1342 		if ((pc > 3)
1343 		 || (pc < 0)) {
1344 			warnx("%s: page control value %d is out of range: 0-3",
1345 			      __func__, pc);
1346 			retval = 1;
1347 			goto bailout;
1348 		}
1349 	}
1350 
1351 
1352 	if ((subpage != -1)
1353 	 && ((subpage > 255)
1354 	  || (subpage < 0))) {
1355 		warnx("%s: subpage code %d is out of range: 0-255", __func__,
1356 		      subpage);
1357 		retval = 1;
1358 		goto bailout;
1359 	}
1360 	if (cdbsize != 0) {
1361 		switch (cdbsize) {
1362 		case 6:
1363 		case 10:
1364 			break;
1365 		default:
1366 			warnx("%s: invalid cdbsize %d, valid sizes are 6 "
1367 			      "and 10", __func__, cdbsize);
1368 			retval = 1;
1369 			goto bailout;
1370 			break;
1371 		}
1372 	} else
1373 		cdbsize = 6;
1374 
1375 	if (subpage == -1)
1376 		subpage = 0;
1377 
1378 	if (cdbsize == 6)
1379 		datalen = 255;
1380 	else
1381 		datalen = 65535;
1382 
1383 	dataptr = (uint8_t *)malloc(datalen);
1384 	if (dataptr == NULL) {
1385 		warn("%s: can't allocate %d bytes", __func__, datalen);
1386 		retval = 1;
1387 		goto bailout;
1388 	}
1389 
1390 	memset(dataptr, 0, datalen);
1391 
1392 	ctl_scsi_mode_sense(io,
1393 			    /*data_ptr*/ dataptr,
1394 			    /*data_len*/ datalen,
1395 			    /*dbd*/ dbd,
1396 			    /*llbaa*/ 0,
1397 			    /*page_code*/ page_code,
1398 			    /*pc*/ pc << 6,
1399 			    /*subpage*/ subpage,
1400 			    /*minimum_cdb_size*/ cdbsize,
1401 			    /*tag_type*/ CTL_TAG_SIMPLE,
1402 			    /*control*/ 0);
1403 
1404 	io->io_hdr.nexus.targ_lun = lun;
1405 	io->io_hdr.nexus.initid = iid;
1406 
1407 	if (cctl_do_io(fd, retries, io, __func__) != 0) {
1408 		retval = 1;
1409 		goto bailout;
1410 	}
1411 
1412 	if ((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS) {
1413 		int pages_len, used_len;
1414 		uint32_t returned_len;
1415 		uint8_t *ndataptr;
1416 
1417 		if (io->scsiio.cdb[0] == MODE_SENSE_6) {
1418 			struct scsi_mode_hdr_6 *hdr6;
1419 			int bdlen;
1420 
1421 			hdr6 = (struct scsi_mode_hdr_6 *)dataptr;
1422 
1423 			returned_len = hdr6->datalen + 1;
1424 			bdlen = hdr6->block_descr_len;
1425 
1426 			ndataptr = (uint8_t *)((uint8_t *)&hdr6[1] + bdlen);
1427 		} else {
1428 			struct scsi_mode_hdr_10 *hdr10;
1429 			int bdlen;
1430 
1431 			hdr10 = (struct scsi_mode_hdr_10 *)dataptr;
1432 
1433 			returned_len = scsi_2btoul(hdr10->datalen) + 2;
1434 			bdlen = scsi_2btoul(hdr10->block_descr_len);
1435 
1436 			ndataptr = (uint8_t *)((uint8_t *)&hdr10[1] + bdlen);
1437 		}
1438 		/* just in case they can give us more than we allocated for */
1439 		returned_len = min(returned_len, datalen);
1440 		pages_len = returned_len - (ndataptr - dataptr);
1441 #if 0
1442 		fprintf(stdout, "returned_len = %d, pages_len = %d\n",
1443 			returned_len, pages_len);
1444 #endif
1445 		if (list == 1) {
1446 			fprintf(stdout, "Supported mode pages:\n");
1447 			for (used_len = 0; used_len < pages_len;) {
1448 				struct scsi_mode_page_header *header;
1449 
1450 				header = (struct scsi_mode_page_header *)
1451 					&ndataptr[used_len];
1452 				fprintf(stdout, "%d\n", header->page_code);
1453 				used_len += header->page_length + 2;
1454 			}
1455 		} else {
1456 			for (used_len = 0; used_len < pages_len; used_len++) {
1457 				fprintf(stdout, "0x%x ", ndataptr[used_len]);
1458 				if (((used_len+1) % 16) == 0)
1459 					fprintf(stdout, "\n");
1460 			}
1461 			fprintf(stdout, "\n");
1462 		}
1463 	} else
1464 		ctl_io_error_print(io, NULL, stderr);
1465 bailout:
1466 
1467 	ctl_scsi_free_io(io);
1468 
1469 	if (dataptr != NULL)
1470 		free(dataptr);
1471 
1472 	return (retval);
1473 }
1474 
1475 static int
1476 cctl_read_capacity(int fd, int lun, int iid, int retries,
1477 		   int argc, char **argv, char *combinedopt)
1478 {
1479 	union ctl_io *io;
1480 	struct scsi_read_capacity_data *data;
1481 	struct scsi_read_capacity_data_long *longdata;
1482 	int cdbsize = -1, retval;
1483 	uint8_t *dataptr;
1484 	int c;
1485 
1486 	cdbsize = 10;
1487 	dataptr = NULL;
1488 	retval = 0;
1489 
1490 	io = ctl_scsi_alloc_io(iid);
1491 	if (io == NULL) {
1492 		warn("%s: can't allocate memory\n", __func__);
1493 		return (1);
1494 	}
1495 
1496 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
1497 		switch (c) {
1498 		case 'c':
1499 			cdbsize = strtol(optarg, NULL, 0);
1500 			break;
1501 		default:
1502 			break;
1503 		}
1504 	}
1505 	if (cdbsize != -1) {
1506 		switch (cdbsize) {
1507 		case 10:
1508 		case 16:
1509 			break;
1510 		default:
1511 			warnx("%s: invalid cdbsize %d, valid sizes are 10 "
1512 			      "and 16", __func__, cdbsize);
1513 			retval = 1;
1514 			goto bailout;
1515 			break; /* NOTREACHED */
1516 		}
1517 	} else
1518 		cdbsize = 10;
1519 
1520 	dataptr = (uint8_t *)malloc(sizeof(*longdata));
1521 	if (dataptr == NULL) {
1522 		warn("%s: can't allocate %zd bytes\n", __func__,
1523 		     sizeof(*longdata));
1524 		retval = 1;
1525 		goto bailout;
1526 	}
1527 	memset(dataptr, 0, sizeof(*longdata));
1528 
1529 retry:
1530 
1531 	switch (cdbsize) {
1532 	case 10:
1533 		ctl_scsi_read_capacity(io,
1534 				       /*data_ptr*/ dataptr,
1535 				       /*data_len*/ sizeof(*longdata),
1536 				       /*addr*/ 0,
1537 				       /*reladr*/ 0,
1538 				       /*pmi*/ 0,
1539 				       /*tag_type*/ CTL_TAG_SIMPLE,
1540 				       /*control*/ 0);
1541 		break;
1542 	case 16:
1543 		ctl_scsi_read_capacity_16(io,
1544 					  /*data_ptr*/ dataptr,
1545 					  /*data_len*/ sizeof(*longdata),
1546 					  /*addr*/ 0,
1547 					  /*reladr*/ 0,
1548 					  /*pmi*/ 0,
1549 					  /*tag_type*/ CTL_TAG_SIMPLE,
1550 					  /*control*/ 0);
1551 		break;
1552 	}
1553 
1554 	io->io_hdr.nexus.initid = iid;
1555 	io->io_hdr.nexus.targ_lun = lun;
1556 
1557 	if (cctl_do_io(fd, retries, io, __func__) != 0) {
1558 		retval = 1;
1559 		goto bailout;
1560 	}
1561 
1562 	if ((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS) {
1563 		uint64_t maxlba;
1564 		uint32_t blocksize;
1565 
1566 		if (cdbsize == 10) {
1567 
1568 			data = (struct scsi_read_capacity_data *)dataptr;
1569 
1570 			maxlba = scsi_4btoul(data->addr);
1571 			blocksize = scsi_4btoul(data->length);
1572 
1573 			if (maxlba == 0xffffffff) {
1574 				cdbsize = 16;
1575 				goto retry;
1576 			}
1577 		} else {
1578 			longdata=(struct scsi_read_capacity_data_long *)dataptr;
1579 
1580 			maxlba = scsi_8btou64(longdata->addr);
1581 			blocksize = scsi_4btoul(longdata->length);
1582 		}
1583 
1584 		fprintf(stdout, "Disk Capacity: %ju, Blocksize: %d\n",
1585 			(uintmax_t)maxlba, blocksize);
1586 	} else {
1587 		ctl_io_error_print(io, NULL, stderr);
1588 	}
1589 bailout:
1590 	ctl_scsi_free_io(io);
1591 
1592 	if (dataptr != NULL)
1593 		free(dataptr);
1594 
1595 	return (retval);
1596 }
1597 
1598 static int
1599 cctl_read_write(int fd, int lun, int iid, int retries,
1600 		int argc, char **argv, char *combinedopt,
1601 		ctladm_cmdfunction command)
1602 {
1603 	union ctl_io *io;
1604 	int file_fd, do_stdio;
1605 	int cdbsize = -1, databytes;
1606 	uint8_t *dataptr;
1607 	char *filename = NULL;
1608 	int datalen = -1, blocksize = -1;
1609 	uint64_t lba = 0;
1610 	int lba_set = 0;
1611 	int retval;
1612 	int c;
1613 
1614 	retval = 0;
1615 	do_stdio = 0;
1616 	dataptr = NULL;
1617 	file_fd = -1;
1618 
1619 	io = ctl_scsi_alloc_io(iid);
1620 	if (io == NULL) {
1621 		warn("%s: can't allocate memory\n", __func__);
1622 		return (1);
1623 	}
1624 
1625 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
1626 		switch (c) {
1627 		case 'N':
1628 			io->io_hdr.flags |= CTL_FLAG_NO_DATAMOVE;
1629 			break;
1630 		case 'b':
1631 			blocksize = strtoul(optarg, NULL, 0);
1632 			break;
1633 		case 'c':
1634 			cdbsize = strtoul(optarg, NULL, 0);
1635 			break;
1636 		case 'd':
1637 			datalen = strtoul(optarg, NULL, 0);
1638 			break;
1639 		case 'f':
1640 			filename = strdup(optarg);
1641 			break;
1642 		case 'l':
1643 			lba = strtoull(optarg, NULL, 0);
1644 			lba_set = 1;
1645 			break;
1646 		default:
1647 			break;
1648 		}
1649 	}
1650 	if (filename == NULL) {
1651 		warnx("%s: you must supply a filename using -f", __func__);
1652 		retval = 1;
1653 		goto bailout;
1654 	}
1655 
1656 	if (datalen == -1) {
1657 		warnx("%s: you must specify the data length with -d", __func__);
1658 		retval = 1;
1659 		goto bailout;
1660 	}
1661 
1662 	if (lba_set == 0) {
1663 		warnx("%s: you must specify the LBA with -l", __func__);
1664 		retval = 1;
1665 		goto bailout;
1666 	}
1667 
1668 	if (blocksize == -1) {
1669 		warnx("%s: you must specify the blocksize with -b", __func__);
1670 		retval = 1;
1671 		goto bailout;
1672 	}
1673 
1674 	if (cdbsize != -1) {
1675 		switch (cdbsize) {
1676 		case 6:
1677 		case 10:
1678 		case 12:
1679 		case 16:
1680 			break;
1681 		default:
1682 			warnx("%s: invalid cdbsize %d, valid sizes are 6, "
1683 			      "10, 12 or 16", __func__, cdbsize);
1684 			retval = 1;
1685 			goto bailout;
1686 			break; /* NOTREACHED */
1687 		}
1688 	} else
1689 		cdbsize = 6;
1690 
1691 	databytes = datalen * blocksize;
1692 	dataptr = (uint8_t *)malloc(databytes);
1693 
1694 	if (dataptr == NULL) {
1695 		warn("%s: can't allocate %d bytes\n", __func__, databytes);
1696 		retval = 1;
1697 		goto bailout;
1698 	}
1699 	if (strcmp(filename, "-") == 0) {
1700 		if (command == CTLADM_CMD_READ)
1701 			file_fd = STDOUT_FILENO;
1702 		else
1703 			file_fd = STDIN_FILENO;
1704 		do_stdio = 1;
1705 	} else {
1706 		file_fd = open(filename, O_RDWR | O_CREAT, S_IRUSR | S_IWUSR);
1707 		if (file_fd == -1) {
1708 			warn("%s: can't open file %s", __func__, filename);
1709 			retval = 1;
1710 			goto bailout;
1711 		}
1712 	}
1713 
1714 	memset(dataptr, 0, databytes);
1715 
1716 	if (command == CTLADM_CMD_WRITE) {
1717 		int bytes_read;
1718 
1719 		bytes_read = read(file_fd, dataptr, databytes);
1720 		if (bytes_read == -1) {
1721 			warn("%s: error reading file %s", __func__, filename);
1722 			retval = 1;
1723 			goto bailout;
1724 		}
1725 		if (bytes_read != databytes) {
1726 			warnx("%s: only read %d bytes from file %s",
1727 			      __func__, bytes_read, filename);
1728 			retval = 1;
1729 			goto bailout;
1730 		}
1731 	}
1732 	ctl_scsi_read_write(io,
1733 			    /*data_ptr*/ dataptr,
1734 			    /*data_len*/ databytes,
1735 			    /*read_op*/ (command == CTLADM_CMD_READ) ? 1 : 0,
1736 			    /*byte2*/ 0,
1737 			    /*minimum_cdb_size*/ cdbsize,
1738 			    /*lba*/ lba,
1739 			    /*num_blocks*/ datalen,
1740 			    /*tag_type*/ CTL_TAG_SIMPLE,
1741 			    /*control*/ 0);
1742 
1743 	io->io_hdr.nexus.targ_lun = lun;
1744 	io->io_hdr.nexus.initid = iid;
1745 
1746 	if (cctl_do_io(fd, retries, io, __func__) != 0) {
1747 		retval = 1;
1748 		goto bailout;
1749 	}
1750 
1751 	if (((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS)
1752 	 && (command == CTLADM_CMD_READ)) {
1753 		int bytes_written;
1754 
1755 		bytes_written = write(file_fd, dataptr, databytes);
1756 		if (bytes_written == -1) {
1757 			warn("%s: can't write to %s", __func__, filename);
1758 			goto bailout;
1759 		}
1760 	} else if ((io->io_hdr.status & CTL_STATUS_MASK) != CTL_SUCCESS)
1761 		ctl_io_error_print(io, NULL, stderr);
1762 
1763 
1764 bailout:
1765 
1766 	ctl_scsi_free_io(io);
1767 
1768 	if (dataptr != NULL)
1769 		free(dataptr);
1770 
1771 	if ((do_stdio == 0)
1772 	 && (file_fd != -1))
1773 		close(file_fd);
1774 
1775 	return (retval);
1776 }
1777 
1778 static int
1779 cctl_get_luns(int fd, int lun, int iid, int retries, struct
1780 	      scsi_report_luns_data **lun_data, uint32_t *num_luns)
1781 {
1782 	union ctl_io *io;
1783 	uint32_t nluns;
1784 	int lun_datalen;
1785 	int retval;
1786 
1787 	retval = 0;
1788 
1789 	io = ctl_scsi_alloc_io(iid);
1790 	if (io == NULL) {
1791 		warnx("%s: can't allocate memory", __func__);
1792 		return (1);
1793 	}
1794 
1795 	/*
1796 	 * lun_data includes space for 1 lun, allocate space for 4 initially.
1797 	 * If that isn't enough, we'll allocate more.
1798 	 */
1799 	nluns = 4;
1800 retry:
1801 	lun_datalen = sizeof(*lun_data) +
1802 		(nluns * sizeof(struct scsi_report_luns_lundata));
1803 	*lun_data = malloc(lun_datalen);
1804 
1805 	if (*lun_data == NULL) {
1806 		warnx("%s: can't allocate memory", __func__);
1807 		ctl_scsi_free_io(io);
1808 		return (1);
1809 	}
1810 
1811 	ctl_scsi_report_luns(io,
1812 			     /*data_ptr*/ (uint8_t *)*lun_data,
1813 			     /*data_len*/ lun_datalen,
1814 			     /*select_report*/ RPL_REPORT_ALL,
1815 			     /*tag_type*/ CTL_TAG_SIMPLE,
1816 			     /*control*/ 0);
1817 
1818 	io->io_hdr.nexus.initid = iid;
1819 	io->io_hdr.nexus.targ_lun = lun;
1820 
1821 	if (cctl_do_io(fd, retries, io, __func__) != 0) {
1822 		retval = 1;
1823 		goto bailout;
1824 	}
1825 
1826 	if ((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS) {
1827 		uint32_t returned_len, returned_luns;
1828 
1829 		returned_len = scsi_4btoul((*lun_data)->length);
1830 		returned_luns = returned_len / 8;
1831 		if (returned_luns > nluns) {
1832 			nluns = returned_luns;
1833 			free(*lun_data);
1834 			goto retry;
1835 		}
1836 		/* These should be the same */
1837 		*num_luns = MIN(returned_luns, nluns);
1838 	} else {
1839 		ctl_io_error_print(io, NULL, stderr);
1840 		retval = 1;
1841 	}
1842 bailout:
1843 	ctl_scsi_free_io(io);
1844 
1845 	return (retval);
1846 }
1847 
1848 static int
1849 cctl_report_luns(int fd, int lun, int iid, int retries)
1850 {
1851 	struct scsi_report_luns_data *lun_data;
1852 	uint32_t num_luns, i;
1853 	int retval;
1854 
1855 	lun_data = NULL;
1856 
1857 	if ((retval = cctl_get_luns(fd, lun, iid, retries, &lun_data,
1858 				   &num_luns)) != 0)
1859 		goto bailout;
1860 
1861 	fprintf(stdout, "%u LUNs returned\n", num_luns);
1862 	for (i = 0; i < num_luns; i++) {
1863 		int lun_val;
1864 
1865 		/*
1866 		 * XXX KDM figure out a way to share this code with
1867 		 * cctl_lunlist()?
1868 		 */
1869 		switch (lun_data->luns[i].lundata[0] & RPL_LUNDATA_ATYP_MASK) {
1870 		case RPL_LUNDATA_ATYP_PERIPH:
1871 			lun_val = lun_data->luns[i].lundata[1];
1872 			break;
1873 		case RPL_LUNDATA_ATYP_FLAT:
1874 			lun_val = (lun_data->luns[i].lundata[0] &
1875 				RPL_LUNDATA_FLAT_LUN_MASK) |
1876 				(lun_data->luns[i].lundata[1] <<
1877 				RPL_LUNDATA_FLAT_LUN_BITS);
1878 			break;
1879 		case RPL_LUNDATA_ATYP_LUN:
1880 		case RPL_LUNDATA_ATYP_EXTLUN:
1881 		default:
1882 			fprintf(stdout, "Unsupported LUN format %d\n",
1883 				lun_data->luns[i].lundata[0] &
1884 				RPL_LUNDATA_ATYP_MASK);
1885 			lun_val = -1;
1886 			break;
1887 		}
1888 		if (lun_val == -1)
1889 			continue;
1890 
1891 		fprintf(stdout, "%d\n", lun_val);
1892 	}
1893 
1894 bailout:
1895 	if (lun_data != NULL)
1896 		free(lun_data);
1897 
1898 	return (retval);
1899 }
1900 
1901 static int
1902 cctl_tur(int fd, int lun, int iid, int retries)
1903 {
1904 	union ctl_io *io;
1905 
1906 	io = ctl_scsi_alloc_io(iid);
1907 	if (io == NULL) {
1908 		fprintf(stderr, "can't allocate memory\n");
1909 		return (1);
1910 	}
1911 
1912 	ctl_scsi_tur(io,
1913 		     /* tag_type */ CTL_TAG_SIMPLE,
1914 		     /* control */ 0);
1915 
1916 	io->io_hdr.nexus.targ_lun = lun;
1917 	io->io_hdr.nexus.initid = iid;
1918 
1919 	if (cctl_do_io(fd, retries, io, __func__) != 0) {
1920 		ctl_scsi_free_io(io);
1921 		return (1);
1922 	}
1923 
1924 	if ((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS)
1925 		fprintf(stdout, "Unit is ready\n");
1926 	else
1927 		ctl_io_error_print(io, NULL, stderr);
1928 
1929 	return (0);
1930 }
1931 
1932 static int
1933 cctl_get_inquiry(int fd, int lun, int iid, int retries,
1934 		 char *path_str, int path_len,
1935 		 struct scsi_inquiry_data *inq_data)
1936 {
1937 	union ctl_io *io;
1938 	int retval;
1939 
1940 	retval = 0;
1941 
1942 	io = ctl_scsi_alloc_io(iid);
1943 	if (io == NULL) {
1944 		warnx("cctl_inquiry: can't allocate memory\n");
1945 		return (1);
1946 	}
1947 
1948 	ctl_scsi_inquiry(/*io*/ io,
1949 			 /*data_ptr*/ (uint8_t *)inq_data,
1950 			 /*data_len*/ sizeof(*inq_data),
1951 			 /*byte2*/ 0,
1952 			 /*page_code*/ 0,
1953 			 /*tag_type*/ CTL_TAG_SIMPLE,
1954 			 /*control*/ 0);
1955 
1956 	io->io_hdr.nexus.targ_lun = lun;
1957 	io->io_hdr.nexus.initid = iid;
1958 
1959 	if (cctl_do_io(fd, retries, io, __func__) != 0) {
1960 		retval = 1;
1961 		goto bailout;
1962 	}
1963 
1964 	if ((io->io_hdr.status & CTL_STATUS_MASK) != CTL_SUCCESS) {
1965 		retval = 1;
1966 		ctl_io_error_print(io, NULL, stderr);
1967 	} else if (path_str != NULL)
1968 		ctl_scsi_path_string(io, path_str, path_len);
1969 
1970 bailout:
1971 	ctl_scsi_free_io(io);
1972 
1973 	return (retval);
1974 }
1975 
1976 static int
1977 cctl_inquiry(int fd, int lun, int iid, int retries)
1978 {
1979 	struct scsi_inquiry_data *inq_data;
1980 	char scsi_path[40];
1981 	int retval;
1982 
1983 	inq_data = malloc(sizeof(*inq_data));
1984 	if (inq_data == NULL) {
1985 		warnx("%s: can't allocate inquiry data", __func__);
1986 		retval = 1;
1987 		goto bailout;
1988 	}
1989 
1990 	if ((retval = cctl_get_inquiry(fd, lun, iid, retries, scsi_path,
1991 				       sizeof(scsi_path), inq_data)) != 0)
1992 		goto bailout;
1993 
1994 	printf("%s", scsi_path);
1995 	scsi_print_inquiry(inq_data);
1996 
1997 bailout:
1998 	if (inq_data != NULL)
1999 		free(inq_data);
2000 
2001 	return (retval);
2002 }
2003 
2004 static int
2005 cctl_req_sense(int fd, int lun, int iid, int retries)
2006 {
2007 	union ctl_io *io;
2008 	struct scsi_sense_data *sense_data;
2009 	int retval;
2010 
2011 	retval = 0;
2012 
2013 	io = ctl_scsi_alloc_io(iid);
2014 	if (io == NULL) {
2015 		warnx("cctl_req_sense: can't allocate memory\n");
2016 		return (1);
2017 	}
2018 	sense_data = malloc(sizeof(*sense_data));
2019 	memset(sense_data, 0, sizeof(*sense_data));
2020 
2021 	ctl_scsi_request_sense(/*io*/ io,
2022 			       /*data_ptr*/ (uint8_t *)sense_data,
2023 			       /*data_len*/ sizeof(*sense_data),
2024 			       /*byte2*/ 0,
2025 			       /*tag_type*/ CTL_TAG_SIMPLE,
2026 			       /*control*/ 0);
2027 
2028 	io->io_hdr.nexus.targ_lun = lun;
2029 	io->io_hdr.nexus.initid = iid;
2030 
2031 	if (cctl_do_io(fd, retries, io, __func__) != 0) {
2032 		retval = 1;
2033 		goto bailout;
2034 	}
2035 
2036 	if ((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS) {
2037 		bcopy(sense_data, &io->scsiio.sense_data, sizeof(*sense_data));
2038 		io->scsiio.sense_len = sizeof(*sense_data);
2039 		ctl_scsi_sense_print(&io->scsiio, NULL, stdout);
2040 	} else
2041 		ctl_io_error_print(io, NULL, stderr);
2042 
2043 bailout:
2044 
2045 	ctl_scsi_free_io(io);
2046 	free(sense_data);
2047 
2048 	return (retval);
2049 }
2050 
2051 static int
2052 cctl_report_target_port_group(int fd, int lun, int iid)
2053 {
2054 	union ctl_io *io;
2055 	uint32_t datalen;
2056 	uint8_t *dataptr;
2057 	int retval;
2058 
2059 	dataptr = NULL;
2060 	retval = 0;
2061 
2062 	io = ctl_scsi_alloc_io(iid);
2063 	if (io == NULL) {
2064 		warn("%s: can't allocate memory", __func__);
2065 		return (1);
2066 	}
2067 
2068 	datalen = 64;
2069 	dataptr = (uint8_t *)malloc(datalen);
2070 	if (dataptr == NULL) {
2071 		warn("%s: can't allocate %d bytes", __func__, datalen);
2072 		retval = 1;
2073 		goto bailout;
2074 	}
2075 
2076 	memset(dataptr, 0, datalen);
2077 
2078 	ctl_scsi_maintenance_in(/*io*/ io,
2079 				/*data_ptr*/ dataptr,
2080 				/*data_len*/ datalen,
2081 				/*action*/ SA_RPRT_TRGT_GRP,
2082 				/*tag_type*/ CTL_TAG_SIMPLE,
2083 				/*control*/ 0);
2084 
2085 	io->io_hdr.nexus.targ_lun = lun;
2086 	io->io_hdr.nexus.initid = iid;
2087 
2088 	if (cctl_do_io(fd, 0, io, __func__) != 0) {
2089 		retval = 1;
2090 		goto bailout;
2091 	}
2092 
2093 	if ((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS) {
2094 		int returned_len, used_len;
2095 
2096 		returned_len = scsi_4btoul(&dataptr[0]) + 4;
2097 
2098 		for (used_len = 0; used_len < returned_len; used_len++) {
2099 			fprintf(stdout, "0x%02x ", dataptr[used_len]);
2100 			if (((used_len+1) % 8) == 0)
2101 				fprintf(stdout, "\n");
2102 		}
2103 		fprintf(stdout, "\n");
2104 	} else
2105 		ctl_io_error_print(io, NULL, stderr);
2106 
2107 bailout:
2108 	ctl_scsi_free_io(io);
2109 
2110 	if (dataptr != NULL)
2111 		free(dataptr);
2112 
2113 	return (retval);
2114 }
2115 
2116 static int
2117 cctl_inquiry_vpd_devid(int fd, int lun, int iid)
2118 {
2119 	union ctl_io *io;
2120 	uint32_t datalen;
2121 	uint8_t *dataptr;
2122 	int retval;
2123 
2124 	retval = 0;
2125 	dataptr = NULL;
2126 
2127 	io = ctl_scsi_alloc_io(iid);
2128 	if (io == NULL) {
2129 		warn("%s: can't allocate memory", __func__);
2130 		return (1);
2131 	}
2132 
2133 	datalen = 256;
2134 	dataptr = (uint8_t *)malloc(datalen);
2135 	if (dataptr == NULL) {
2136 		warn("%s: can't allocate %d bytes", __func__, datalen);
2137 		retval = 1;
2138 		goto bailout;
2139 	}
2140 
2141 	memset(dataptr, 0, datalen);
2142 
2143 	ctl_scsi_inquiry(/*io*/        io,
2144 			 /*data_ptr*/  dataptr,
2145 			 /*data_len*/  datalen,
2146 			 /*byte2*/     SI_EVPD,
2147 			 /*page_code*/ SVPD_DEVICE_ID,
2148 			 /*tag_type*/  CTL_TAG_SIMPLE,
2149 			 /*control*/   0);
2150 
2151 	io->io_hdr.nexus.targ_lun = lun;
2152 	io->io_hdr.nexus.initid = iid;
2153 
2154 	if (cctl_do_io(fd, 0, io, __func__) != 0) {
2155 		retval = 1;
2156 		goto bailout;
2157 	}
2158 
2159 	if ((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS) {
2160 		int returned_len, used_len;
2161 
2162 		returned_len = scsi_2btoul(&dataptr[2]) + 4;
2163 
2164 		for (used_len = 0; used_len < returned_len; used_len++) {
2165 			fprintf(stdout, "0x%02x ", dataptr[used_len]);
2166 			if (((used_len+1) % 8) == 0)
2167 				fprintf(stdout, "\n");
2168 		}
2169 		fprintf(stdout, "\n");
2170 	} else
2171 		ctl_io_error_print(io, NULL, stderr);
2172 
2173 bailout:
2174 	ctl_scsi_free_io(io);
2175 
2176 	if (dataptr != NULL)
2177 		free(dataptr);
2178 
2179 	return (retval);
2180 }
2181 
2182 static int
2183 cctl_persistent_reserve_in(int fd, int lun, int iid,
2184                            int argc, char **argv, char *combinedopt,
2185 			   int retry_count)
2186 {
2187 	union ctl_io *io;
2188 	uint32_t datalen;
2189 	uint8_t *dataptr;
2190 	int action = -1;
2191 	int retval;
2192 	int c;
2193 
2194 	retval = 0;
2195 	dataptr = NULL;
2196 
2197 	io = ctl_scsi_alloc_io(iid);
2198 	if (io == NULL) {
2199 		warn("%s: can't allocate memory", __func__);
2200 		return (1);
2201 	}
2202 
2203 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
2204 		switch (c) {
2205 		case 'a':
2206 			action = strtol(optarg, NULL, 0);
2207 			break;
2208 		default:
2209 			break;
2210 		}
2211 	}
2212 
2213 	if (action < 0 || action > 2) {
2214 		warn("action must be specified and in the range: 0-2");
2215 		retval = 1;
2216 		goto bailout;
2217 	}
2218 
2219 
2220 	datalen = 256;
2221 	dataptr = (uint8_t *)malloc(datalen);
2222 	if (dataptr == NULL) {
2223 		warn("%s: can't allocate %d bytes", __func__, datalen);
2224 		retval = 1;
2225 		goto bailout;
2226 	}
2227 
2228 	memset(dataptr, 0, datalen);
2229 
2230 	ctl_scsi_persistent_res_in(io,
2231 				   /*data_ptr*/ dataptr,
2232 				   /*data_len*/ datalen,
2233 				   /*action*/   action,
2234 				   /*tag_type*/ CTL_TAG_SIMPLE,
2235 				   /*control*/  0);
2236 
2237 	io->io_hdr.nexus.targ_lun = lun;
2238 	io->io_hdr.nexus.initid = iid;
2239 
2240 	if (cctl_do_io(fd, retry_count, io, __func__) != 0) {
2241 		retval = 1;
2242 		goto bailout;
2243 	}
2244 
2245 	if ((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS) {
2246 		int returned_len, used_len;
2247 
2248 		switch (action) {
2249 		case 0:
2250 			returned_len = scsi_4btoul(&dataptr[4]) + 8;
2251 			returned_len = min(returned_len, 256);
2252 			break;
2253 		case 1:
2254 			returned_len = scsi_4btoul(&dataptr[4]) + 8;
2255 			break;
2256 		case 2:
2257 			returned_len = 8;
2258 			break;
2259 		default:
2260 			warnx("%s: invalid action %d", __func__, action);
2261 			goto bailout;
2262 			break; /* NOTREACHED */
2263 		}
2264 
2265 		for (used_len = 0; used_len < returned_len; used_len++) {
2266 			fprintf(stdout, "0x%02x ", dataptr[used_len]);
2267 			if (((used_len+1) % 8) == 0)
2268 				fprintf(stdout, "\n");
2269 		}
2270 		fprintf(stdout, "\n");
2271 	} else
2272 		ctl_io_error_print(io, NULL, stderr);
2273 
2274 bailout:
2275 	ctl_scsi_free_io(io);
2276 
2277 	if (dataptr != NULL)
2278 		free(dataptr);
2279 
2280 	return (retval);
2281 }
2282 
2283 static int
2284 cctl_persistent_reserve_out(int fd, int lun, int iid,
2285 			    int argc, char **argv, char *combinedopt,
2286 			    int retry_count)
2287 {
2288 	union ctl_io *io;
2289 	uint32_t datalen;
2290 	uint64_t key = 0, sa_key = 0;
2291 	int action = -1, restype = -1;
2292 	uint8_t *dataptr;
2293 	int retval;
2294 	int c;
2295 
2296 	retval = 0;
2297 	dataptr = NULL;
2298 
2299 	io = ctl_scsi_alloc_io(iid);
2300 	if (io == NULL) {
2301 		warn("%s: can't allocate memory", __func__);
2302 		return (1);
2303 	}
2304 
2305 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
2306 		switch (c) {
2307 		case 'a':
2308 			action = strtol(optarg, NULL, 0);
2309 			break;
2310 		case 'k':
2311 			key = strtoull(optarg, NULL, 0);
2312 			break;
2313 		case 'r':
2314 			restype = strtol(optarg, NULL, 0);
2315 			break;
2316 		case 's':
2317 			sa_key = strtoull(optarg, NULL, 0);
2318 			break;
2319 		default:
2320 			break;
2321 		}
2322 	}
2323 	if (action < 0 || action > 5) {
2324 		warn("action must be specified and in the range: 0-5");
2325 		retval = 1;
2326 		goto bailout;
2327 	}
2328 
2329 	if (restype < 0 || restype > 5) {
2330 		if (action != 0 && action != 5 && action != 3) {
2331 			warn("'restype' must specified and in the range: 0-5");
2332 			retval = 1;
2333 			goto bailout;
2334 		}
2335 	}
2336 
2337 	datalen = 24;
2338 	dataptr = (uint8_t *)malloc(datalen);
2339 	if (dataptr == NULL) {
2340 		warn("%s: can't allocate %d bytes", __func__, datalen);
2341 		retval = 1;
2342 		goto bailout;
2343 	}
2344 
2345 	memset(dataptr, 0, datalen);
2346 
2347 	ctl_scsi_persistent_res_out(io,
2348 				    /*data_ptr*/ dataptr,
2349 				    /*data_len*/ datalen,
2350 				    /*action*/   action,
2351 				    /*type*/     restype,
2352 				    /*key*/      key,
2353 				    /*sa key*/   sa_key,
2354 				    /*tag_type*/ CTL_TAG_SIMPLE,
2355 				    /*control*/  0);
2356 
2357 	io->io_hdr.nexus.targ_lun = lun;
2358 	io->io_hdr.nexus.initid = iid;
2359 
2360 	if (cctl_do_io(fd, retry_count, io, __func__) != 0) {
2361 		retval = 1;
2362 		goto bailout;
2363 	}
2364 	if ((io->io_hdr.status & CTL_STATUS_MASK) == CTL_SUCCESS) {
2365 		char scsi_path[40];
2366 		ctl_scsi_path_string(io, scsi_path, sizeof(scsi_path));
2367 		fprintf( stdout, "%sPERSISTENT RESERVE OUT executed "
2368 			"successfully\n", scsi_path);
2369 	} else
2370 		ctl_io_error_print(io, NULL, stderr);
2371 
2372 bailout:
2373 	ctl_scsi_free_io(io);
2374 
2375 	if (dataptr != NULL)
2376 		free(dataptr);
2377 
2378 	return (retval);
2379 }
2380 
2381 static int
2382 cctl_create_lun(int fd, int argc, char **argv, char *combinedopt)
2383 {
2384 	struct ctl_lun_req req;
2385 	int device_type = -1;
2386 	uint64_t lun_size = 0;
2387 	uint32_t blocksize = 0, req_lun_id = 0;
2388 	char *serial_num = NULL;
2389 	char *device_id = NULL;
2390 	int lun_size_set = 0, blocksize_set = 0, lun_id_set = 0;
2391 	char *backend_name = NULL;
2392 	nvlist_t *option_list;
2393 	int retval = 0, c;
2394 
2395 	option_list = nvlist_create(0);
2396 	if (option_list == NULL)
2397 		err(1, "%s: unable to allocate nvlist", __func__);
2398 
2399 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
2400 		switch (c) {
2401 		case 'b':
2402 			backend_name = strdup(optarg);
2403 			break;
2404 		case 'B':
2405 			blocksize = strtoul(optarg, NULL, 0);
2406 			blocksize_set = 1;
2407 			break;
2408 		case 'd':
2409 			device_id = strdup(optarg);
2410 			break;
2411 		case 'l':
2412 			req_lun_id = strtoul(optarg, NULL, 0);
2413 			lun_id_set = 1;
2414 			break;
2415 		case 'o': {
2416 			char *tmpstr;
2417 			char *name, *value;
2418 
2419 			tmpstr = strdup(optarg);
2420 			name = strsep(&tmpstr, "=");
2421 			if (name == NULL) {
2422 				warnx("%s: option -o takes \"name=value\""
2423 				      "argument", __func__);
2424 				retval = 1;
2425 				goto bailout;
2426 			}
2427 			value = strsep(&tmpstr, "=");
2428 			if (value == NULL) {
2429 				warnx("%s: option -o takes \"name=value\""
2430 				      "argument", __func__);
2431 				retval = 1;
2432 				goto bailout;
2433 			}
2434 			free(tmpstr);
2435 			nvlist_add_string(option_list, name, value);
2436 			break;
2437 		}
2438 		case 's':
2439 			if (strcasecmp(optarg, "auto") != 0) {
2440 				retval = expand_number(optarg, &lun_size);
2441 				if (retval != 0) {
2442 					warn("%s: invalid -s argument",
2443 					    __func__);
2444 					retval = 1;
2445 					goto bailout;
2446 				}
2447 			}
2448 			lun_size_set = 1;
2449 			break;
2450 		case 'S':
2451 			serial_num = strdup(optarg);
2452 			break;
2453 		case 't':
2454 			device_type = strtoul(optarg, NULL, 0);
2455 			break;
2456 		default:
2457 			break;
2458 		}
2459 	}
2460 
2461 	if (backend_name == NULL) {
2462 		warnx("%s: backend name (-b) must be specified", __func__);
2463 		retval = 1;
2464 		goto bailout;
2465 	}
2466 
2467 	bzero(&req, sizeof(req));
2468 
2469 	strlcpy(req.backend, backend_name, sizeof(req.backend));
2470 	req.reqtype = CTL_LUNREQ_CREATE;
2471 
2472 	if (blocksize_set != 0)
2473 		req.reqdata.create.blocksize_bytes = blocksize;
2474 
2475 	if (lun_size_set != 0)
2476 		req.reqdata.create.lun_size_bytes = lun_size;
2477 
2478 	if (lun_id_set != 0) {
2479 		req.reqdata.create.flags |= CTL_LUN_FLAG_ID_REQ;
2480 		req.reqdata.create.req_lun_id = req_lun_id;
2481 	}
2482 
2483 	req.reqdata.create.flags |= CTL_LUN_FLAG_DEV_TYPE;
2484 
2485 	if (device_type != -1)
2486 		req.reqdata.create.device_type = device_type;
2487 	else
2488 		req.reqdata.create.device_type = T_DIRECT;
2489 
2490 	if (serial_num != NULL) {
2491 		strlcpy(req.reqdata.create.serial_num, serial_num,
2492 			sizeof(req.reqdata.create.serial_num));
2493 		req.reqdata.create.flags |= CTL_LUN_FLAG_SERIAL_NUM;
2494 	}
2495 
2496 	if (device_id != NULL) {
2497 		strlcpy(req.reqdata.create.device_id, device_id,
2498 			sizeof(req.reqdata.create.device_id));
2499 		req.reqdata.create.flags |= CTL_LUN_FLAG_DEVID;
2500 	}
2501 
2502 	req.args = nvlist_pack(option_list, &req.args_len);
2503 	if (req.args == NULL) {
2504 		warn("%s: error packing nvlist", __func__);
2505 		retval = 1;
2506 		goto bailout;
2507 	}
2508 
2509 	retval = ioctl(fd, CTL_LUN_REQ, &req);
2510 	free(req.args);
2511 	if (retval == -1) {
2512 		warn("%s: error issuing CTL_LUN_REQ ioctl", __func__);
2513 		retval = 1;
2514 		goto bailout;
2515 	}
2516 
2517 	switch (req.status) {
2518 	case CTL_LUN_ERROR:
2519 		warnx("LUN creation error: %s", req.error_str);
2520 		retval = 1;
2521 		goto bailout;
2522 	case CTL_LUN_WARNING:
2523 		warnx("LUN creation warning: %s", req.error_str);
2524 		break;
2525 	case CTL_LUN_OK:
2526 		break;
2527 	default:
2528 		warnx("unknown LUN creation status: %d", req.status);
2529 		retval = 1;
2530 		goto bailout;
2531 	}
2532 
2533 	fprintf(stdout, "LUN created successfully\n");
2534 	fprintf(stdout, "backend:       %s\n", req.backend);
2535 	fprintf(stdout, "device type:   %d\n",req.reqdata.create.device_type);
2536 	fprintf(stdout, "LUN size:      %ju bytes\n",
2537 		(uintmax_t)req.reqdata.create.lun_size_bytes);
2538 	fprintf(stdout, "blocksize      %u bytes\n",
2539 		req.reqdata.create.blocksize_bytes);
2540 	fprintf(stdout, "LUN ID:        %d\n", req.reqdata.create.req_lun_id);
2541 	fprintf(stdout, "Serial Number: %s\n", req.reqdata.create.serial_num);
2542 	fprintf(stdout, "Device ID:     %s\n", req.reqdata.create.device_id);
2543 
2544 bailout:
2545 	nvlist_destroy(option_list);
2546 	return (retval);
2547 }
2548 
2549 static int
2550 cctl_rm_lun(int fd, int argc, char **argv, char *combinedopt)
2551 {
2552 	struct ctl_lun_req req;
2553 	uint32_t lun_id = 0;
2554 	int lun_id_set = 0;
2555 	char *backend_name = NULL;
2556 	nvlist_t *option_list;
2557 	int retval = 0, c;
2558 
2559 	option_list = nvlist_create(0);
2560 	if (option_list == NULL)
2561 		err(1, "%s: unable to allocate nvlist", __func__);
2562 
2563 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
2564 		switch (c) {
2565 		case 'b':
2566 			backend_name = strdup(optarg);
2567 			break;
2568 		case 'l':
2569 			lun_id = strtoul(optarg, NULL, 0);
2570 			lun_id_set = 1;
2571 			break;
2572 		case 'o': {
2573 			char *tmpstr;
2574 			char *name, *value;
2575 
2576 			tmpstr = strdup(optarg);
2577 			name = strsep(&tmpstr, "=");
2578 			if (name == NULL) {
2579 				warnx("%s: option -o takes \"name=value\""
2580 				      "argument", __func__);
2581 				retval = 1;
2582 				goto bailout;
2583 			}
2584 			value = strsep(&tmpstr, "=");
2585 			if (value == NULL) {
2586 				warnx("%s: option -o takes \"name=value\""
2587 				      "argument", __func__);
2588 				retval = 1;
2589 				goto bailout;
2590 			}
2591 			free(tmpstr);
2592 			nvlist_add_string(option_list, name, value);
2593 			break;
2594 		}
2595 		default:
2596 			break;
2597 		}
2598 	}
2599 
2600 	if (backend_name == NULL)
2601 		errx(1, "%s: backend name (-b) must be specified", __func__);
2602 
2603 	if (lun_id_set == 0)
2604 		errx(1, "%s: LUN id (-l) must be specified", __func__);
2605 
2606 	bzero(&req, sizeof(req));
2607 
2608 	strlcpy(req.backend, backend_name, sizeof(req.backend));
2609 	req.reqtype = CTL_LUNREQ_RM;
2610 
2611 	req.reqdata.rm.lun_id = lun_id;
2612 
2613 	req.args = nvlist_pack(option_list, &req.args_len);
2614 	if (req.args == NULL) {
2615 		warn("%s: error packing nvlist", __func__);
2616 		retval = 1;
2617 		goto bailout;
2618 	}
2619 
2620 	retval = ioctl(fd, CTL_LUN_REQ, &req);
2621 	free(req.args);
2622 	if (retval == -1) {
2623 		warn("%s: error issuing CTL_LUN_REQ ioctl", __func__);
2624 		retval = 1;
2625 		goto bailout;
2626 	}
2627 
2628 	switch (req.status) {
2629 	case CTL_LUN_ERROR:
2630 		warnx("LUN removal error: %s", req.error_str);
2631 		retval = 1;
2632 		goto bailout;
2633 	case CTL_LUN_WARNING:
2634 		warnx("LUN removal warning: %s", req.error_str);
2635 		break;
2636 	case CTL_LUN_OK:
2637 		break;
2638 	default:
2639 		warnx("unknown LUN removal status: %d", req.status);
2640 		retval = 1;
2641 		goto bailout;
2642 	}
2643 
2644 	printf("LUN %d removed successfully\n", lun_id);
2645 
2646 bailout:
2647 	nvlist_destroy(option_list);
2648 	return (retval);
2649 }
2650 
2651 static int
2652 cctl_modify_lun(int fd, int argc, char **argv, char *combinedopt)
2653 {
2654 	struct ctl_lun_req req;
2655 	uint64_t lun_size = 0;
2656 	uint32_t lun_id = 0;
2657 	int lun_id_set = 0, lun_size_set = 0;
2658 	char *backend_name = NULL;
2659 	nvlist_t *option_list;
2660 	int retval = 0, c;
2661 
2662 	option_list = nvlist_create(0);
2663 	if (option_list == NULL)
2664 		err(1, "%s: unable to allocate nvlist", __func__);
2665 
2666 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
2667 		switch (c) {
2668 		case 'b':
2669 			backend_name = strdup(optarg);
2670 			break;
2671 		case 'l':
2672 			lun_id = strtoul(optarg, NULL, 0);
2673 			lun_id_set = 1;
2674 			break;
2675 		case 'o': {
2676 			char *tmpstr;
2677 			char *name, *value;
2678 
2679 			tmpstr = strdup(optarg);
2680 			name = strsep(&tmpstr, "=");
2681 			if (name == NULL) {
2682 				warnx("%s: option -o takes \"name=value\""
2683 				      "argument", __func__);
2684 				retval = 1;
2685 				goto bailout;
2686 			}
2687 			value = strsep(&tmpstr, "=");
2688 			if (value == NULL) {
2689 				warnx("%s: option -o takes \"name=value\""
2690 				      "argument", __func__);
2691 				retval = 1;
2692 				goto bailout;
2693 			}
2694 			free(tmpstr);
2695 			nvlist_add_string(option_list, name, value);
2696 			break;
2697 		}
2698 		case 's':
2699 			if (strcasecmp(optarg, "auto") != 0) {
2700 				retval = expand_number(optarg, &lun_size);
2701 				if (retval != 0) {
2702 					warn("%s: invalid -s argument",
2703 					    __func__);
2704 					retval = 1;
2705 					goto bailout;
2706 				}
2707 			}
2708 			lun_size_set = 1;
2709 			break;
2710 		default:
2711 			break;
2712 		}
2713 	}
2714 
2715 	if (backend_name == NULL)
2716 		errx(1, "%s: backend name (-b) must be specified", __func__);
2717 
2718 	if (lun_id_set == 0)
2719 		errx(1, "%s: LUN id (-l) must be specified", __func__);
2720 
2721 	if (lun_size_set == 0 && nvlist_empty(option_list))
2722 		errx(1, "%s: size (-s) or options (-o) must be specified",
2723 		    __func__);
2724 
2725 	bzero(&req, sizeof(req));
2726 
2727 	strlcpy(req.backend, backend_name, sizeof(req.backend));
2728 	req.reqtype = CTL_LUNREQ_MODIFY;
2729 
2730 	req.reqdata.modify.lun_id = lun_id;
2731 	req.reqdata.modify.lun_size_bytes = lun_size;
2732 
2733 	req.args = nvlist_pack(option_list, &req.args_len);
2734 	if (req.args == NULL) {
2735 		warn("%s: error packing nvlist", __func__);
2736 		retval = 1;
2737 		goto bailout;
2738 	}
2739 
2740 	retval = ioctl(fd, CTL_LUN_REQ, &req);
2741 	free(req.args);
2742 	if (retval == -1) {
2743 		warn("%s: error issuing CTL_LUN_REQ ioctl", __func__);
2744 		retval = 1;
2745 		goto bailout;
2746 	}
2747 
2748 	switch (req.status) {
2749 	case CTL_LUN_ERROR:
2750 		warnx("LUN modification error: %s", req.error_str);
2751 		retval = 1;
2752 		goto bailout;
2753 	case CTL_LUN_WARNING:
2754 		warnx("LUN modification warning: %s", req.error_str);
2755 		break;
2756 	case CTL_LUN_OK:
2757 		break;
2758 	default:
2759 		warnx("unknown LUN modification status: %d", req.status);
2760 		retval = 1;
2761 		goto bailout;
2762 	}
2763 
2764 	printf("LUN %d modified successfully\n", lun_id);
2765 
2766 bailout:
2767 	nvlist_destroy(option_list);
2768 	return (retval);
2769 }
2770 
2771 struct cctl_islist_conn {
2772 	int connection_id;
2773 	char *initiator;
2774 	char *initiator_addr;
2775 	char *initiator_alias;
2776 	char *target;
2777 	char *target_alias;
2778 	char *header_digest;
2779 	char *data_digest;
2780 	char *max_recv_data_segment_length;
2781 	char *max_send_data_segment_length;
2782 	char *max_burst_length;
2783 	char *first_burst_length;
2784 	char *offload;
2785 	int immediate_data;
2786 	int iser;
2787 	STAILQ_ENTRY(cctl_islist_conn) links;
2788 };
2789 
2790 struct cctl_islist_data {
2791 	int num_conns;
2792 	STAILQ_HEAD(,cctl_islist_conn) conn_list;
2793 	struct cctl_islist_conn *cur_conn;
2794 	int level;
2795 	struct sbuf *cur_sb[32];
2796 };
2797 
2798 static void
2799 cctl_islist_start_element(void *user_data, const char *name, const char **attr)
2800 {
2801 	int i;
2802 	struct cctl_islist_data *islist;
2803 	struct cctl_islist_conn *cur_conn;
2804 
2805 	islist = (struct cctl_islist_data *)user_data;
2806 	cur_conn = islist->cur_conn;
2807 	islist->level++;
2808 	if ((u_int)islist->level >= (sizeof(islist->cur_sb) /
2809 	    sizeof(islist->cur_sb[0])))
2810 		errx(1, "%s: too many nesting levels, %zd max", __func__,
2811 		     sizeof(islist->cur_sb) / sizeof(islist->cur_sb[0]));
2812 
2813 	islist->cur_sb[islist->level] = sbuf_new_auto();
2814 	if (islist->cur_sb[islist->level] == NULL)
2815 		err(1, "%s: Unable to allocate sbuf", __func__);
2816 
2817 	if (strcmp(name, "connection") == 0) {
2818 		if (cur_conn != NULL)
2819 			errx(1, "%s: improper connection element nesting",
2820 			    __func__);
2821 
2822 		cur_conn = calloc(1, sizeof(*cur_conn));
2823 		if (cur_conn == NULL)
2824 			err(1, "%s: cannot allocate %zd bytes", __func__,
2825 			    sizeof(*cur_conn));
2826 
2827 		islist->num_conns++;
2828 		islist->cur_conn = cur_conn;
2829 
2830 		STAILQ_INSERT_TAIL(&islist->conn_list, cur_conn, links);
2831 
2832 		for (i = 0; attr[i] != NULL; i += 2) {
2833 			if (strcmp(attr[i], "id") == 0) {
2834 				cur_conn->connection_id =
2835 				    strtoull(attr[i+1], NULL, 0);
2836 			} else {
2837 				errx(1,
2838 				    "%s: invalid connection attribute %s = %s",
2839 				     __func__, attr[i], attr[i+1]);
2840 			}
2841 		}
2842 	}
2843 }
2844 
2845 static void
2846 cctl_islist_end_element(void *user_data, const char *name)
2847 {
2848 	struct cctl_islist_data *islist;
2849 	struct cctl_islist_conn *cur_conn;
2850 	char *str;
2851 
2852 	islist = (struct cctl_islist_data *)user_data;
2853 	cur_conn = islist->cur_conn;
2854 
2855 	if ((cur_conn == NULL)
2856 	 && (strcmp(name, "ctlislist") != 0))
2857 		errx(1, "%s: cur_conn == NULL! (name = %s)", __func__, name);
2858 
2859 	if (islist->cur_sb[islist->level] == NULL)
2860 		errx(1, "%s: no valid sbuf at level %d (name %s)", __func__,
2861 		     islist->level, name);
2862 
2863 	sbuf_finish(islist->cur_sb[islist->level]);
2864 	str = strdup(sbuf_data(islist->cur_sb[islist->level]));
2865 	if (str == NULL)
2866 		err(1, "%s can't allocate %zd bytes for string", __func__,
2867 		    sbuf_len(islist->cur_sb[islist->level]));
2868 
2869 	sbuf_delete(islist->cur_sb[islist->level]);
2870 	islist->cur_sb[islist->level] = NULL;
2871 	islist->level--;
2872 
2873 	if (strcmp(name, "initiator") == 0) {
2874 		cur_conn->initiator = str;
2875 		str = NULL;
2876 	} else if (strcmp(name, "initiator_addr") == 0) {
2877 		cur_conn->initiator_addr = str;
2878 		str = NULL;
2879 	} else if (strcmp(name, "initiator_alias") == 0) {
2880 		cur_conn->initiator_alias = str;
2881 		str = NULL;
2882 	} else if (strcmp(name, "target") == 0) {
2883 		cur_conn->target = str;
2884 		str = NULL;
2885 	} else if (strcmp(name, "target_alias") == 0) {
2886 		cur_conn->target_alias = str;
2887 		str = NULL;
2888 	} else if (strcmp(name, "target_portal_group_tag") == 0) {
2889 	} else if (strcmp(name, "header_digest") == 0) {
2890 		cur_conn->header_digest = str;
2891 		str = NULL;
2892 	} else if (strcmp(name, "data_digest") == 0) {
2893 		cur_conn->data_digest = str;
2894 		str = NULL;
2895 	} else if (strcmp(name, "max_recv_data_segment_length") == 0) {
2896 		cur_conn->max_recv_data_segment_length = str;
2897 		str = NULL;
2898 	} else if (strcmp(name, "max_send_data_segment_length") == 0) {
2899 		cur_conn->max_send_data_segment_length = str;
2900 		str = NULL;
2901 	} else if (strcmp(name, "max_burst_length") == 0) {
2902 		cur_conn->max_burst_length = str;
2903 		str = NULL;
2904 	} else if (strcmp(name, "first_burst_length") == 0) {
2905 		cur_conn->first_burst_length = str;
2906 		str = NULL;
2907 	} else if (strcmp(name, "offload") == 0) {
2908 		cur_conn->offload = str;
2909 		str = NULL;
2910 	} else if (strcmp(name, "immediate_data") == 0) {
2911 		cur_conn->immediate_data = atoi(str);
2912 	} else if (strcmp(name, "iser") == 0) {
2913 		cur_conn->iser = atoi(str);
2914 	} else if (strcmp(name, "connection") == 0) {
2915 		islist->cur_conn = NULL;
2916 	} else if (strcmp(name, "ctlislist") == 0) {
2917 		/* Nothing. */
2918 	} else {
2919 		/*
2920 		 * Unknown element; ignore it for forward compatibility.
2921 		 */
2922 	}
2923 
2924 	free(str);
2925 }
2926 
2927 static void
2928 cctl_islist_char_handler(void *user_data, const XML_Char *str, int len)
2929 {
2930 	struct cctl_islist_data *islist;
2931 
2932 	islist = (struct cctl_islist_data *)user_data;
2933 
2934 	sbuf_bcat(islist->cur_sb[islist->level], str, len);
2935 }
2936 
2937 static int
2938 cctl_islist(int fd, int argc, char **argv, char *combinedopt)
2939 {
2940 	struct ctl_iscsi req;
2941 	struct cctl_islist_data islist;
2942 	struct cctl_islist_conn *conn;
2943 	XML_Parser parser;
2944 	char *conn_str;
2945 	int conn_len;
2946 	int dump_xml = 0;
2947 	int c, retval, verbose = 0;
2948 
2949 	retval = 0;
2950 	conn_len = 4096;
2951 
2952 	bzero(&islist, sizeof(islist));
2953 	STAILQ_INIT(&islist.conn_list);
2954 
2955 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
2956 		switch (c) {
2957 		case 'v':
2958 			verbose = 1;
2959 			break;
2960 		case 'x':
2961 			dump_xml = 1;
2962 			break;
2963 		default:
2964 			break;
2965 		}
2966 	}
2967 
2968 retry:
2969 	conn_str = malloc(conn_len);
2970 
2971 	bzero(&req, sizeof(req));
2972 	req.type = CTL_ISCSI_LIST;
2973 	req.data.list.alloc_len = conn_len;
2974 	req.data.list.conn_xml = conn_str;
2975 
2976 	if (ioctl(fd, CTL_ISCSI, &req) == -1) {
2977 		warn("%s: error issuing CTL_ISCSI ioctl", __func__);
2978 		retval = 1;
2979 		goto bailout;
2980 	}
2981 
2982 	if (req.status == CTL_ISCSI_ERROR) {
2983 		warnx("%s: error returned from CTL_ISCSI ioctl:\n%s",
2984 		      __func__, req.error_str);
2985 	} else if (req.status == CTL_ISCSI_LIST_NEED_MORE_SPACE) {
2986 		conn_len = conn_len << 1;
2987 		goto retry;
2988 	}
2989 
2990 	if (dump_xml != 0) {
2991 		printf("%s", conn_str);
2992 		goto bailout;
2993 	}
2994 
2995 	parser = XML_ParserCreate(NULL);
2996 	if (parser == NULL) {
2997 		warn("%s: Unable to create XML parser", __func__);
2998 		retval = 1;
2999 		goto bailout;
3000 	}
3001 
3002 	XML_SetUserData(parser, &islist);
3003 	XML_SetElementHandler(parser, cctl_islist_start_element,
3004 	    cctl_islist_end_element);
3005 	XML_SetCharacterDataHandler(parser, cctl_islist_char_handler);
3006 
3007 	retval = XML_Parse(parser, conn_str, strlen(conn_str), 1);
3008 	if (retval != 1) {
3009 		warnx("%s: Unable to parse XML: Error %d", __func__,
3010 		    XML_GetErrorCode(parser));
3011 		XML_ParserFree(parser);
3012 		retval = 1;
3013 		goto bailout;
3014 	}
3015 	retval = 0;
3016 	XML_ParserFree(parser);
3017 
3018 	if (verbose != 0) {
3019 		STAILQ_FOREACH(conn, &islist.conn_list, links) {
3020 			printf("%-25s %d\n", "Session ID:", conn->connection_id);
3021 			printf("%-25s %s\n", "Initiator name:", conn->initiator);
3022 			printf("%-25s %s\n", "Initiator portal:", conn->initiator_addr);
3023 			printf("%-25s %s\n", "Initiator alias:", conn->initiator_alias);
3024 			printf("%-25s %s\n", "Target name:", conn->target);
3025 			printf("%-25s %s\n", "Target alias:", conn->target_alias);
3026 			printf("%-25s %s\n", "Header digest:", conn->header_digest);
3027 			printf("%-25s %s\n", "Data digest:", conn->data_digest);
3028 			printf("%-25s %s\n", "MaxRecvDataSegmentLength:", conn->max_recv_data_segment_length);
3029 			printf("%-25s %s\n", "MaxSendDataSegmentLength:", conn->max_send_data_segment_length);
3030 			printf("%-25s %s\n", "MaxBurstLen:", conn->max_burst_length);
3031 			printf("%-25s %s\n", "FirstBurstLen:", conn->first_burst_length);
3032 			printf("%-25s %s\n", "ImmediateData:", conn->immediate_data ? "Yes" : "No");
3033 			printf("%-25s %s\n", "iSER (RDMA):", conn->iser ? "Yes" : "No");
3034 			printf("%-25s %s\n", "Offload driver:", conn->offload);
3035 			printf("\n");
3036 		}
3037 	} else {
3038 		printf("%4s %-16s %-36s %-36s\n", "ID", "Portal", "Initiator name",
3039 		    "Target name");
3040 		STAILQ_FOREACH(conn, &islist.conn_list, links) {
3041 			printf("%4u %-16s %-36s %-36s\n",
3042 			    conn->connection_id, conn->initiator_addr, conn->initiator,
3043 			    conn->target);
3044 		}
3045 	}
3046 bailout:
3047 	free(conn_str);
3048 
3049 	return (retval);
3050 }
3051 
3052 static int
3053 cctl_islogout(int fd, int argc, char **argv, char *combinedopt)
3054 {
3055 	struct ctl_iscsi req;
3056 	int retval = 0, c;
3057 	int all = 0, connection_id = -1, nargs = 0;
3058 	char *initiator_name = NULL, *initiator_addr = NULL;
3059 
3060 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
3061 		switch (c) {
3062 		case 'a':
3063 			all = 1;
3064 			nargs++;
3065 			break;
3066 		case 'c':
3067 			connection_id = strtoul(optarg, NULL, 0);
3068 			nargs++;
3069 			break;
3070 		case 'i':
3071 			initiator_name = strdup(optarg);
3072 			if (initiator_name == NULL)
3073 				err(1, "%s: strdup", __func__);
3074 			nargs++;
3075 			break;
3076 		case 'p':
3077 			initiator_addr = strdup(optarg);
3078 			if (initiator_addr == NULL)
3079 				err(1, "%s: strdup", __func__);
3080 			nargs++;
3081 			break;
3082 		default:
3083 			break;
3084 		}
3085 	}
3086 
3087 	if (nargs == 0)
3088 		errx(1, "%s: either -a, -c, -i, or -p must be specified",
3089 		    __func__);
3090 	if (nargs > 1)
3091 		errx(1, "%s: only one of -a, -c, -i, or -p may be specified",
3092 		    __func__);
3093 
3094 	bzero(&req, sizeof(req));
3095 	req.type = CTL_ISCSI_LOGOUT;
3096 	req.data.logout.connection_id = connection_id;
3097 	if (initiator_addr != NULL)
3098 		strlcpy(req.data.logout.initiator_addr,
3099 		    initiator_addr, sizeof(req.data.logout.initiator_addr));
3100 	if (initiator_name != NULL)
3101 		strlcpy(req.data.logout.initiator_name,
3102 		    initiator_name, sizeof(req.data.logout.initiator_name));
3103 	if (all != 0)
3104 		req.data.logout.all = 1;
3105 
3106 	if (ioctl(fd, CTL_ISCSI, &req) == -1) {
3107 		warn("%s: error issuing CTL_ISCSI ioctl", __func__);
3108 		retval = 1;
3109 		goto bailout;
3110 	}
3111 
3112 	if (req.status != CTL_ISCSI_OK) {
3113 		warnx("%s: error returned from CTL iSCSI logout request:\n%s",
3114 		      __func__, req.error_str);
3115 		retval = 1;
3116 		goto bailout;
3117 	}
3118 
3119 	printf("iSCSI logout requests submitted\n");
3120 
3121 bailout:
3122 	return (retval);
3123 }
3124 
3125 static int
3126 cctl_isterminate(int fd, int argc, char **argv, char *combinedopt)
3127 {
3128 	struct ctl_iscsi req;
3129 	int retval = 0, c;
3130 	int all = 0, connection_id = -1, nargs = 0;
3131 	char *initiator_name = NULL, *initiator_addr = NULL;
3132 
3133 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
3134 		switch (c) {
3135 		case 'a':
3136 			all = 1;
3137 			nargs++;
3138 			break;
3139 		case 'c':
3140 			connection_id = strtoul(optarg, NULL, 0);
3141 			nargs++;
3142 			break;
3143 		case 'i':
3144 			initiator_name = strdup(optarg);
3145 			if (initiator_name == NULL)
3146 				err(1, "%s: strdup", __func__);
3147 			nargs++;
3148 			break;
3149 		case 'p':
3150 			initiator_addr = strdup(optarg);
3151 			if (initiator_addr == NULL)
3152 				err(1, "%s: strdup", __func__);
3153 			nargs++;
3154 			break;
3155 		default:
3156 			break;
3157 		}
3158 	}
3159 
3160 	if (nargs == 0)
3161 		errx(1, "%s: either -a, -c, -i, or -p must be specified",
3162 		    __func__);
3163 	if (nargs > 1)
3164 		errx(1, "%s: only one of -a, -c, -i, or -p may be specified",
3165 		    __func__);
3166 
3167 	bzero(&req, sizeof(req));
3168 	req.type = CTL_ISCSI_TERMINATE;
3169 	req.data.terminate.connection_id = connection_id;
3170 	if (initiator_addr != NULL)
3171 		strlcpy(req.data.terminate.initiator_addr,
3172 		    initiator_addr, sizeof(req.data.terminate.initiator_addr));
3173 	if (initiator_name != NULL)
3174 		strlcpy(req.data.terminate.initiator_name,
3175 		    initiator_name, sizeof(req.data.terminate.initiator_name));
3176 	if (all != 0)
3177 		req.data.terminate.all = 1;
3178 
3179 	if (ioctl(fd, CTL_ISCSI, &req) == -1) {
3180 		warn("%s: error issuing CTL_ISCSI ioctl", __func__);
3181 		retval = 1;
3182 		goto bailout;
3183 	}
3184 
3185 	if (req.status != CTL_ISCSI_OK) {
3186 		warnx("%s: error returned from CTL iSCSI connection "
3187 		    "termination request:\n%s", __func__, req.error_str);
3188 		retval = 1;
3189 		goto bailout;
3190 	}
3191 
3192 	printf("iSCSI connections terminated\n");
3193 
3194 bailout:
3195 	return (retval);
3196 }
3197 
3198 /*
3199  * Name/value pair used for per-LUN attributes.
3200  */
3201 struct cctl_lun_nv {
3202 	char *name;
3203 	char *value;
3204 	STAILQ_ENTRY(cctl_lun_nv) links;
3205 };
3206 
3207 /*
3208  * Backend LUN information.
3209  */
3210 struct cctl_lun {
3211 	uint64_t lun_id;
3212 	char *backend_type;
3213 	uint64_t size_blocks;
3214 	uint32_t blocksize;
3215 	char *serial_number;
3216 	char *device_id;
3217 	STAILQ_HEAD(,cctl_lun_nv) attr_list;
3218 	STAILQ_ENTRY(cctl_lun) links;
3219 };
3220 
3221 struct cctl_devlist_data {
3222 	int num_luns;
3223 	STAILQ_HEAD(,cctl_lun) lun_list;
3224 	struct cctl_lun *cur_lun;
3225 	int level;
3226 	struct sbuf *cur_sb[32];
3227 };
3228 
3229 static void
3230 cctl_start_element(void *user_data, const char *name, const char **attr)
3231 {
3232 	int i;
3233 	struct cctl_devlist_data *devlist;
3234 	struct cctl_lun *cur_lun;
3235 
3236 	devlist = (struct cctl_devlist_data *)user_data;
3237 	cur_lun = devlist->cur_lun;
3238 	devlist->level++;
3239 	if ((u_int)devlist->level >= (sizeof(devlist->cur_sb) /
3240 	    sizeof(devlist->cur_sb[0])))
3241 		errx(1, "%s: too many nesting levels, %zd max", __func__,
3242 		     sizeof(devlist->cur_sb) / sizeof(devlist->cur_sb[0]));
3243 
3244 	devlist->cur_sb[devlist->level] = sbuf_new_auto();
3245 	if (devlist->cur_sb[devlist->level] == NULL)
3246 		err(1, "%s: Unable to allocate sbuf", __func__);
3247 
3248 	if (strcmp(name, "lun") == 0) {
3249 		if (cur_lun != NULL)
3250 			errx(1, "%s: improper lun element nesting", __func__);
3251 
3252 		cur_lun = calloc(1, sizeof(*cur_lun));
3253 		if (cur_lun == NULL)
3254 			err(1, "%s: cannot allocate %zd bytes", __func__,
3255 			    sizeof(*cur_lun));
3256 
3257 		devlist->num_luns++;
3258 		devlist->cur_lun = cur_lun;
3259 
3260 		STAILQ_INIT(&cur_lun->attr_list);
3261 		STAILQ_INSERT_TAIL(&devlist->lun_list, cur_lun, links);
3262 
3263 		for (i = 0; attr[i] != NULL; i += 2) {
3264 			if (strcmp(attr[i], "id") == 0) {
3265 				cur_lun->lun_id = strtoull(attr[i+1], NULL, 0);
3266 			} else {
3267 				errx(1, "%s: invalid LUN attribute %s = %s",
3268 				     __func__, attr[i], attr[i+1]);
3269 			}
3270 		}
3271 	}
3272 }
3273 
3274 static void
3275 cctl_end_element(void *user_data, const char *name)
3276 {
3277 	struct cctl_devlist_data *devlist;
3278 	struct cctl_lun *cur_lun;
3279 	char *str;
3280 
3281 	devlist = (struct cctl_devlist_data *)user_data;
3282 	cur_lun = devlist->cur_lun;
3283 
3284 	if ((cur_lun == NULL)
3285 	 && (strcmp(name, "ctllunlist") != 0))
3286 		errx(1, "%s: cur_lun == NULL! (name = %s)", __func__, name);
3287 
3288 	if (devlist->cur_sb[devlist->level] == NULL)
3289 		errx(1, "%s: no valid sbuf at level %d (name %s)", __func__,
3290 		     devlist->level, name);
3291 
3292 	if (sbuf_finish(devlist->cur_sb[devlist->level]) != 0)
3293 		err(1, "%s: sbuf_finish", __func__);
3294 	str = strdup(sbuf_data(devlist->cur_sb[devlist->level]));
3295 	if (str == NULL)
3296 		err(1, "%s can't allocate %zd bytes for string", __func__,
3297 		    sbuf_len(devlist->cur_sb[devlist->level]));
3298 
3299 	if (strlen(str) == 0) {
3300 		free(str);
3301 		str = NULL;
3302 	}
3303 
3304 	sbuf_delete(devlist->cur_sb[devlist->level]);
3305 	devlist->cur_sb[devlist->level] = NULL;
3306 	devlist->level--;
3307 
3308 	if (strcmp(name, "backend_type") == 0) {
3309 		cur_lun->backend_type = str;
3310 		str = NULL;
3311 	} else if (strcmp(name, "size") == 0) {
3312 		cur_lun->size_blocks = strtoull(str, NULL, 0);
3313 	} else if (strcmp(name, "blocksize") == 0) {
3314 		cur_lun->blocksize = strtoul(str, NULL, 0);
3315 	} else if (strcmp(name, "serial_number") == 0) {
3316 		cur_lun->serial_number = str;
3317 		str = NULL;
3318 	} else if (strcmp(name, "device_id") == 0) {
3319 		cur_lun->device_id = str;
3320 		str = NULL;
3321 	} else if (strcmp(name, "lun") == 0) {
3322 		devlist->cur_lun = NULL;
3323 	} else if (strcmp(name, "ctllunlist") == 0) {
3324 		/* Nothing. */
3325 	} else {
3326 		struct cctl_lun_nv *nv;
3327 
3328 		nv = calloc(1, sizeof(*nv));
3329 		if (nv == NULL)
3330 			err(1, "%s: can't allocate %zd bytes for nv pair",
3331 			    __func__, sizeof(*nv));
3332 
3333 		nv->name = strdup(name);
3334 		if (nv->name == NULL)
3335 			err(1, "%s: can't allocated %zd bytes for string",
3336 			    __func__, strlen(name));
3337 
3338 		nv->value = str;
3339 		str = NULL;
3340 		STAILQ_INSERT_TAIL(&cur_lun->attr_list, nv, links);
3341 	}
3342 
3343 	free(str);
3344 }
3345 
3346 static void
3347 cctl_char_handler(void *user_data, const XML_Char *str, int len)
3348 {
3349 	struct cctl_devlist_data *devlist;
3350 
3351 	devlist = (struct cctl_devlist_data *)user_data;
3352 
3353 	sbuf_bcat(devlist->cur_sb[devlist->level], str, len);
3354 }
3355 
3356 static int
3357 cctl_devlist(int fd, int argc, char **argv, char *combinedopt)
3358 {
3359 	struct ctl_lun_list list;
3360 	struct cctl_devlist_data devlist;
3361 	struct cctl_lun *lun;
3362 	XML_Parser parser;
3363 	char *lun_str;
3364 	int lun_len;
3365 	int dump_xml = 0;
3366 	int retval, c;
3367 	char *backend = NULL;
3368 	int verbose = 0;
3369 
3370 	retval = 0;
3371 	lun_len = 4096;
3372 
3373 	bzero(&devlist, sizeof(devlist));
3374 	STAILQ_INIT(&devlist.lun_list);
3375 
3376 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
3377 		switch (c) {
3378 		case 'b':
3379 			backend = strdup(optarg);
3380 			break;
3381 		case 'v':
3382 			verbose++;
3383 			break;
3384 		case 'x':
3385 			dump_xml = 1;
3386 			break;
3387 		default:
3388 			break;
3389 		}
3390 	}
3391 
3392 retry:
3393 	lun_str = malloc(lun_len);
3394 
3395 	bzero(&list, sizeof(list));
3396 	list.alloc_len = lun_len;
3397 	list.status = CTL_LUN_LIST_NONE;
3398 	list.lun_xml = lun_str;
3399 
3400 	if (ioctl(fd, CTL_LUN_LIST, &list) == -1) {
3401 		warn("%s: error issuing CTL_LUN_LIST ioctl", __func__);
3402 		retval = 1;
3403 		goto bailout;
3404 	}
3405 
3406 	if (list.status == CTL_LUN_LIST_ERROR) {
3407 		warnx("%s: error returned from CTL_LUN_LIST ioctl:\n%s",
3408 		      __func__, list.error_str);
3409 	} else if (list.status == CTL_LUN_LIST_NEED_MORE_SPACE) {
3410 		lun_len = lun_len << 1;
3411 		goto retry;
3412 	}
3413 
3414 	if (dump_xml != 0) {
3415 		printf("%s", lun_str);
3416 		goto bailout;
3417 	}
3418 
3419 	parser = XML_ParserCreate(NULL);
3420 	if (parser == NULL) {
3421 		warn("%s: Unable to create XML parser", __func__);
3422 		retval = 1;
3423 		goto bailout;
3424 	}
3425 
3426 	XML_SetUserData(parser, &devlist);
3427 	XML_SetElementHandler(parser, cctl_start_element, cctl_end_element);
3428 	XML_SetCharacterDataHandler(parser, cctl_char_handler);
3429 
3430 	retval = XML_Parse(parser, lun_str, strlen(lun_str), 1);
3431 	if (retval != 1) {
3432 		warnx("%s: Unable to parse XML: Error %d", __func__,
3433 		    XML_GetErrorCode(parser));
3434 		XML_ParserFree(parser);
3435 		retval = 1;
3436 		goto bailout;
3437 	}
3438 	retval = 0;
3439 	XML_ParserFree(parser);
3440 
3441 	printf("LUN Backend  %18s %4s %-16s %-16s\n", "Size (Blocks)", "BS",
3442 	       "Serial Number", "Device ID");
3443 	STAILQ_FOREACH(lun, &devlist.lun_list, links) {
3444 		struct cctl_lun_nv *nv;
3445 
3446 		if ((backend != NULL)
3447 		 && (strcmp(lun->backend_type, backend) != 0))
3448 			continue;
3449 
3450 		printf("%3ju %-8s %18ju %4u %-16s %-16s\n",
3451 		       (uintmax_t)lun->lun_id,
3452 		       lun->backend_type, (uintmax_t)lun->size_blocks,
3453 		       lun->blocksize, lun->serial_number, lun->device_id);
3454 
3455 		if (verbose == 0)
3456 			continue;
3457 
3458 		STAILQ_FOREACH(nv, &lun->attr_list, links) {
3459 			printf("      %s=%s\n", nv->name, nv->value);
3460 		}
3461 	}
3462 bailout:
3463 	free(lun_str);
3464 
3465 	return (retval);
3466 }
3467 
3468 /*
3469  * Port information.
3470  */
3471 struct cctl_port {
3472 	uint64_t port_id;
3473 	char *online;
3474 	char *frontend_type;
3475 	char *name;
3476 	int pp, vp;
3477 	char *target, *port, *lun_map;
3478 	STAILQ_HEAD(,cctl_lun_nv) init_list;
3479 	STAILQ_HEAD(,cctl_lun_nv) lun_list;
3480 	STAILQ_HEAD(,cctl_lun_nv) attr_list;
3481 	STAILQ_ENTRY(cctl_port) links;
3482 };
3483 
3484 struct cctl_portlist_data {
3485 	int num_ports;
3486 	STAILQ_HEAD(,cctl_port) port_list;
3487 	struct cctl_port *cur_port;
3488 	int level;
3489 	uint64_t cur_id;
3490 	struct sbuf *cur_sb[32];
3491 };
3492 
3493 static void
3494 cctl_start_pelement(void *user_data, const char *name, const char **attr)
3495 {
3496 	int i;
3497 	struct cctl_portlist_data *portlist;
3498 	struct cctl_port *cur_port;
3499 
3500 	portlist = (struct cctl_portlist_data *)user_data;
3501 	cur_port = portlist->cur_port;
3502 	portlist->level++;
3503 	if ((u_int)portlist->level >= (sizeof(portlist->cur_sb) /
3504 	    sizeof(portlist->cur_sb[0])))
3505 		errx(1, "%s: too many nesting levels, %zd max", __func__,
3506 		     sizeof(portlist->cur_sb) / sizeof(portlist->cur_sb[0]));
3507 
3508 	portlist->cur_sb[portlist->level] = sbuf_new_auto();
3509 	if (portlist->cur_sb[portlist->level] == NULL)
3510 		err(1, "%s: Unable to allocate sbuf", __func__);
3511 
3512 	portlist->cur_id = 0;
3513 	for (i = 0; attr[i] != NULL; i += 2) {
3514 		if (strcmp(attr[i], "id") == 0) {
3515 			portlist->cur_id = strtoull(attr[i+1], NULL, 0);
3516 			break;
3517 		}
3518 	}
3519 
3520 	if (strcmp(name, "targ_port") == 0) {
3521 		if (cur_port != NULL)
3522 			errx(1, "%s: improper port element nesting", __func__);
3523 
3524 		cur_port = calloc(1, sizeof(*cur_port));
3525 		if (cur_port == NULL)
3526 			err(1, "%s: cannot allocate %zd bytes", __func__,
3527 			    sizeof(*cur_port));
3528 
3529 		portlist->num_ports++;
3530 		portlist->cur_port = cur_port;
3531 
3532 		STAILQ_INIT(&cur_port->init_list);
3533 		STAILQ_INIT(&cur_port->lun_list);
3534 		STAILQ_INIT(&cur_port->attr_list);
3535 		cur_port->port_id = portlist->cur_id;
3536 		STAILQ_INSERT_TAIL(&portlist->port_list, cur_port, links);
3537 	}
3538 }
3539 
3540 static void
3541 cctl_end_pelement(void *user_data, const char *name)
3542 {
3543 	struct cctl_portlist_data *portlist;
3544 	struct cctl_port *cur_port;
3545 	char *str;
3546 
3547 	portlist = (struct cctl_portlist_data *)user_data;
3548 	cur_port = portlist->cur_port;
3549 
3550 	if ((cur_port == NULL)
3551 	 && (strcmp(name, "ctlportlist") != 0))
3552 		errx(1, "%s: cur_port == NULL! (name = %s)", __func__, name);
3553 
3554 	if (portlist->cur_sb[portlist->level] == NULL)
3555 		errx(1, "%s: no valid sbuf at level %d (name %s)", __func__,
3556 		     portlist->level, name);
3557 
3558 	if (sbuf_finish(portlist->cur_sb[portlist->level]) != 0)
3559 		err(1, "%s: sbuf_finish", __func__);
3560 	str = strdup(sbuf_data(portlist->cur_sb[portlist->level]));
3561 	if (str == NULL)
3562 		err(1, "%s can't allocate %zd bytes for string", __func__,
3563 		    sbuf_len(portlist->cur_sb[portlist->level]));
3564 
3565 	if (strlen(str) == 0) {
3566 		free(str);
3567 		str = NULL;
3568 	}
3569 
3570 	sbuf_delete(portlist->cur_sb[portlist->level]);
3571 	portlist->cur_sb[portlist->level] = NULL;
3572 	portlist->level--;
3573 
3574 	if (strcmp(name, "frontend_type") == 0) {
3575 		cur_port->frontend_type = str;
3576 		str = NULL;
3577 	} else if (strcmp(name, "port_name") == 0) {
3578 		cur_port->name = str;
3579 		str = NULL;
3580 	} else if (strcmp(name, "online") == 0) {
3581 		cur_port->online = str;
3582 		str = NULL;
3583 	} else if (strcmp(name, "physical_port") == 0) {
3584 		cur_port->pp = strtoull(str, NULL, 0);
3585 	} else if (strcmp(name, "virtual_port") == 0) {
3586 		cur_port->vp = strtoull(str, NULL, 0);
3587 	} else if (strcmp(name, "target") == 0) {
3588 		cur_port->target = str;
3589 		str = NULL;
3590 	} else if (strcmp(name, "port") == 0) {
3591 		cur_port->port = str;
3592 		str = NULL;
3593 	} else if (strcmp(name, "lun_map") == 0) {
3594 		cur_port->lun_map = str;
3595 		str = NULL;
3596 	} else if (strcmp(name, "targ_port") == 0) {
3597 		portlist->cur_port = NULL;
3598 	} else if (strcmp(name, "ctlportlist") == 0) {
3599 		/* Nothing. */
3600 	} else {
3601 		struct cctl_lun_nv *nv;
3602 
3603 		nv = calloc(1, sizeof(*nv));
3604 		if (nv == NULL)
3605 			err(1, "%s: can't allocate %zd bytes for nv pair",
3606 			    __func__, sizeof(*nv));
3607 
3608 		if (strcmp(name, "initiator") == 0 ||
3609 		    strcmp(name, "lun") == 0)
3610 			asprintf(&nv->name, "%ju", portlist->cur_id);
3611 		else
3612 			nv->name = strdup(name);
3613 		if (nv->name == NULL)
3614 			err(1, "%s: can't allocated %zd bytes for string",
3615 			    __func__, strlen(name));
3616 
3617 		nv->value = str;
3618 		str = NULL;
3619 		if (strcmp(name, "initiator") == 0)
3620 			STAILQ_INSERT_TAIL(&cur_port->init_list, nv, links);
3621 		else if (strcmp(name, "lun") == 0)
3622 			STAILQ_INSERT_TAIL(&cur_port->lun_list, nv, links);
3623 		else
3624 			STAILQ_INSERT_TAIL(&cur_port->attr_list, nv, links);
3625 	}
3626 
3627 	free(str);
3628 }
3629 
3630 static void
3631 cctl_char_phandler(void *user_data, const XML_Char *str, int len)
3632 {
3633 	struct cctl_portlist_data *portlist;
3634 
3635 	portlist = (struct cctl_portlist_data *)user_data;
3636 
3637 	sbuf_bcat(portlist->cur_sb[portlist->level], str, len);
3638 }
3639 
3640 static int
3641 cctl_portlist(int fd, int argc, char **argv, char *combinedopt)
3642 {
3643 	struct ctl_lun_list list;
3644 	struct cctl_portlist_data portlist;
3645 	struct cctl_port *port;
3646 	XML_Parser parser;
3647 	char *port_str = NULL;
3648 	int port_len;
3649 	int dump_xml = 0;
3650 	int retval, c;
3651 	char *frontend = NULL;
3652 	uint64_t portarg = UINT64_MAX;
3653 	int verbose = 0, init = 0, lun = 0, quiet = 0;
3654 
3655 	retval = 0;
3656 	port_len = 4096;
3657 
3658 	bzero(&portlist, sizeof(portlist));
3659 	STAILQ_INIT(&portlist.port_list);
3660 
3661 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
3662 		switch (c) {
3663 		case 'f':
3664 			frontend = strdup(optarg);
3665 			break;
3666 		case 'i':
3667 			init++;
3668 			break;
3669 		case 'l':
3670 			lun++;
3671 			break;
3672 		case 'p':
3673 			portarg = strtoll(optarg, NULL, 0);
3674 			break;
3675 		case 'q':
3676 			quiet++;
3677 			break;
3678 		case 'v':
3679 			verbose++;
3680 			break;
3681 		case 'x':
3682 			dump_xml = 1;
3683 			break;
3684 		default:
3685 			break;
3686 		}
3687 	}
3688 
3689 retry:
3690 	port_str = (char *)realloc(port_str, port_len);
3691 
3692 	bzero(&list, sizeof(list));
3693 	list.alloc_len = port_len;
3694 	list.status = CTL_LUN_LIST_NONE;
3695 	list.lun_xml = port_str;
3696 
3697 	if (ioctl(fd, CTL_PORT_LIST, &list) == -1) {
3698 		warn("%s: error issuing CTL_PORT_LIST ioctl", __func__);
3699 		retval = 1;
3700 		goto bailout;
3701 	}
3702 
3703 	if (list.status == CTL_LUN_LIST_ERROR) {
3704 		warnx("%s: error returned from CTL_PORT_LIST ioctl:\n%s",
3705 		      __func__, list.error_str);
3706 	} else if (list.status == CTL_LUN_LIST_NEED_MORE_SPACE) {
3707 		port_len = port_len << 1;
3708 		goto retry;
3709 	}
3710 
3711 	if (dump_xml != 0) {
3712 		printf("%s", port_str);
3713 		goto bailout;
3714 	}
3715 
3716 	parser = XML_ParserCreate(NULL);
3717 	if (parser == NULL) {
3718 		warn("%s: Unable to create XML parser", __func__);
3719 		retval = 1;
3720 		goto bailout;
3721 	}
3722 
3723 	XML_SetUserData(parser, &portlist);
3724 	XML_SetElementHandler(parser, cctl_start_pelement, cctl_end_pelement);
3725 	XML_SetCharacterDataHandler(parser, cctl_char_phandler);
3726 
3727 	retval = XML_Parse(parser, port_str, strlen(port_str), 1);
3728 	if (retval != 1) {
3729 		warnx("%s: Unable to parse XML: Error %d", __func__,
3730 		    XML_GetErrorCode(parser));
3731 		XML_ParserFree(parser);
3732 		retval = 1;
3733 		goto bailout;
3734 	}
3735 	retval = 0;
3736 	XML_ParserFree(parser);
3737 
3738 	if (quiet == 0)
3739 		printf("Port Online Frontend Name     pp vp\n");
3740 	STAILQ_FOREACH(port, &portlist.port_list, links) {
3741 		struct cctl_lun_nv *nv;
3742 
3743 		if ((frontend != NULL)
3744 		 && (strcmp(port->frontend_type, frontend) != 0))
3745 			continue;
3746 
3747 		if ((portarg != UINT64_MAX) && (portarg != port->port_id))
3748 			continue;
3749 
3750 		printf("%-4ju %-6s %-8s %-8s %-2d %-2d %s\n",
3751 		    (uintmax_t)port->port_id, port->online,
3752 		    port->frontend_type, port->name, port->pp, port->vp,
3753 		    port->port ? port->port : "");
3754 
3755 		if (init || verbose) {
3756 			if (port->target)
3757 				printf("  Target: %s\n", port->target);
3758 			STAILQ_FOREACH(nv, &port->init_list, links) {
3759 				printf("  Initiator %s: %s\n",
3760 				    nv->name, nv->value);
3761 			}
3762 		}
3763 
3764 		if (lun || verbose) {
3765 			if (port->lun_map) {
3766 				STAILQ_FOREACH(nv, &port->lun_list, links)
3767 					printf("  LUN %s: %s\n",
3768 					    nv->name, nv->value);
3769 				if (STAILQ_EMPTY(&port->lun_list))
3770 					printf("  No LUNs mapped\n");
3771 			} else
3772 				printf("  All LUNs mapped\n");
3773 		}
3774 
3775 		if (verbose) {
3776 			STAILQ_FOREACH(nv, &port->attr_list, links) {
3777 				printf("      %s=%s\n", nv->name, nv->value);
3778 			}
3779 		}
3780 	}
3781 bailout:
3782 	free(port_str);
3783 
3784 	return (retval);
3785 }
3786 
3787 static int
3788 cctl_lunmap(int fd, int argc, char **argv, char *combinedopt)
3789 {
3790 	struct ctl_lun_map lm;
3791 	int retval = 0, c;
3792 
3793 	retval = 0;
3794 	lm.port = UINT32_MAX;
3795 	lm.plun = UINT32_MAX;
3796 	lm.lun = UINT32_MAX;
3797 
3798 	while ((c = getopt(argc, argv, combinedopt)) != -1) {
3799 		switch (c) {
3800 		case 'p':
3801 			lm.port = strtoll(optarg, NULL, 0);
3802 			break;
3803 		case 'l':
3804 			lm.plun = strtoll(optarg, NULL, 0);
3805 			break;
3806 		case 'L':
3807 			lm.lun = strtoll(optarg, NULL, 0);
3808 			break;
3809 		default:
3810 			break;
3811 		}
3812 	}
3813 
3814 	if (ioctl(fd, CTL_LUN_MAP, &lm) == -1) {
3815 		warn("%s: error issuing CTL_LUN_MAP ioctl", __func__);
3816 		retval = 1;
3817 	}
3818 
3819 	return (retval);
3820 }
3821 
3822 void
3823 usage(int error)
3824 {
3825 	fprintf(error ? stderr : stdout,
3826 "Usage:\n"
3827 "Primary commands:\n"
3828 "         ctladm tur         [dev_id][general options]\n"
3829 "         ctladm inquiry     [dev_id][general options]\n"
3830 "         ctladm devid       [dev_id][general options]\n"
3831 "         ctladm reqsense    [dev_id][general options]\n"
3832 "         ctladm reportluns  [dev_id][general options]\n"
3833 "         ctladm read        [dev_id][general options] <-l lba> <-d len>\n"
3834 "                            <-f file|-> <-b blocksize> [-c cdbsize][-N]\n"
3835 "         ctladm write       [dev_id][general options] <-l lba> <-d len>\n"
3836 "                            <-f file|-> <-b blocksize> [-c cdbsize][-N]\n"
3837 "         ctladm readcap     [dev_id][general options] [-c cdbsize]\n"
3838 "         ctladm modesense   [dev_id][general options] <-m page|-l> [-P pc]\n"
3839 "                            [-d] [-S subpage] [-c cdbsize]\n"
3840 "         ctladm prin        [dev_id][general options] <-a action>\n"
3841 "         ctladm prout       [dev_id][general options] <-a action>\n"
3842 "                            <-r restype] [-k key] [-s sa_key]\n"
3843 "         ctladm rtpg        [dev_id][general options]\n"
3844 "         ctladm start       [dev_id][general options] [-i] [-o]\n"
3845 "         ctladm stop        [dev_id][general options] [-i] [-o]\n"
3846 "         ctladm synccache   [dev_id][general options] [-l lba]\n"
3847 "                            [-b blockcount] [-r] [-i] [-c cdbsize]\n"
3848 "         ctladm create      <-b backend> [-B blocksize] [-d device_id]\n"
3849 "                            [-l lun_id] [-o name=value] [-s size_bytes]\n"
3850 "                            [-S serial_num] [-t dev_type]\n"
3851 "         ctladm remove      <-b backend> <-l lun_id> [-o name=value]\n"
3852 "         ctladm modify      <-b backend> <-l lun_id> <-s size_bytes>\n"
3853 "         ctladm devlist     [-b backend] [-v] [-x]\n"
3854 "         ctladm lunlist\n"
3855 "         ctladm lunmap      -p targ_port [-l pLUN] [-L cLUN]\n"
3856 "         ctladm delay       [dev_id] <-l datamove|done> [-T oneshot|cont]\n"
3857 "                            [-t secs]\n"
3858 "         ctladm inject      [dev_id] <-i action> <-p pattern> [-r lba,len]\n"
3859 "                            [-s len fmt [args]] [-c] [-d delete_id]\n"
3860 "         ctladm port        <-o <on|off> | [-w wwnn][-W wwpn]>\n"
3861 "                            [-p targ_port] [-t port_type]\n"
3862 "                            <-c> [-d driver] [-O name=value]\n"
3863 "                            <-r> <-p targ_port>\n"
3864 "         ctladm portlist    [-f frontend] [-i] [-p targ_port] [-q] [-v] [-x]\n"
3865 "         ctladm islist      [-v | -x]\n"
3866 "         ctladm islogout    <-a | -c connection-id | -i name | -p portal>\n"
3867 "         ctladm isterminate <-a | -c connection-id | -i name | -p portal>\n"
3868 "         ctladm dumpooa\n"
3869 "         ctladm dumpstructs\n"
3870 "         ctladm help\n"
3871 "General Options:\n"
3872 "-I intiator_id           : defaults to 7, used to change the initiator id\n"
3873 "-C retries               : specify the number of times to retry this command\n"
3874 "-D devicename            : specify the device to operate on\n"
3875 "                         : (default is %s)\n"
3876 "read/write options:\n"
3877 "-l lba                   : logical block address\n"
3878 "-d len                   : read/write length, in blocks\n"
3879 "-f file|-                : write/read data to/from file or stdout/stdin\n"
3880 "-b blocksize             : block size, in bytes\n"
3881 "-c cdbsize               : specify minimum cdb size: 6, 10, 12 or 16\n"
3882 "-N                       : do not copy data to/from userland\n"
3883 "readcapacity options:\n"
3884 "-c cdbsize               : specify minimum cdb size: 10 or 16\n"
3885 "modesense options:\n"
3886 "-m page                  : specify the mode page to view\n"
3887 "-l                       : request a list of supported pages\n"
3888 "-P pc                    : specify the page control value: 0-3 (current,\n"
3889 "                           changeable, default, saved, respectively)\n"
3890 "-d                       : disable block descriptors for mode sense\n"
3891 "-S subpage               : specify a subpage\n"
3892 "-c cdbsize               : specify minimum cdb size: 6 or 10\n"
3893 "persistent reserve in options:\n"
3894 "-a action                : specify the action value: 0-2 (read key, read\n"
3895 "                           reservation, read capabilities, respectively)\n"
3896 "persistent reserve out options:\n"
3897 "-a action                : specify the action value: 0-5 (register, reserve,\n"
3898 "                           release, clear, preempt, register and ignore)\n"
3899 "-k key                   : key value\n"
3900 "-s sa_key                : service action value\n"
3901 "-r restype               : specify the reservation type: 0-5(wr ex, ex ac,\n"
3902 "                           wr ex ro, ex ac ro, wr ex ar, ex ac ar)\n"
3903 "start/stop options:\n"
3904 "-i                       : set the immediate bit (CTL does not support this)\n"
3905 "-o                       : set the on/offline bit\n"
3906 "synccache options:\n"
3907 "-l lba                   : set the starting LBA\n"
3908 "-b blockcount            : set the length to sync in blocks\n"
3909 "-r                       : set the relative addressing bit\n"
3910 "-i                       : set the immediate bit\n"
3911 "-c cdbsize               : specify minimum cdb size: 10 or 16\n"
3912 "create options:\n"
3913 "-b backend               : backend name (\"block\", \"ramdisk\", etc.)\n"
3914 "-B blocksize             : LUN blocksize in bytes (some backends)\n"
3915 "-d device_id             : SCSI VPD page 0x83 ID\n"
3916 "-l lun_id                : requested LUN number\n"
3917 "-o name=value            : backend-specific options, multiple allowed\n"
3918 "-s size_bytes            : LUN size in bytes (some backends)\n"
3919 "-S serial_num            : SCSI VPD page 0x80 serial number\n"
3920 "-t dev_type              : SCSI device type (0=disk, 3=processor)\n"
3921 "remove options:\n"
3922 "-b backend               : backend name (\"block\", \"ramdisk\", etc.)\n"
3923 "-l lun_id                : LUN number to delete\n"
3924 "-o name=value            : backend-specific options, multiple allowed\n"
3925 "devlist options:\n"
3926 "-b backend               : list devices from specified backend only\n"
3927 "-v                       : be verbose, show backend attributes\n"
3928 "-x                       : dump raw XML\n"
3929 "delay options:\n"
3930 "-l datamove|done         : delay command at datamove or done phase\n"
3931 "-T oneshot               : delay one command, then resume normal completion\n"
3932 "-T cont                  : delay all commands\n"
3933 "-t secs                  : number of seconds to delay\n"
3934 "inject options:\n"
3935 "-i error_action          : action to perform\n"
3936 "-p pattern               : command pattern to look for\n"
3937 "-r lba,len               : LBA range for pattern\n"
3938 "-s len fmt [args]        : sense data for custom sense action\n"
3939 "-c                       : continuous operation\n"
3940 "-d delete_id             : error id to delete\n"
3941 "port options:\n"
3942 "-c                       : create new ioctl or iscsi frontend port\n"
3943 "-d                       : specify ioctl or iscsi frontend type\n"
3944 "-l                       : list frontend ports\n"
3945 "-o on|off                : turn frontend ports on or off\n"
3946 "-O pp|vp                 : create new frontend port using pp and/or vp\n"
3947 "-w wwnn                  : set WWNN for one frontend\n"
3948 "-W wwpn                  : set WWPN for one frontend\n"
3949 "-t port_type             : specify fc, scsi, ioctl, internal frontend type\n"
3950 "-p targ_port             : specify target port number\n"
3951 "-r                       : remove frontend port\n"
3952 "-q                       : omit header in list output\n"
3953 "-x                       : output port list in XML format\n"
3954 "portlist options:\n"
3955 "-f frontend              : specify frontend type\n"
3956 "-i                       : report target and initiators addresses\n"
3957 "-l                       : report LUN mapping\n"
3958 "-p targ_port             : specify target port number\n"
3959 "-q                       : omit header in list output\n"
3960 "-v                       : verbose output (report all port options)\n"
3961 "-x                       : output port list in XML format\n"
3962 "lunmap options:\n"
3963 "-p targ_port             : specify target port number\n"
3964 "-L pLUN                  : specify port-visible LUN\n"
3965 "-L cLUN                  : specify CTL LUN\n",
3966 CTL_DEFAULT_DEV);
3967 }
3968 
3969 int
3970 main(int argc, char **argv)
3971 {
3972 	int c;
3973 	ctladm_cmdfunction command;
3974 	ctladm_cmdargs cmdargs;
3975 	ctladm_optret optreturn;
3976 	char *device;
3977 	const char *mainopt = "C:D:I:";
3978 	const char *subopt = NULL;
3979 	char combinedopt[256];
3980 	int lun;
3981 	int optstart = 2;
3982 	int retval, fd;
3983 	int retries;
3984 	int initid;
3985 	int saved_errno;
3986 
3987 	retval = 0;
3988 	cmdargs = CTLADM_ARG_NONE;
3989 	command = CTLADM_CMD_HELP;
3990 	device = NULL;
3991 	fd = -1;
3992 	retries = 0;
3993 	lun = 0;
3994 	initid = 7;
3995 
3996 	if (argc < 2) {
3997 		usage(1);
3998 		retval = 1;
3999 		goto bailout;
4000 	}
4001 
4002 	/*
4003 	 * Get the base option.
4004 	 */
4005 	optreturn = getoption(option_table,argv[1], &command, &cmdargs,&subopt);
4006 
4007 	if (optreturn == CC_OR_AMBIGUOUS) {
4008 		warnx("ambiguous option %s", argv[1]);
4009 		usage(0);
4010 		exit(1);
4011 	} else if (optreturn == CC_OR_NOT_FOUND) {
4012 		warnx("option %s not found", argv[1]);
4013 		usage(0);
4014 		exit(1);
4015 	}
4016 
4017 	if (cmdargs & CTLADM_ARG_NEED_TL) {
4018 		if ((argc < 3) || (!isdigit(argv[2][0]))) {
4019 			warnx("option %s requires a lun argument",
4020 			      argv[1]);
4021 			usage(0);
4022 			exit(1);
4023 		}
4024 		lun = strtol(argv[2], NULL, 0);
4025 
4026 		cmdargs |= CTLADM_ARG_TARG_LUN;
4027 		optstart++;
4028 	}
4029 
4030 	/*
4031 	 * Ahh, getopt(3) is a pain.
4032 	 *
4033 	 * This is a gross hack.  There really aren't many other good
4034 	 * options (excuse the pun) for parsing options in a situation like
4035 	 * this.  getopt is kinda braindead, so you end up having to run
4036 	 * through the options twice, and give each invocation of getopt
4037 	 * the option string for the other invocation.
4038 	 *
4039 	 * You would think that you could just have two groups of options.
4040 	 * The first group would get parsed by the first invocation of
4041 	 * getopt, and the second group would get parsed by the second
4042 	 * invocation of getopt.  It doesn't quite work out that way.  When
4043 	 * the first invocation of getopt finishes, it leaves optind pointing
4044 	 * to the argument _after_ the first argument in the second group.
4045 	 * So when the second invocation of getopt comes around, it doesn't
4046 	 * recognize the first argument it gets and then bails out.
4047 	 *
4048 	 * A nice alternative would be to have a flag for getopt that says
4049 	 * "just keep parsing arguments even when you encounter an unknown
4050 	 * argument", but there isn't one.  So there's no real clean way to
4051 	 * easily parse two sets of arguments without having one invocation
4052 	 * of getopt know about the other.
4053 	 *
4054 	 * Without this hack, the first invocation of getopt would work as
4055 	 * long as the generic arguments are first, but the second invocation
4056 	 * (in the subfunction) would fail in one of two ways.  In the case
4057 	 * where you don't set optreset, it would fail because optind may be
4058 	 * pointing to the argument after the one it should be pointing at.
4059 	 * In the case where you do set optreset, and reset optind, it would
4060 	 * fail because getopt would run into the first set of options, which
4061 	 * it doesn't understand.
4062 	 *
4063 	 * All of this would "sort of" work if you could somehow figure out
4064 	 * whether optind had been incremented one option too far.  The
4065 	 * mechanics of that, however, are more daunting than just giving
4066 	 * both invocations all of the expect options for either invocation.
4067 	 *
4068 	 * Needless to say, I wouldn't mind if someone invented a better
4069 	 * (non-GPL!) command line parsing interface than getopt.  I
4070 	 * wouldn't mind if someone added more knobs to getopt to make it
4071 	 * work better.  Who knows, I may talk myself into doing it someday,
4072 	 * if the standards weenies let me.  As it is, it just leads to
4073 	 * hackery like this and causes people to avoid it in some cases.
4074 	 *
4075 	 * KDM, September 8th, 1998
4076 	 */
4077 	if (subopt != NULL)
4078 		sprintf(combinedopt, "%s%s", mainopt, subopt);
4079 	else
4080 		sprintf(combinedopt, "%s", mainopt);
4081 
4082 	/*
4083 	 * Start getopt processing at argv[2/3], since we've already
4084 	 * accepted argv[1..2] as the command name, and as a possible
4085 	 * device name.
4086 	 */
4087 	optind = optstart;
4088 
4089 	/*
4090 	 * Now we run through the argument list looking for generic
4091 	 * options, and ignoring options that possibly belong to
4092 	 * subfunctions.
4093 	 */
4094 	while ((c = getopt(argc, argv, combinedopt))!= -1){
4095 		switch (c) {
4096 		case 'C':
4097 			cmdargs |= CTLADM_ARG_RETRIES;
4098 			retries = strtol(optarg, NULL, 0);
4099 			break;
4100 		case 'D':
4101 			device = strdup(optarg);
4102 			cmdargs |= CTLADM_ARG_DEVICE;
4103 			break;
4104 		case 'I':
4105 			cmdargs |= CTLADM_ARG_INITIATOR;
4106 			initid = strtol(optarg, NULL, 0);
4107 			break;
4108 		default:
4109 			break;
4110 		}
4111 	}
4112 
4113 	if ((cmdargs & CTLADM_ARG_INITIATOR) == 0)
4114 		initid = 7;
4115 
4116 	optind = optstart;
4117 	optreset = 1;
4118 
4119 	/*
4120 	 * Default to opening the CTL device for now.
4121 	 */
4122 	if (((cmdargs & CTLADM_ARG_DEVICE) == 0)
4123 	 && (command != CTLADM_CMD_HELP)) {
4124 		device = strdup(CTL_DEFAULT_DEV);
4125 		cmdargs |= CTLADM_ARG_DEVICE;
4126 	}
4127 
4128 	if ((cmdargs & CTLADM_ARG_DEVICE)
4129 	 && (command != CTLADM_CMD_HELP)) {
4130 		fd = open(device, O_RDWR);
4131 		if (fd == -1 && errno == ENOENT) {
4132 			saved_errno = errno;
4133 			retval = kldload("ctl");
4134 			if (retval != -1)
4135 				fd = open(device, O_RDWR);
4136 			else
4137 				errno = saved_errno;
4138 		}
4139 		if (fd == -1) {
4140 			fprintf(stderr, "%s: error opening %s: %s\n",
4141 				argv[0], device, strerror(errno));
4142 			retval = 1;
4143 			goto bailout;
4144 		}
4145 #ifdef	WANT_ISCSI
4146 		else {
4147 			if (modfind("cfiscsi") == -1 &&
4148 			    kldload("cfiscsi") == -1)
4149 				warn("couldn't load cfiscsi");
4150 		}
4151 #endif
4152 	} else if ((command != CTLADM_CMD_HELP)
4153 		&& ((cmdargs & CTLADM_ARG_DEVICE) == 0)) {
4154 		fprintf(stderr, "%s: you must specify a device with the "
4155 			"--device argument for this command\n", argv[0]);
4156 		command = CTLADM_CMD_HELP;
4157 		retval = 1;
4158 	}
4159 
4160 	switch (command) {
4161 	case CTLADM_CMD_TUR:
4162 		retval = cctl_tur(fd, lun, initid, retries);
4163 		break;
4164 	case CTLADM_CMD_INQUIRY:
4165 		retval = cctl_inquiry(fd, lun, initid, retries);
4166 		break;
4167 	case CTLADM_CMD_REQ_SENSE:
4168 		retval = cctl_req_sense(fd, lun, initid, retries);
4169 		break;
4170 	case CTLADM_CMD_REPORT_LUNS:
4171 		retval = cctl_report_luns(fd, lun, initid, retries);
4172 		break;
4173 	case CTLADM_CMD_CREATE:
4174 		retval = cctl_create_lun(fd, argc, argv, combinedopt);
4175 		break;
4176 	case CTLADM_CMD_RM:
4177 		retval = cctl_rm_lun(fd, argc, argv, combinedopt);
4178 		break;
4179 	case CTLADM_CMD_DEVLIST:
4180 		retval = cctl_devlist(fd, argc, argv, combinedopt);
4181 		break;
4182 	case CTLADM_CMD_READ:
4183 	case CTLADM_CMD_WRITE:
4184 		retval = cctl_read_write(fd, lun, initid, retries,
4185 					 argc, argv, combinedopt, command);
4186 		break;
4187 	case CTLADM_CMD_PORT:
4188 		retval = cctl_port(fd, argc, argv, combinedopt);
4189 		break;
4190 	case CTLADM_CMD_PORTLIST:
4191 		retval = cctl_portlist(fd, argc, argv, combinedopt);
4192 		break;
4193 	case CTLADM_CMD_LUNMAP:
4194 		retval = cctl_lunmap(fd, argc, argv, combinedopt);
4195 		break;
4196 	case CTLADM_CMD_READCAPACITY:
4197 		retval = cctl_read_capacity(fd, lun, initid, retries,
4198 					    argc, argv, combinedopt);
4199 		break;
4200 	case CTLADM_CMD_MODESENSE:
4201 		retval = cctl_mode_sense(fd, lun, initid, retries,
4202 					 argc, argv, combinedopt);
4203 		break;
4204 	case CTLADM_CMD_START:
4205 	case CTLADM_CMD_STOP:
4206 		retval = cctl_start_stop(fd, lun, initid, retries,
4207 					 (command == CTLADM_CMD_START) ? 1 : 0,
4208 					 argc, argv, combinedopt);
4209 		break;
4210 	case CTLADM_CMD_SYNC_CACHE:
4211 		retval = cctl_sync_cache(fd, lun, initid, retries,
4212 					 argc, argv, combinedopt);
4213 		break;
4214 	case CTLADM_CMD_LUNLIST:
4215 		retval = cctl_lunlist(fd);
4216 		break;
4217 	case CTLADM_CMD_DELAY:
4218 		retval = cctl_delay(fd, lun, argc, argv, combinedopt);
4219 		break;
4220 	case CTLADM_CMD_ERR_INJECT:
4221 		retval = cctl_error_inject(fd, lun, argc, argv,
4222 					   combinedopt);
4223 		break;
4224 	case CTLADM_CMD_DUMPOOA:
4225 		retval = cctl_dump_ooa(fd, argc, argv);
4226 		break;
4227 	case CTLADM_CMD_DUMPSTRUCTS:
4228 		retval = cctl_dump_structs(fd, cmdargs);
4229 		break;
4230 	case CTLADM_CMD_PRES_IN:
4231 		retval = cctl_persistent_reserve_in(fd, lun, initid,
4232 		                                    argc, argv, combinedopt,
4233 						    retries);
4234 		break;
4235 	case CTLADM_CMD_PRES_OUT:
4236 		retval = cctl_persistent_reserve_out(fd, lun, initid,
4237 						     argc, argv, combinedopt,
4238 						     retries);
4239 		break;
4240 	case CTLADM_CMD_INQ_VPD_DEVID:
4241 	        retval = cctl_inquiry_vpd_devid(fd, lun, initid);
4242 		break;
4243 	case CTLADM_CMD_RTPG:
4244 	        retval = cctl_report_target_port_group(fd, lun, initid);
4245 		break;
4246 	case CTLADM_CMD_MODIFY:
4247 	        retval = cctl_modify_lun(fd, argc, argv, combinedopt);
4248 		break;
4249 	case CTLADM_CMD_ISLIST:
4250 	        retval = cctl_islist(fd, argc, argv, combinedopt);
4251 		break;
4252 	case CTLADM_CMD_ISLOGOUT:
4253 	        retval = cctl_islogout(fd, argc, argv, combinedopt);
4254 		break;
4255 	case CTLADM_CMD_ISTERMINATE:
4256 	        retval = cctl_isterminate(fd, argc, argv, combinedopt);
4257 		break;
4258 	case CTLADM_CMD_HELP:
4259 	default:
4260 		usage(retval);
4261 		break;
4262 	}
4263 bailout:
4264 
4265 	if (fd != -1)
4266 		close(fd);
4267 
4268 	exit (retval);
4269 }
4270 
4271 /*
4272  * vim: ts=8
4273  */
4274