xref: /freebsd/usr.sbin/mpsutil/mps_cmd.c (revision 2f513db72b034fd5ef7f080b11be5c711c15186a)
1 /*-
2  * Copyright (c) 2015 Baptiste Daroussin <bapt@FreeBSD.org>
3  *
4  * Copyright (c) 2015 Netflix, Inc.
5  * Written by: Scott Long <scottl@freebsd.org>
6  *
7  * Copyright (c) 2008 Yahoo!, Inc.
8  * All rights reserved.
9  * Written by: John Baldwin <jhb@FreeBSD.org>
10  *
11  * Redistribution and use in source and binary forms, with or without
12  * modification, are permitted provided that the following conditions
13  * are met:
14  * 1. Redistributions of source code must retain the above copyright
15  *    notice, this list of conditions and the following disclaimer.
16  * 2. Redistributions in binary form must reproduce the above copyright
17  *    notice, this list of conditions and the following disclaimer in the
18  *    documentation and/or other materials provided with the distribution.
19  * 3. Neither the name of the author nor the names of any co-contributors
20  *    may be used to endorse or promote products derived from this software
21  *    without specific prior written permission.
22  *
23  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
24  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
25  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
27  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
28  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
29  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
30  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
32  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
33  * SUCH DAMAGE.
34  */
35 
36 #include <sys/cdefs.h>
37 __RCSID("$FreeBSD$");
38 
39 #include <sys/param.h>
40 #include <sys/errno.h>
41 #include <sys/ioctl.h>
42 #if 0
43 #include <sys/mps_ioctl.h>
44 #else
45 #include "mps_ioctl.h"
46 #include "mpr_ioctl.h"
47 #endif
48 #include <sys/sysctl.h>
49 #include <sys/uio.h>
50 
51 #include <err.h>
52 #include <fcntl.h>
53 #include <stdio.h>
54 #include <stdlib.h>
55 #include <string.h>
56 #include <unistd.h>
57 
58 #include "mpsutil.h"
59 
60 #ifndef USE_MPT_IOCTLS
61 #define USE_MPT_IOCTLS
62 #endif
63 
64 static const char *mps_ioc_status_codes[] = {
65 	"Success",				/* 0x0000 */
66 	"Invalid function",
67 	"Busy",
68 	"Invalid scatter-gather list",
69 	"Internal error",
70 	"Reserved",
71 	"Insufficient resources",
72 	"Invalid field",
73 	"Invalid state",			/* 0x0008 */
74 	"Operation state not supported",
75 	NULL,
76 	NULL,
77 	NULL,
78 	NULL,
79 	NULL,
80 	NULL,
81 	NULL,					/* 0x0010 */
82 	NULL,
83 	NULL,
84 	NULL,
85 	NULL,
86 	NULL,
87 	NULL,
88 	NULL,
89 	NULL,					/* 0x0018 */
90 	NULL,
91 	NULL,
92 	NULL,
93 	NULL,
94 	NULL,
95 	NULL,
96 	NULL,
97 	"Invalid configuration action",		/* 0x0020 */
98 	"Invalid configuration type",
99 	"Invalid configuration page",
100 	"Invalid configuration data",
101 	"No configuration defaults",
102 	"Unable to commit configuration change",
103 	NULL,
104 	NULL,
105 	NULL,					/* 0x0028 */
106 	NULL,
107 	NULL,
108 	NULL,
109 	NULL,
110 	NULL,
111 	NULL,
112 	NULL,
113 	NULL,					/* 0x0030 */
114 	NULL,
115 	NULL,
116 	NULL,
117 	NULL,
118 	NULL,
119 	NULL,
120 	NULL,
121 	NULL,					/* 0x0038 */
122 	NULL,
123 	NULL,
124 	NULL,
125 	NULL,
126 	NULL,
127 	NULL,
128 	NULL,
129 	"Recovered SCSI error",			/* 0x0040 */
130 	"Invalid SCSI bus",
131 	"Invalid SCSI target ID",
132 	"SCSI device not there",
133 	"SCSI data overrun",
134 	"SCSI data underrun",
135 	"SCSI I/O error",
136 	"SCSI protocol error",
137 	"SCSI task terminated",			/* 0x0048 */
138 	"SCSI residual mismatch",
139 	"SCSI task management failed",
140 	"SCSI I/O controller terminated",
141 	"SCSI external controller terminated",
142 	"EEDP guard error",
143 	"EEDP reference tag error",
144 	"EEDP application tag error",
145 	NULL,					/* 0x0050 */
146 	NULL,
147 	NULL,
148 	NULL,
149 	NULL,
150 	NULL,
151 	NULL,
152 	NULL,
153 	NULL,					/* 0x0058 */
154 	NULL,
155 	NULL,
156 	NULL,
157 	NULL,
158 	NULL,
159 	NULL,
160 	NULL,
161 	"SCSI target priority I/O",		/* 0x0060 */
162 	"Invalid SCSI target port",
163 	"Invalid SCSI target I/O index",
164 	"SCSI target aborted",
165 	"No connection retryable",
166 	"No connection",
167 	"FC aborted",
168 	"Invalid FC receive ID",
169 	"FC did invalid",			/* 0x0068 */
170 	"FC node logged out",
171 	"Transfer count mismatch",
172 	"STS data not set",
173 	"FC exchange canceled",
174 	"Data offset error",
175 	"Too much write data",
176 	"IU too short",
177 	"ACK NAK timeout",			/* 0x0070 */
178 	"NAK received",
179 	NULL,
180 	NULL,
181 	NULL,
182 	NULL,
183 	NULL,
184 	NULL,
185 	NULL,					/* 0x0078 */
186 	NULL,
187 	NULL,
188 	NULL,
189 	NULL,
190 	NULL,
191 	NULL,
192 	NULL,
193 	"LAN device not found",			/* 0x0080 */
194 	"LAN device failure",
195 	"LAN transmit error",
196 	"LAN transmit aborted",
197 	"LAN receive error",
198 	"LAN receive aborted",
199 	"LAN partial packet",
200 	"LAN canceled",
201 	NULL,					/* 0x0088 */
202 	NULL,
203 	NULL,
204 	NULL,
205 	NULL,
206 	NULL,
207 	NULL,
208 	NULL,
209 	"SAS SMP request failed",		/* 0x0090 */
210 	"SAS SMP data overrun",
211 	NULL,
212 	NULL,
213 	NULL,
214 	NULL,
215 	NULL,
216 	NULL,
217 	"Inband aborted",			/* 0x0098 */
218 	"No inband connection",
219 	NULL,
220 	NULL,
221 	NULL,
222 	NULL,
223 	NULL,
224 	NULL,
225 	"Diagnostic released",			/* 0x00A0 */
226 };
227 
228 struct mprs_pass_thru {
229         uint64_t        PtrRequest;
230         uint64_t        PtrReply;
231         uint64_t        PtrData;
232         uint32_t        RequestSize;
233         uint32_t        ReplySize;
234         uint32_t        DataSize;
235         uint32_t        DataDirection;
236         uint64_t        PtrDataOut;
237         uint32_t        DataOutSize;
238         uint32_t        Timeout;
239 };
240 
241 struct mprs_btdh_mapping {
242         uint16_t        TargetID;
243         uint16_t        Bus;
244         uint16_t        DevHandle;
245         uint16_t        Reserved;
246 };
247 
248 const char *
249 mps_ioc_status(U16 IOCStatus)
250 {
251 	static char buffer[16];
252 
253 	IOCStatus &= MPI2_IOCSTATUS_MASK;
254 	if (IOCStatus < sizeof(mps_ioc_status_codes) / sizeof(char *) &&
255 	    mps_ioc_status_codes[IOCStatus] != NULL)
256 		return (mps_ioc_status_codes[IOCStatus]);
257 	snprintf(buffer, sizeof(buffer), "Status: 0x%04x", IOCStatus);
258 	return (buffer);
259 }
260 
261 #ifdef USE_MPT_IOCTLS
262 int
263 mps_map_btdh(int fd, uint16_t *devhandle, uint16_t *bus, uint16_t *target)
264 {
265 	int error;
266 	struct mprs_btdh_mapping map;
267 
268 	map.Bus = *bus;
269 	map.TargetID = *target;
270 	map.DevHandle = *devhandle;
271 
272 	if ((error = ioctl(fd, MPTIOCTL_BTDH_MAPPING, &map)) != 0) {
273 		error = errno;
274 		warn("Failed to map bus/target/device");
275 		return (error);
276 	}
277 
278 	*bus = map.Bus;
279 	*target = map.TargetID;
280 	*devhandle = map.DevHandle;
281 
282 	return (0);
283 }
284 
285 int
286 mps_set_slot_status(int fd, U16 handle, U16 slot, U32 status)
287 {
288 	MPI2_SEP_REQUEST req;
289 	MPI2_SEP_REPLY reply;
290 
291 	bzero(&req, sizeof(req));
292 	req.Function = MPI2_FUNCTION_SCSI_ENCLOSURE_PROCESSOR;
293 	req.Action = MPI2_SEP_REQ_ACTION_WRITE_STATUS;
294 	req.Flags = MPI2_SEP_REQ_FLAGS_ENCLOSURE_SLOT_ADDRESS;
295 	req.EnclosureHandle = handle;
296 	req.Slot = slot;
297 	req.SlotStatus = status;
298 
299 	if (mps_pass_command(fd, &req, sizeof(req), &reply, sizeof(reply),
300 	    NULL, 0, NULL, 0, 30) != 0)
301 		return (errno);
302 
303 	if (!IOC_STATUS_SUCCESS(reply.IOCStatus))
304 		return (EIO);
305 	return (0);
306 }
307 
308 int
309 mps_read_config_page_header(int fd, U8 PageType, U8 PageNumber, U32 PageAddress,
310     MPI2_CONFIG_PAGE_HEADER *header, U16 *IOCStatus)
311 {
312 	MPI2_CONFIG_REQUEST req;
313 	MPI2_CONFIG_REPLY reply;
314 
315 	bzero(&req, sizeof(req));
316 	req.Function = MPI2_FUNCTION_CONFIG;
317 	req.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
318 	req.Header.PageType = PageType;
319 	req.Header.PageNumber = PageNumber;
320 	req.PageAddress = PageAddress;
321 
322 	if (mps_pass_command(fd, &req, sizeof(req), &reply, sizeof(reply),
323 	    NULL, 0, NULL, 0, 30))
324 		return (errno);
325 
326 	if (!IOC_STATUS_SUCCESS(reply.IOCStatus)) {
327 		if (IOCStatus != NULL)
328 			*IOCStatus = reply.IOCStatus;
329 		return (EIO);
330 	}
331 	if (header == NULL)
332 		return (EINVAL);
333 	*header = reply.Header;
334 	return (0);
335 }
336 
337 int
338 mps_read_ext_config_page_header(int fd, U8 ExtPageType, U8 PageNumber, U32 PageAddress, MPI2_CONFIG_PAGE_HEADER *header, U16 *ExtPageLength, U16 *IOCStatus)
339 {
340 	MPI2_CONFIG_REQUEST req;
341 	MPI2_CONFIG_REPLY reply;
342 
343 	bzero(&req, sizeof(req));
344 	req.Function = MPI2_FUNCTION_CONFIG;
345 	req.Action = MPI2_CONFIG_ACTION_PAGE_HEADER;
346 	req.Header.PageType = MPI2_CONFIG_PAGETYPE_EXTENDED;
347 	req.ExtPageType = ExtPageType;
348 	req.Header.PageNumber = PageNumber;
349 	req.PageAddress = PageAddress;
350 
351 	if (mps_pass_command(fd, &req, sizeof(req), &reply, sizeof(reply),
352 	    NULL, 0, NULL, 0, 30))
353 		return (errno);
354 
355 	if (!IOC_STATUS_SUCCESS(reply.IOCStatus)) {
356 		if (IOCStatus != NULL)
357 			*IOCStatus = reply.IOCStatus;
358 		return (EIO);
359 	}
360 	if ((header == NULL) || (ExtPageLength == NULL))
361 		return (EINVAL);
362 	*header = reply.Header;
363 	*ExtPageLength = reply.ExtPageLength;
364 	return (0);
365 }
366 
367 void *
368 mps_read_config_page(int fd, U8 PageType, U8 PageNumber, U32 PageAddress,
369     U16 *IOCStatus)
370 {
371 	MPI2_CONFIG_REQUEST req;
372 	MPI2_CONFIG_PAGE_HEADER header;
373 	MPI2_CONFIG_REPLY reply;
374 	void *buf;
375 	int error, len;
376 
377 	bzero(&header, sizeof(header));
378 	error = mps_read_config_page_header(fd, PageType, PageNumber,
379 	    PageAddress, &header, IOCStatus);
380 	if (error) {
381 		errno = error;
382 		return (NULL);
383 	}
384 
385 	bzero(&req, sizeof(req));
386 	req.Function = MPI2_FUNCTION_CONFIG;
387 	req.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
388 	req.PageAddress = PageAddress;
389 	req.Header = header;
390 	if (req.Header.PageLength == 0)
391 		req.Header.PageLength = 4;
392 
393 	len = req.Header.PageLength * 4;
394 	buf = malloc(len);
395 	if (mps_pass_command(fd, &req, sizeof(req), &reply, sizeof(reply),
396 	    buf, len, NULL, 0, 30)) {
397 		error = errno;
398 		free(buf);
399 		errno = error;
400 		return (NULL);
401 	}
402 	if (!IOC_STATUS_SUCCESS(reply.IOCStatus)) {
403 		if (IOCStatus != NULL)
404 			*IOCStatus = reply.IOCStatus;
405 		else
406 			warnx("Reading config page failed: 0x%x %s",
407 			    reply.IOCStatus, mps_ioc_status(reply.IOCStatus));
408 		free(buf);
409 		errno = EIO;
410 		return (NULL);
411 	}
412 	return (buf);
413 }
414 
415 void *
416 mps_read_extended_config_page(int fd, U8 ExtPageType, U8 PageVersion,
417     U8 PageNumber, U32 PageAddress, U16 *IOCStatus)
418 {
419 	MPI2_CONFIG_REQUEST req;
420 	MPI2_CONFIG_PAGE_HEADER header;
421 	MPI2_CONFIG_REPLY reply;
422 	U16 pagelen;
423 	void *buf;
424 	int error, len;
425 
426 	if (IOCStatus != NULL)
427 		*IOCStatus = MPI2_IOCSTATUS_SUCCESS;
428 	bzero(&header, sizeof(header));
429 	error = mps_read_ext_config_page_header(fd, ExtPageType, PageNumber,
430 	    PageAddress, &header, &pagelen, IOCStatus);
431 	if (error) {
432 		errno = error;
433 		return (NULL);
434 	}
435 
436 	bzero(&req, sizeof(req));
437 	req.Function = MPI2_FUNCTION_CONFIG;
438 	req.Action = MPI2_CONFIG_ACTION_PAGE_READ_CURRENT;
439 	req.PageAddress = PageAddress;
440 	req.Header = header;
441 	if (pagelen == 0)
442 		pagelen = 4;
443 	req.ExtPageLength = pagelen;
444 	req.ExtPageType = ExtPageType;
445 
446 	len = pagelen * 4;
447 	buf = malloc(len);
448 	if (mps_pass_command(fd, &req, sizeof(req), &reply, sizeof(reply),
449 	    buf, len, NULL, 0, 30)) {
450 		error = errno;
451 		free(buf);
452 		errno = error;
453 		return (NULL);
454 	}
455 	if (!IOC_STATUS_SUCCESS(reply.IOCStatus)) {
456 		if (IOCStatus != NULL)
457 			*IOCStatus = reply.IOCStatus;
458 		else
459 			warnx("Reading extended config page failed: %s",
460 			    mps_ioc_status(reply.IOCStatus));
461 		free(buf);
462 		errno = EIO;
463 		return (NULL);
464 	}
465 	return (buf);
466 }
467 
468 int
469 mps_firmware_send(int fd, unsigned char *fw, uint32_t len, bool bios)
470 {
471 	MPI2_FW_DOWNLOAD_REQUEST req;
472 	MPI2_FW_DOWNLOAD_REPLY reply;
473 
474 	bzero(&req, sizeof(req));
475 	bzero(&reply, sizeof(reply));
476 	req.Function = MPI2_FUNCTION_FW_DOWNLOAD;
477 	req.ImageType = bios ? MPI2_FW_DOWNLOAD_ITYPE_BIOS : MPI2_FW_DOWNLOAD_ITYPE_FW;
478 	req.TotalImageSize = len;
479 	req.MsgFlags = MPI2_FW_DOWNLOAD_MSGFLGS_LAST_SEGMENT;
480 
481 	if (mps_user_command(fd, &req, sizeof(req), &reply, sizeof(reply),
482 	    fw, len, 0)) {
483 		return (-1);
484 	}
485 	return (0);
486 }
487 
488 int
489 mps_firmware_get(int fd, unsigned char **firmware, bool bios)
490 {
491 	MPI2_FW_UPLOAD_REQUEST req;
492 	MPI2_FW_UPLOAD_REPLY reply;
493 	int size;
494 
495 	*firmware = NULL;
496 	bzero(&req, sizeof(req));
497 	bzero(&reply, sizeof(reply));
498 	req.Function = MPI2_FUNCTION_FW_UPLOAD;
499 	req.ImageType = bios ? MPI2_FW_DOWNLOAD_ITYPE_BIOS : MPI2_FW_DOWNLOAD_ITYPE_FW;
500 
501 	if (mps_user_command(fd, &req, sizeof(req), &reply, sizeof(reply),
502 	    NULL, 0, 0)) {
503 		return (-1);
504 	}
505 	if (reply.ActualImageSize == 0) {
506 		return (-1);
507 	}
508 
509 	size = reply.ActualImageSize;
510 	*firmware = calloc(size, sizeof(unsigned char));
511 	if (*firmware == NULL) {
512 		warn("calloc");
513 		return (-1);
514 	}
515 	if (mps_user_command(fd, &req, sizeof(req), &reply, sizeof(reply),
516 	    *firmware, size, 0)) {
517 		free(*firmware);
518 		return (-1);
519 	}
520 
521 	return (size);
522 }
523 
524 #else
525 
526 int
527 mps_read_config_page_header(int fd, U8 PageType, U8 PageNumber, U32 PageAddress,
528     MPI2_CONFIG_PAGE_HEADER *header, U16 *IOCStatus)
529 {
530 	struct mps_cfg_page_req req;
531 
532 	if (IOCStatus != NULL)
533 		*IOCStatus = MPI2_IOCSTATUS_SUCCESS;
534 	if (header == NULL)
535 		return (EINVAL);
536 	bzero(&req, sizeof(req));
537 	req.header.PageType = PageType;
538 	req.header.PageNumber = PageNumber;
539 	req.page_address = PageAddress;
540 	if (ioctl(fd, MPSIO_READ_CFG_HEADER, &req) < 0)
541 		return (errno);
542 	if (!IOC_STATUS_SUCCESS(req.ioc_status)) {
543 		if (IOCStatus != NULL)
544 			*IOCStatus = req.ioc_status;
545 		return (EIO);
546 	}
547 	bcopy(&req.header, header, sizeof(*header));
548 	return (0);
549 }
550 
551 void *
552 mps_read_config_page(int fd, U8 PageType, U8 PageNumber, U32 PageAddress,
553     U16 *IOCStatus)
554 {
555 	struct mps_cfg_page_req req;
556 	void *buf;
557 	int error;
558 
559 	error = mps_read_config_page_header(fd, PageType, PageNumber,
560 	    PageAddress, &req.header, IOCStatus);
561 	if (error) {
562 		errno = error;
563 		return (NULL);
564 	}
565 
566 	if (req.header.PageLength == 0)
567 		req.header.PageLength = 4;
568 	req.len = req.header.PageLength * 4;
569 	buf = malloc(req.len);
570 	req.buf = buf;
571 	bcopy(&req.header, buf, sizeof(req.header));
572 	if (ioctl(fd, MPSIO_READ_CFG_PAGE, &req) < 0) {
573 		error = errno;
574 		free(buf);
575 		errno = error;
576 		return (NULL);
577 	}
578 	if (!IOC_STATUS_SUCCESS(req.ioc_status)) {
579 		if (IOCStatus != NULL)
580 			*IOCStatus = req.ioc_status;
581 		else
582 			warnx("Reading config page failed: 0x%x %s",
583 			    req.ioc_status, mps_ioc_status(req.ioc_status));
584 		free(buf);
585 		errno = EIO;
586 		return (NULL);
587 	}
588 	return (buf);
589 }
590 
591 void *
592 mps_read_extended_config_page(int fd, U8 ExtPageType, U8 PageVersion,
593     U8 PageNumber, U32 PageAddress, U16 *IOCStatus)
594 {
595 	struct mps_ext_cfg_page_req req;
596 	void *buf;
597 	int error;
598 
599 	if (IOCStatus != NULL)
600 		*IOCStatus = MPI2_IOCSTATUS_SUCCESS;
601 	bzero(&req, sizeof(req));
602 	req.header.PageVersion = PageVersion;
603 	req.header.PageNumber = PageNumber;
604 	req.header.ExtPageType = ExtPageType;
605 	req.page_address = PageAddress;
606 	if (ioctl(fd, MPSIO_READ_EXT_CFG_HEADER, &req) < 0)
607 		return (NULL);
608 	if (!IOC_STATUS_SUCCESS(req.ioc_status)) {
609 		if (IOCStatus != NULL)
610 			*IOCStatus = req.ioc_status;
611 		else
612 			warnx("Reading extended config page header failed: %s",
613 			    mps_ioc_status(req.ioc_status));
614 		errno = EIO;
615 		return (NULL);
616 	}
617 	req.len = req.header.ExtPageLength * 4;
618 	buf = malloc(req.len);
619 	req.buf = buf;
620 	bcopy(&req.header, buf, sizeof(req.header));
621 	if (ioctl(fd, MPSIO_READ_EXT_CFG_PAGE, &req) < 0) {
622 		error = errno;
623 		free(buf);
624 		errno = error;
625 		return (NULL);
626 	}
627 	if (!IOC_STATUS_SUCCESS(req.ioc_status)) {
628 		if (IOCStatus != NULL)
629 			*IOCStatus = req.ioc_status;
630 		else
631 			warnx("Reading extended config page failed: %s",
632 			    mps_ioc_status(req.ioc_status));
633 		free(buf);
634 		errno = EIO;
635 		return (NULL);
636 	}
637 	return (buf);
638 }
639 #endif
640 
641 int
642 mps_open(int unit)
643 {
644 	char path[MAXPATHLEN];
645 
646 	snprintf(path, sizeof(path), "/dev/mp%s%d", is_mps ? "s": "r", unit);
647 	return (open(path, O_RDWR));
648 }
649 
650 int
651 mps_user_command(int fd, void *req, uint32_t req_len, void *reply,
652         uint32_t reply_len, void *buffer, int len, uint32_t flags)
653 {
654 	struct mps_usr_command cmd;
655 
656 	bzero(&cmd, sizeof(struct mps_usr_command));
657 	cmd.req = req;
658 	cmd.req_len = req_len;
659 	cmd.rpl = reply;
660 	cmd.rpl_len = reply_len;
661 	cmd.buf = buffer;
662 	cmd.len = len;
663 	cmd.flags = flags;
664 
665 	if (ioctl(fd, is_mps ? MPSIO_MPS_COMMAND : MPRIO_MPR_COMMAND, &cmd) < 0)
666 		return (errno);
667 	return (0);
668 }
669 
670 int
671 mps_pass_command(int fd, void *req, uint32_t req_len, void *reply,
672 	uint32_t reply_len, void *data_in, uint32_t datain_len, void *data_out,
673 	uint32_t dataout_len, uint32_t timeout)
674 {
675 	struct mprs_pass_thru pass;
676 
677 	bzero(&pass, sizeof(pass));
678 	pass.PtrRequest = (uint64_t)(uintptr_t)req;
679 	pass.PtrReply = (uint64_t)(uintptr_t)reply;
680 	pass.RequestSize = req_len;
681 	pass.ReplySize = reply_len;
682 	if (datain_len && dataout_len) {
683 		pass.PtrData = (uint64_t)(uintptr_t)data_in;
684 		pass.PtrDataOut = (uint64_t)(uintptr_t)data_out;
685 		pass.DataSize = datain_len;
686 		pass.DataOutSize = dataout_len;
687 		if (is_mps) {
688 			pass.DataDirection = MPS_PASS_THRU_DIRECTION_BOTH;
689 		} else {
690 			pass.DataDirection = MPR_PASS_THRU_DIRECTION_BOTH;
691 		}
692 	} else if (datain_len) {
693 		pass.PtrData = (uint64_t)(uintptr_t)data_in;
694 		pass.DataSize = datain_len;
695 		if (is_mps) {
696 			pass.DataDirection = MPS_PASS_THRU_DIRECTION_READ;
697 		} else {
698 			pass.DataDirection = MPR_PASS_THRU_DIRECTION_READ;
699 		}
700 	} else if (dataout_len) {
701 		pass.PtrData = (uint64_t)(uintptr_t)data_out;
702 		pass.DataSize = dataout_len;
703 		if (is_mps) {
704 			pass.DataDirection = MPS_PASS_THRU_DIRECTION_WRITE;
705 		} else {
706 			pass.DataDirection = MPR_PASS_THRU_DIRECTION_WRITE;
707 		}
708 	} else {
709 		if (is_mps) {
710 			pass.DataDirection = MPS_PASS_THRU_DIRECTION_NONE;
711 		} else {
712 			pass.DataDirection = MPR_PASS_THRU_DIRECTION_NONE;
713 		}
714 	}
715 	pass.Timeout = timeout;
716 
717 	if (ioctl(fd, MPTIOCTL_PASS_THRU, &pass) < 0)
718 		return (errno);
719 	return (0);
720 }
721 
722 MPI2_IOC_FACTS_REPLY *
723 mps_get_iocfacts(int fd)
724 {
725 	MPI2_IOC_FACTS_REPLY *facts;
726 	MPI2_IOC_FACTS_REQUEST req;
727 	char msgver[8], sysctlname[128];
728 	size_t len, factslen;
729 	int error;
730 
731 	snprintf(sysctlname, sizeof(sysctlname), "dev.%s.%d.msg_version",
732 	    is_mps ? "mps" : "mpr", mps_unit);
733 
734 	factslen = sizeof(MPI2_IOC_FACTS_REPLY);
735 	len = sizeof(msgver);
736 	error = sysctlbyname(sysctlname, msgver, &len, NULL, 0);
737 	if (error == 0) {
738 		if (strncmp(msgver, "2.6", sizeof(msgver)) == 0)
739 			factslen += 4;
740 	}
741 
742 	facts = malloc(factslen);
743 	if (facts == NULL) {
744 		errno = ENOMEM;
745 		return (NULL);
746 	}
747 
748 	bzero(&req, factslen);
749 	req.Function = MPI2_FUNCTION_IOC_FACTS;
750 
751 #if 1
752 	error = mps_pass_command(fd, &req, sizeof(MPI2_IOC_FACTS_REQUEST),
753 	    facts, factslen, NULL, 0, NULL, 0, 10);
754 #else
755 	error = mps_user_command(fd, &req, sizeof(MPI2_IOC_FACTS_REQUEST),
756 	    facts, factslen, NULL, 0, 0);
757 #endif
758 	if (error) {
759 		free(facts);
760 		return (NULL);
761 	}
762 
763 	if (!IOC_STATUS_SUCCESS(facts->IOCStatus)) {
764 		free(facts);
765 		errno = EINVAL;
766 		return (NULL);
767 	}
768 	return (facts);
769 }
770 
771