xref: /illumos-gate/usr/src/lib/fm/libdiskstatus/common/ds_scsi.c (revision f3af49816e370d667d566ab703e94b81305a536e)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2007 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 #pragma ident	"%Z%%M%	%I%	%E% SMI"
27 
28 #include <assert.h>
29 #include <errno.h>
30 #include <libdiskstatus.h>
31 #include <limits.h>
32 #include <stdlib.h>
33 #include <strings.h>
34 #include <sys/fm/io/scsi.h>
35 
36 #include "ds_scsi.h"
37 #include "ds_scsi_sim.h"
38 #include "ds_scsi_uscsi.h"
39 
40 typedef struct ds_scsi_info {
41 	disk_status_t		*si_dsp;
42 	void			*si_sim;
43 	int			si_cdblen;
44 	int			si_supp_mode;
45 	int			si_supp_log;
46 	int			si_extensions;
47 	int			si_reftemp;
48 	scsi_ms_hdrs_t		si_hdrs;
49 	scsi_ie_page_t		si_iec_current;
50 	scsi_ie_page_t		si_iec_changeable;
51 	nvlist_t		*si_state_modepage;
52 	nvlist_t		*si_state_logpage;
53 	nvlist_t		*si_state_iec;
54 } ds_scsi_info_t;
55 
56 #define	scsi_set_errno(sip, errno)	(ds_set_errno((sip)->si_dsp, (errno)))
57 
58 /*
59  * Table to validate log pages
60  */
61 typedef int (*logpage_validation_fn_t)(ds_scsi_info_t *,
62     scsi_log_parameter_header_t *, int, nvlist_t *);
63 typedef int (*logpage_analyze_fn_t)(ds_scsi_info_t *,
64     scsi_log_parameter_header_t *, int);
65 
66 typedef struct logpage_validation_entry {
67 	uchar_t			ve_code;
68 	int			ve_supported;
69 	const char		*ve_desc;
70 	logpage_validation_fn_t	ve_validate;
71 	logpage_analyze_fn_t	ve_analyze;
72 } logpage_validation_entry_t;
73 
74 static int logpage_ie_verify(ds_scsi_info_t *,
75     scsi_log_parameter_header_t *, int, nvlist_t *);
76 static int logpage_temp_verify(ds_scsi_info_t *,
77     scsi_log_parameter_header_t *, int, nvlist_t *);
78 static int logpage_selftest_verify(ds_scsi_info_t *,
79     scsi_log_parameter_header_t *, int, nvlist_t *);
80 
81 static int logpage_ie_analyze(ds_scsi_info_t *,
82     scsi_log_parameter_header_t *, int);
83 static int logpage_temp_analyze(ds_scsi_info_t *,
84     scsi_log_parameter_header_t *, int);
85 static int logpage_selftest_analyze(ds_scsi_info_t *,
86     scsi_log_parameter_header_t *, int);
87 
88 static struct logpage_validation_entry log_validation[] = {
89 	{ LOGPAGE_IE,		LOGPAGE_SUPP_IE,
90 	    "informational-exceptions",
91 	    logpage_ie_verify,	logpage_ie_analyze },
92 	{ LOGPAGE_TEMP,		LOGPAGE_SUPP_TEMP,
93 	    "temperature",
94 	    logpage_temp_verify, logpage_temp_analyze },
95 	{ LOGPAGE_SELFTEST,	LOGPAGE_SUPP_SELFTEST,
96 	    "self-test",
97 	    logpage_selftest_verify, logpage_selftest_analyze }
98 };
99 
100 #define	NLOG_VALIDATION	(sizeof (log_validation) / sizeof (log_validation[0]))
101 
102 /*
103  * Given an extended sense page, retrieves the sense key, as well as the
104  * additional sense code information.
105  */
106 static void
107 scsi_translate_error(struct scsi_extended_sense *rq, uint_t *skeyp,
108     uint_t *ascp, uint_t *ascqp)
109 {
110 	struct scsi_descr_sense_hdr *sdsp =
111 	    (struct scsi_descr_sense_hdr *)rq;
112 
113 	*skeyp = rq->es_key;
114 
115 	/*
116 	 * Get asc, ascq and info field from sense data.  There are two
117 	 * possible formats (fixed sense data and descriptor sense data)
118 	 * depending on the value of es_code.
119 	 */
120 	switch (rq->es_code) {
121 	case CODE_FMT_DESCR_CURRENT:
122 	case CODE_FMT_DESCR_DEFERRED:
123 
124 		*ascp = sdsp->ds_add_code;
125 		*ascqp = sdsp->ds_qual_code;
126 		break;
127 
128 	case CODE_FMT_FIXED_CURRENT:
129 	case CODE_FMT_FIXED_DEFERRED:
130 	default:
131 
132 		if (rq->es_add_len >= 6) {
133 			*ascp = rq->es_add_code;
134 			*ascqp = rq->es_qual_code;
135 		} else {
136 			*ascp = 0xff;
137 			*ascqp = 0xff;
138 		}
139 		break;
140 	}
141 }
142 
143 /*
144  * Routines built atop the bare uscsi commands, which take into account the
145  * command length, automatically translate any scsi errors, and transparently
146  * call into the simulator if active.
147  */
148 static int
149 scsi_mode_select(ds_scsi_info_t *sip, uchar_t page_code, int options,
150     void *buf, uint_t buflen, scsi_ms_hdrs_t *headers, uint_t *skp,
151     uint_t *ascp, uint_t *ascqp)
152 {
153 	int result;
154 	struct scsi_extended_sense sense;
155 	int senselen = sizeof (struct scsi_extended_sense);
156 	struct mode_page *mp = (struct mode_page *)buf;
157 
158 	assert(sip->si_cdblen == MODE_CMD_LEN_6 ||
159 	    sip->si_cdblen == MODE_CMD_LEN_10);
160 	assert(headers->ms_length == sip->si_cdblen);
161 
162 	bzero(&sense, sizeof (struct scsi_extended_sense));
163 
164 	if (mp->ps) {
165 		options |= MODE_SELECT_SP;
166 		mp->ps = 0;
167 	} else {
168 		options &= ~MODE_SELECT_SP;
169 	}
170 
171 	if (sip->si_cdblen == MODE_CMD_LEN_6) {
172 		/* The following fields are reserved during mode select: */
173 		headers->ms_hdr.g0.ms_header.length = 0;
174 		headers->ms_hdr.g0.ms_header.device_specific = 0;
175 
176 		if (sip->si_sim)
177 			result = simscsi_mode_select(sip->si_sim,
178 			    page_code, options, buf, buflen,
179 			    &headers->ms_hdr.g0, &sense, &senselen);
180 		else
181 			result = uscsi_mode_select(sip->si_dsp->ds_fd,
182 			    page_code, options, buf, buflen,
183 			    &headers->ms_hdr.g0, &sense, &senselen);
184 	} else {
185 		/* The following fields are reserved during mode select: */
186 		headers->ms_hdr.g1.ms_header.length = 0;
187 		headers->ms_hdr.g1.ms_header.device_specific = 0;
188 
189 		if (sip->si_sim)
190 			result = simscsi_mode_select_10(sip->si_sim,
191 			    page_code, options, buf, buflen,
192 			    &headers->ms_hdr.g1, &sense, &senselen);
193 		else
194 			result = uscsi_mode_select_10(sip->si_dsp->ds_fd,
195 			    page_code, options, buf, buflen,
196 			    &headers->ms_hdr.g1, &sense, &senselen);
197 	}
198 
199 	if (result != 0)
200 		scsi_translate_error(&sense, skp, ascp, ascqp);
201 
202 	return (result);
203 }
204 
205 static int
206 scsi_mode_sense(ds_scsi_info_t *sip, uchar_t page_code, uchar_t pc,
207     void *buf, uint_t buflen, scsi_ms_hdrs_t *headers, uint_t *skp,
208     uint_t *ascp, uint_t *ascqp)
209 {
210 	int result;
211 	struct scsi_extended_sense sense;
212 	int senselen = sizeof (struct scsi_extended_sense);
213 
214 	assert(sip->si_cdblen == MODE_CMD_LEN_6 ||
215 	    sip->si_cdblen == MODE_CMD_LEN_10);
216 
217 	bzero(&sense, sizeof (struct scsi_extended_sense));
218 
219 	bzero(headers, sizeof (scsi_ms_hdrs_t));
220 	headers->ms_length = sip->si_cdblen;
221 
222 	if (sip->si_cdblen == MODE_CMD_LEN_6) {
223 		if (sip->si_sim)
224 			result = simscsi_mode_sense(sip->si_sim,
225 			    page_code, pc, buf, buflen, &headers->ms_hdr.g0,
226 			    &sense, &senselen);
227 		else
228 			result = uscsi_mode_sense(sip->si_dsp->ds_fd, page_code,
229 			    pc, buf, buflen, &headers->ms_hdr.g0, &sense,
230 			    &senselen);
231 	} else {
232 		if (sip->si_sim)
233 			result = simscsi_mode_sense_10(sip->si_sim,
234 			    page_code, pc, buf, buflen, &headers->ms_hdr.g1,
235 			    &sense, &senselen);
236 		else
237 			result = uscsi_mode_sense_10(sip->si_dsp->ds_fd,
238 			    page_code, pc, buf, buflen, &headers->ms_hdr.g1,
239 			    &sense, &senselen);
240 	}
241 
242 	if (result != 0)
243 		scsi_translate_error(&sense, skp, ascp, ascqp);
244 
245 	return (result);
246 }
247 
248 static int
249 scsi_request_sense(ds_scsi_info_t *sip, uint_t *skp, uint_t *ascp,
250     uint_t *ascqp)
251 {
252 	struct scsi_extended_sense sense, sensebuf;
253 	int senselen = sizeof (struct scsi_extended_sense);
254 	int sensebuflen = sizeof (struct scsi_extended_sense);
255 	int result;
256 
257 	bzero(&sense, sizeof (struct scsi_extended_sense));
258 	bzero(&sensebuf, sizeof (struct scsi_extended_sense));
259 
260 	if (sip->si_sim)
261 		result = simscsi_request_sense(sip->si_sim,
262 		    (caddr_t)&sensebuf, sensebuflen, &sense, &senselen);
263 	else
264 		result = uscsi_request_sense(sip->si_dsp->ds_fd,
265 		    (caddr_t)&sensebuf, sensebuflen, &sense, &senselen);
266 
267 	if (result == 0)
268 		scsi_translate_error(&sensebuf, skp, ascp, ascqp);
269 	else
270 		scsi_translate_error(&sense, skp, ascp, ascqp);
271 
272 	return (result);
273 }
274 
275 static int
276 scsi_log_sense(ds_scsi_info_t *sip, int page_code, int page_control,
277     caddr_t page_data, int page_size, uint_t *skp, uint_t *ascp, uint_t *ascqp)
278 {
279 	int result;
280 	struct scsi_extended_sense sense;
281 	int senselen = sizeof (struct scsi_extended_sense);
282 
283 	if (sip->si_sim)
284 		result = simscsi_log_sense(sip->si_sim,
285 		    page_code, page_control, page_data, page_size, &sense,
286 		    &senselen);
287 	else
288 		result = uscsi_log_sense(sip->si_dsp->ds_fd,
289 		    page_code, page_control, page_data, page_size, &sense,
290 		    &senselen);
291 
292 	if (result != 0)
293 		scsi_translate_error(&sense, skp, ascp, ascqp);
294 
295 	return (result);
296 }
297 
298 /*
299  * Given a list of supported mode pages, determine if the given page is present.
300  */
301 static boolean_t
302 mode_page_present(uchar_t *pgdata, uint_t pgdatalen, uchar_t pagecode)
303 {
304 	uint_t i = 0;
305 	struct mode_page *pg;
306 	boolean_t found = B_FALSE;
307 
308 	/*
309 	 * The mode page list contains all mode pages supported by the device,
310 	 * one after the other.
311 	 */
312 	while (i < pgdatalen) {
313 		pg = (struct mode_page *)&pgdata[i];
314 
315 		if (pg->code == pagecode) {
316 			found = B_TRUE;
317 			break;
318 		}
319 
320 		i += MODESENSE_PAGE_LEN(pg);
321 	}
322 
323 	return (found);
324 }
325 
326 /*
327  * Load mode pages and check that the appropriate pages are supported.
328  *
329  * As part of this process, we determine which form of the MODE SENSE / MODE
330  * SELECT command to use (the 6-byte or 10-byte version) by executing a MODE
331  * SENSE command for a page that should be implemented by the device.
332  */
333 static int
334 load_modepages(ds_scsi_info_t *sip)
335 {
336 	int allpages_buflen;
337 	uchar_t *allpages;
338 	scsi_ms_hdrs_t headers;
339 	int result;
340 	uint_t skey, asc, ascq;
341 	int datalength = 0;
342 	scsi_ms_header_t *smh = &headers.ms_hdr.g0;
343 	scsi_ms_header_g1_t *smh_g1 = &headers.ms_hdr.g1;
344 	nvlist_t *nvl;
345 
346 	allpages_buflen = MAX_BUFLEN(scsi_ms_header_g1_t);
347 	if ((allpages = calloc(allpages_buflen, 1)) == NULL)
348 		return (scsi_set_errno(sip, EDS_NOMEM));
349 
350 	bzero(&headers, sizeof (headers));
351 
352 	/*
353 	 * Attempt a mode sense(6).  If that fails, try a mode sense(10)
354 	 *
355 	 * allpages is allocated to be of the maximum size for either a mode
356 	 * sense(6) or mode sense(10) MODEPAGE_ALLPAGES response.
357 	 *
358 	 * Note that the length passed into uscsi_mode_sense should be set to
359 	 * the maximum size of the parameter response, which in this case is
360 	 * UCHAR_MAX - the size of the headers/block descriptors.
361 	 */
362 	sip->si_cdblen = MODE_CMD_LEN_6;
363 	if ((result = scsi_mode_sense(sip, MODEPAGE_ALLPAGES, PC_CURRENT,
364 	    (caddr_t)allpages, UCHAR_MAX - sizeof (scsi_ms_header_t),
365 	    &headers, &skey, &asc, &ascq)) == 0) {
366 		/*
367 		 * Compute the data length of the page that contains all mode
368 		 * sense pages.  This is a bit tricky because the format of the
369 		 * response from the lun is:
370 		 *
371 		 * header: <length> <medium type byte> <dev specific byte>
372 		 *	   <block descriptor length>
373 		 *	   [<optional block descriptor>]
374 		 * data:   [<mode page data> <mode page data> ...]
375 		 *
376 		 * Since the length field in the header describes the length of
377 		 * the entire response.  This includes the header, but NOT
378 		 * the length field itself, which is 1 or 2 bytes depending on
379 		 * which mode sense type (6- or 10- byte) is being executed.
380 		 *
381 		 * So, the data length equals the length value in the header
382 		 * plus 1 (because the length byte was not included in the
383 		 * length count), minus [[the sum of the length of the header
384 		 * and the length of the block descriptor]].
385 		 */
386 		datalength = (smh->ms_header.length +
387 		    sizeof (smh->ms_header.length)) -
388 		    (sizeof (struct mode_header) +
389 		    smh->ms_header.bdesc_length);
390 	} else if (SCSI_INVALID_OPCODE(skey, asc, ascq)) {
391 		/*
392 		 * Fallback and try the 10-byte version of the command.
393 		 */
394 		sip->si_cdblen = MODE_CMD_LEN_10;
395 		result = scsi_mode_sense(sip, MODEPAGE_ALLPAGES,
396 		    PC_CURRENT, (caddr_t)allpages, allpages_buflen,
397 		    &headers, &skey, &asc, &ascq);
398 
399 		if (result == 0) {
400 			datalength = (BE_16(smh_g1->ms_header.length) +
401 			    sizeof (smh_g1->ms_header.length)) -
402 			    (sizeof (struct mode_header_g1) +
403 			    BE_16(smh_g1->ms_header.bdesc_length));
404 
405 		}
406 	}
407 
408 	if (result == 0 && datalength >= 0) {
409 		if (nvlist_add_int8(sip->si_dsp->ds_state, "command-length",
410 		    sip->si_cdblen == MODE_CMD_LEN_6 ? 6 : 10) != 0) {
411 			free(allpages);
412 			return (scsi_set_errno(sip, EDS_NOMEM));
413 		}
414 
415 		nvl = NULL;
416 		if (nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0) != 0 ||
417 		    nvlist_add_nvlist(sip->si_dsp->ds_state, "modepages",
418 		    nvl) != 0) {
419 			free(allpages);
420 			nvlist_free(nvl);
421 			return (scsi_set_errno(sip, EDS_NOMEM));
422 		}
423 
424 		nvlist_free(nvl);
425 		result = nvlist_lookup_nvlist(sip->si_dsp->ds_state,
426 		    "modepages", &sip->si_state_modepage);
427 		assert(result == 0);
428 
429 		/*
430 		 * One of the sets of the commands (above) succeeded, so now
431 		 * look for the mode pages we need and record them appropriately
432 		 */
433 		if (mode_page_present(allpages, datalength,
434 		    MODEPAGE_INFO_EXCPT)) {
435 
436 			nvl = NULL;
437 			if (nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0) != 0 ||
438 			    nvlist_add_nvlist(sip->si_state_modepage,
439 			    "informational-exceptions", nvl) != 0) {
440 				free(allpages);
441 				nvlist_free(nvl);
442 				return (scsi_set_errno(sip, EDS_NOMEM));
443 			}
444 			nvlist_free(nvl);
445 			sip->si_supp_mode |= MODEPAGE_SUPP_IEC;
446 			result = nvlist_lookup_nvlist(sip->si_state_modepage,
447 			    "informational-exceptions", &sip->si_state_iec);
448 			assert(result == 0);
449 		}
450 
451 	} else {
452 		/*
453 		 * If the device failed to respond to one of the basic commands,
454 		 * then assume it's not a SCSI device or otherwise doesn't
455 		 * support the necessary transport.
456 		 */
457 		if (datalength < 0)
458 			dprintf("command returned invalid data length (%d)\n",
459 			    datalength);
460 		else
461 			dprintf("failed to load modepages (KEY=0x%x "
462 			    "ASC=0x%x ASCQ=0x%x)\n", skey, asc, ascq);
463 
464 		result = scsi_set_errno(sip, EDS_NO_TRANSPORT);
465 	}
466 
467 	free(allpages);
468 	return (result);
469 }
470 
471 /*
472  * Verify a single logpage.  This will do some generic validation and then call
473  * the logpage-specific function for further verification.
474  */
475 static int
476 verify_logpage(ds_scsi_info_t *sip, logpage_validation_entry_t *lp)
477 {
478 	scsi_log_header_t *lhp;
479 	struct scsi_extended_sense sense;
480 	int buflen;
481 	int log_length;
482 	int result = 0;
483 	uint_t kp, asc, ascq;
484 	nvlist_t *nvl;
485 
486 	buflen = MAX_BUFLEN(scsi_log_header_t);
487 	if ((lhp = calloc(buflen, 1)) == NULL)
488 		return (scsi_set_errno(sip, EDS_NOMEM));
489 	bzero(&sense, sizeof (struct scsi_extended_sense));
490 
491 	nvl = NULL;
492 	if (nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0) != 0 ||
493 	    nvlist_add_nvlist(sip->si_state_logpage, lp->ve_desc, nvl) != 0) {
494 		nvlist_free(nvl);
495 		free(lhp);
496 		return (scsi_set_errno(sip, EDS_NOMEM));
497 	}
498 	nvlist_free(nvl);
499 	result = nvlist_lookup_nvlist(sip->si_state_logpage, lp->ve_desc, &nvl);
500 	assert(result == 0);
501 
502 	result = scsi_log_sense(sip, lp->ve_code,
503 	    PC_CUMULATIVE, (caddr_t)lhp, buflen, &kp, &asc, &ascq);
504 
505 	if (result == 0) {
506 		log_length = BE_16(lhp->lh_length);
507 		if (nvlist_add_uint16(nvl, "length", log_length) != 0) {
508 			free(lhp);
509 			return (scsi_set_errno(sip, EDS_NOMEM));
510 		}
511 
512 		if (lp->ve_validate(sip, (scsi_log_parameter_header_t *)
513 		    (((char *)lhp) + sizeof (scsi_log_header_t)),
514 		    log_length, nvl) != 0) {
515 			free(lhp);
516 			return (-1);
517 		}
518 	} else {
519 		dprintf("failed to load %s log page (KEY=0x%x "
520 		    "ASC=0x%x ASCQ=0x%x)\n", lp->ve_desc, kp, asc, ascq);
521 	}
522 
523 	free(lhp);
524 	return (0);
525 }
526 
527 /*
528  * Load log pages and determine which pages are supported.
529  */
530 static int
531 load_logpages(ds_scsi_info_t *sip)
532 {
533 	int buflen;
534 	scsi_supported_log_pages_t *sp;
535 	struct scsi_extended_sense sense;
536 	int result;
537 	uint_t sk, asc, ascq;
538 	int i, j;
539 	nvlist_t *nvl;
540 
541 	buflen = MAX_BUFLEN(scsi_log_header_t);
542 	if ((sp = calloc(buflen, 1)) == NULL)
543 		return (scsi_set_errno(sip, EDS_NOMEM));
544 
545 	bzero(&sense, sizeof (struct scsi_extended_sense));
546 
547 	if ((result = scsi_log_sense(sip, LOGPAGE_SUPP_LIST,
548 	    PC_CUMULATIVE, (caddr_t)sp, buflen, &sk, &asc, &ascq)) == 0) {
549 		int pagecount = BE_16(sp->slp_hdr.lh_length);
550 
551 		for (i = 0; i < pagecount; i++) {
552 			for (j = 0; j < NLOG_VALIDATION; j++) {
553 				if (log_validation[j].ve_code ==
554 				    sp->slp_pages[i])
555 					sip->si_supp_log |=
556 					    log_validation[j].ve_supported;
557 			}
558 		}
559 	}
560 
561 	free(sp);
562 	if (result == 0) {
563 		nvl = NULL;
564 		if (nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0) != 0 ||
565 		    nvlist_add_nvlist(sip->si_dsp->ds_state, "logpages",
566 		    nvl) != 0) {
567 			nvlist_free(nvl);
568 			return (scsi_set_errno(sip, EDS_NOMEM));
569 		}
570 
571 		nvlist_free(nvl);
572 		result = nvlist_lookup_nvlist(sip->si_dsp->ds_state,
573 		    "logpages", &sip->si_state_logpage);
574 		assert(result == 0);
575 
576 		/*
577 		 * Validate the logpage contents.
578 		 */
579 		for (i = 0; i < NLOG_VALIDATION; i++) {
580 			if ((sip->si_supp_log &
581 			    log_validation[i].ve_supported) == 0)
582 				continue;
583 
584 			/*
585 			 * verify_logpage will clear the supported bit if
586 			 * verification fails.
587 			 */
588 			if (verify_logpage(sip, &log_validation[i]) != 0)
589 				return (-1);
590 		}
591 
592 	} else {
593 		dprintf("failed to get log pages "
594 		    "(KEY=0x%x ASC=0x%x ASCq=0x%x)\n", sk, asc, ascq);
595 	}
596 
597 	/*
598 	 * We always return 0 here, even if the required log pages aren't
599 	 * supported.
600 	 */
601 	return (0);
602 }
603 
604 /*
605  * Verify that the IE log page is sane.  This log page is potentially chock-full
606  * of vendor specific information that we do not know how to access.  All we can
607  * do is check for the generic predictive failure bit.  If this log page is not
608  * well-formed, then bail out.
609  */
610 static int
611 logpage_ie_verify(ds_scsi_info_t *sip, scsi_log_parameter_header_t *lphp,
612     int log_length, nvlist_t *nvl)
613 {
614 	int i, plen = 0;
615 	boolean_t seen = B_FALSE;
616 	scsi_ie_log_param_t *iep =
617 	    (scsi_ie_log_param_t *)lphp;
618 
619 	for (i = 0; i < log_length; i += plen) {
620 		iep = (scsi_ie_log_param_t *)((char *)iep + plen);
621 
622 		if (BE_16(iep->ie_hdr.lph_param) == LOGPARAM_IE) {
623 			if (nvlist_add_boolean_value(nvl, "general",
624 			    B_TRUE) != 0)
625 				return (scsi_set_errno(sip, EDS_NOMEM));
626 
627 			if (lphp->lph_length < LOGPARAM_IE_MIN_LEN) {
628 				if (nvlist_add_uint8(nvl,
629 				    "invalid-length", lphp->lph_length) != 0)
630 					return (scsi_set_errno(sip, EDS_NOMEM));
631 			} else {
632 				seen = B_TRUE;
633 			}
634 			break;
635 		}
636 
637 		plen = iep->ie_hdr.lph_length +
638 		    sizeof (scsi_log_parameter_header_t);
639 	}
640 
641 	if (!seen) {
642 		sip->si_supp_log &= ~LOGPAGE_SUPP_IE;
643 		dprintf("IE logpage validation failed\n");
644 	}
645 
646 	return (0);
647 }
648 
649 /*
650  * Verify the contents of the temperature log page.  The temperature log page
651  * contains two log parameters: the current temperature, and (optionally) the
652  * reference temperature.  For the verification phase, we check that the two
653  * parameters we care about are well-formed.  If there is no reference
654  * temperature, then we cannot use the page for monitoring purposes.
655  */
656 static int
657 logpage_temp_verify(ds_scsi_info_t *sip,
658     scsi_log_parameter_header_t *lphp, int log_length, nvlist_t *nvl)
659 {
660 	int i, plen = 0;
661 	boolean_t has_reftemp = B_FALSE;
662 	boolean_t bad_length = B_FALSE;
663 	ushort_t param_code;
664 
665 	for (i = 0; i < log_length; i += plen) {
666 		lphp = (scsi_log_parameter_header_t *)((char *)lphp + plen);
667 		param_code = BE_16(lphp->lph_param);
668 
669 		switch (param_code) {
670 		case LOGPARAM_TEMP_CURTEMP:
671 			if (nvlist_add_boolean_value(nvl, "current-temperature",
672 			    B_TRUE) != 0)
673 				return (scsi_set_errno(sip, EDS_NOMEM));
674 			if (lphp->lph_length != LOGPARAM_TEMP_LEN) {
675 				if (nvlist_add_uint8(nvl,
676 				    "invalid-length", lphp->lph_length) != 0)
677 					return (scsi_set_errno(sip, EDS_NOMEM));
678 				bad_length = B_TRUE;
679 			}
680 			break;
681 
682 		case LOGPARAM_TEMP_REFTEMP:
683 			if (nvlist_add_boolean_value(nvl,
684 			    "reference-temperature", B_TRUE) != 0)
685 				return (scsi_set_errno(sip, EDS_NOMEM));
686 			if (lphp->lph_length != LOGPARAM_TEMP_LEN) {
687 				if (nvlist_add_uint8(nvl,
688 				    "invalid-length", lphp->lph_length) != 0)
689 					return (scsi_set_errno(sip, EDS_NOMEM));
690 				bad_length = B_TRUE;
691 			}
692 			has_reftemp = B_TRUE;
693 			break;
694 		}
695 
696 		plen = lphp->lph_length +
697 		    sizeof (scsi_log_parameter_header_t);
698 	}
699 
700 	if (bad_length || !has_reftemp) {
701 		sip->si_supp_log &= ~LOGPAGE_SUPP_TEMP;
702 		dprintf("temperature logpage validation failed\n");
703 	}
704 
705 	return (0);
706 }
707 
708 /*
709  * Verify the contents of the self test log page.  The log supports a maximum of
710  * 20 entries, where each entry's parameter code is its index in the log.  We
711  * check that the parameter codes fall within this range, and that the size of
712  * each page is what we expect.  It's perfectly acceptable for there to be no
713  * entries in this log, so we must also be sure to validate the contents as part
714  * of the analysis phase.
715  */
716 static int
717 logpage_selftest_verify(ds_scsi_info_t *sip,
718     scsi_log_parameter_header_t *lphp, int log_length, nvlist_t *nvl)
719 {
720 	int i, plen = 0;
721 	boolean_t bad = B_FALSE;
722 	int entries = 0;
723 	ushort_t param_code;
724 
725 	for (i = 0; i < log_length; i += plen, entries++) {
726 		lphp = (scsi_log_parameter_header_t *)((char *)lphp + plen);
727 		param_code = BE_16(lphp->lph_param);
728 
729 		if (param_code < LOGPAGE_SELFTEST_MIN_PARAM_CODE ||
730 		    param_code > LOGPAGE_SELFTEST_MAX_PARAM_CODE) {
731 			if (nvlist_add_uint16(nvl, "invalid-param-code",
732 			    param_code) != 0)
733 				return (scsi_set_errno(sip, EDS_NOMEM));
734 			bad = B_TRUE;
735 			break;
736 		}
737 
738 		if (lphp->lph_length != LOGPAGE_SELFTEST_PARAM_LEN) {
739 			if (nvlist_add_uint8(nvl, "invalid-length",
740 			    lphp->lph_length) != 0)
741 				return (scsi_set_errno(sip, EDS_NOMEM));
742 			bad = B_TRUE;
743 			break;
744 
745 		}
746 
747 		plen = lphp->lph_length +
748 		    sizeof (scsi_log_parameter_header_t);
749 	}
750 
751 	if (bad) {
752 		sip->si_supp_log &= ~LOGPAGE_SUPP_SELFTEST;
753 		dprintf("selftest logpage validation failed\n");
754 	}
755 
756 	return (0);
757 }
758 
759 /*
760  * Load the current IE mode pages
761  */
762 static int
763 load_ie_modepage(ds_scsi_info_t *sip)
764 {
765 	struct scsi_ms_hdrs junk_hdrs;
766 	int result;
767 	uint_t skey, asc, ascq;
768 
769 	if (!(sip->si_supp_mode & MODEPAGE_SUPP_IEC))
770 		return (0);
771 
772 	bzero(&sip->si_iec_current, sizeof (sip->si_iec_current));
773 	bzero(&sip->si_iec_changeable, sizeof (sip->si_iec_changeable));
774 
775 	if ((result = scsi_mode_sense(sip,
776 	    MODEPAGE_INFO_EXCPT, PC_CURRENT, &sip->si_iec_current,
777 	    MODEPAGE_INFO_EXCPT_LEN, &sip->si_hdrs, &skey, &asc,
778 	    &ascq)) == 0) {
779 		result = scsi_mode_sense(sip,
780 		    MODEPAGE_INFO_EXCPT, PC_CHANGEABLE,
781 		    &sip->si_iec_changeable,
782 		    MODEPAGE_INFO_EXCPT_LEN, &junk_hdrs, &skey, &asc, &ascq);
783 	}
784 
785 	if (result != 0) {
786 		dprintf("failed to get IEC modepage (KEY=0x%x "
787 		    "ASC=0x%x ASCQ=0x%x)", skey, asc, ascq);
788 		sip->si_supp_mode &= ~MODEPAGE_SUPP_IEC;
789 	} else  {
790 		if (nvlist_add_boolean_value(sip->si_state_iec,
791 		    "dexcpt", sip->si_iec_current.ie_dexcpt) != 0 ||
792 		    nvlist_add_boolean_value(sip->si_state_iec,
793 		    "logerr", sip->si_iec_current.ie_logerr) != 0 ||
794 		    nvlist_add_uint8(sip->si_state_iec,
795 		    "mrie", sip->si_iec_current.ie_mrie) != 0 ||
796 		    nvlist_add_boolean_value(sip->si_state_iec,
797 		    "test", sip->si_iec_current.ie_test) != 0 ||
798 		    nvlist_add_boolean_value(sip->si_state_iec,
799 		    "ewasc", sip->si_iec_current.ie_ewasc) != 0 ||
800 		    nvlist_add_boolean_value(sip->si_state_iec,
801 		    "perf", sip->si_iec_current.ie_perf) != 0 ||
802 		    nvlist_add_boolean_value(sip->si_state_iec,
803 		    "ebf", sip->si_iec_current.ie_ebf) != 0 ||
804 		    nvlist_add_uint32(sip->si_state_iec,
805 		    "interval-timer",
806 		    BE_32(sip->si_iec_current.ie_interval_timer)) != 0 ||
807 		    nvlist_add_uint32(sip->si_state_iec,
808 		    "report-count",
809 		    BE_32(sip->si_iec_current.ie_report_count)) != 0)
810 			return (scsi_set_errno(sip, EDS_NOMEM));
811 	}
812 
813 	return (0);
814 }
815 
816 /*
817  * Enable IE reporting.  We prefer the following settings:
818  *
819  * (1) DEXCPT = 0
820  * (3) MRIE = 6 (IE_REPORT_ON_REQUEST)
821  * (4) EWASC = 1
822  * (6) REPORT COUNT = 0x00000001
823  * (7) LOGERR = 1
824  *
825  * However, not all drives support changing these values, and the current state
826  * may be useful enough as-is.  For example, some drives support IE logging, but
827  * don't support changing the MRIE.  In this case, we can still use the
828  * information provided by the log page.
829  */
830 static int
831 scsi_enable_ie(ds_scsi_info_t *sip, boolean_t *changed)
832 {
833 	scsi_ie_page_t new_iec_page;
834 	scsi_ms_hdrs_t hdrs;
835 	uint_t skey, asc, ascq;
836 
837 	if (!(sip->si_supp_mode & MODEPAGE_SUPP_IEC))
838 		return (0);
839 
840 	bzero(&new_iec_page, sizeof (new_iec_page));
841 	bzero(&hdrs, sizeof (hdrs));
842 
843 	(void) memcpy(&new_iec_page, &sip->si_iec_current,
844 	    sizeof (new_iec_page));
845 
846 	if (IEC_IE_CHANGEABLE(sip->si_iec_changeable))
847 		new_iec_page.ie_dexcpt = 0;
848 
849 	if (IEC_MRIE_CHANGEABLE(sip->si_iec_changeable))
850 		new_iec_page.ie_mrie = IE_REPORT_ON_REQUEST;
851 
852 	/*
853 	 * We only want to enable warning reporting if we are able to change the
854 	 * mrie to report on request.  Otherwise, we risk unnecessarily
855 	 * interrupting normal SCSI commands with a CHECK CONDITION code.
856 	 */
857 	if (IEC_EWASC_CHANGEABLE(sip->si_iec_changeable)) {
858 		if (new_iec_page.ie_mrie == IE_REPORT_ON_REQUEST)
859 			new_iec_page.ie_ewasc = 1;
860 		else
861 			new_iec_page.ie_ewasc = 0;
862 	}
863 
864 	if (IEC_RPTCNT_CHANGEABLE(sip->si_iec_changeable))
865 		new_iec_page.ie_report_count = BE_32(1);
866 
867 	if (IEC_LOGERR_CHANGEABLE(sip->si_iec_changeable))
868 		new_iec_page.ie_logerr = 1;
869 
870 	/*
871 	 * Now compare the new mode page with the existing one.
872 	 * if there's no difference, there's no need for a mode select
873 	 */
874 	if (memcmp(&new_iec_page, &sip->si_iec_current,
875 	    MODEPAGE_INFO_EXCPT_LEN) == 0) {
876 		*changed = B_FALSE;
877 	} else {
878 		(void) memcpy(&hdrs, &sip->si_hdrs, sizeof (sip->si_hdrs));
879 
880 		if (scsi_mode_select(sip,
881 		    MODEPAGE_INFO_EXCPT, MODE_SELECT_PF, &new_iec_page,
882 		    MODEPAGE_INFO_EXCPT_LEN, &hdrs, &skey, &asc, &ascq) == 0) {
883 			*changed = B_TRUE;
884 		} else {
885 			dprintf("failed to enable IE (KEY=0x%x "
886 			    "ASC=0x%x ASCQ=0x%x)\n", skey, asc, ascq);
887 			*changed = B_FALSE;
888 		}
889 	}
890 
891 	if (nvlist_add_boolean_value(sip->si_state_iec, "changed",
892 	    *changed) != 0)
893 		return (scsi_set_errno(sip, EDS_NOMEM));
894 
895 	return (0);
896 }
897 
898 /*
899  * Clear the GLTSD bit, indicating log pages should be saved to non-volatile
900  * storage.
901  */
902 static int
903 clear_gltsd(ds_scsi_info_t *sip)
904 {
905 	scsi_ms_hdrs_t hdrs, junk_hdrs;
906 	struct mode_control_scsi3 control_pg_cur, control_pg_chg;
907 	int result;
908 	uint_t skey, asc, ascq;
909 
910 	bzero(&hdrs, sizeof (hdrs));
911 	bzero(&control_pg_cur, sizeof (control_pg_cur));
912 	bzero(&control_pg_chg, sizeof (control_pg_chg));
913 
914 	result = scsi_mode_sense(sip,
915 	    MODEPAGE_CTRL_MODE, PC_CURRENT, &control_pg_cur,
916 	    MODEPAGE_CTRL_MODE_LEN, &hdrs, &skey, &asc, &ascq);
917 
918 	if (result != 0) {
919 		dprintf("failed to read Control mode page (KEY=0x%x "
920 		    "ASC=0x%x ASCQ=0x%x)\n", skey, asc, ascq);
921 	} else if (control_pg_cur.mode_page.length !=
922 	    PAGELENGTH_MODE_CONTROL_SCSI3) {
923 		dprintf("SCSI-3 control mode page not supported\n");
924 	} else if ((result = scsi_mode_sense(sip,
925 	    MODEPAGE_CTRL_MODE, PC_CHANGEABLE, &control_pg_chg,
926 	    MODEPAGE_CTRL_MODE_LEN, &junk_hdrs, &skey, &asc, &ascq))
927 	    != 0) {
928 		dprintf("failed to read changeable Control mode page (KEY=0x%x "
929 		    "ASC=0x%x ASCQ=0x%x)\n", skey, asc, ascq);
930 	} else if (control_pg_cur.gltsd && !GLTSD_CHANGEABLE(control_pg_chg)) {
931 		dprintf("gltsd is set and not changeable\n");
932 		if (nvlist_add_boolean_value(sip->si_dsp->ds_state,
933 		    "gltsd", control_pg_cur.gltsd) != 0)
934 			return (scsi_set_errno(sip, EDS_NOMEM));
935 	} else if (control_pg_cur.gltsd) {
936 		control_pg_cur.gltsd = 0;
937 		result = scsi_mode_select(sip,
938 		    MODEPAGE_CTRL_MODE, MODE_SELECT_PF, &control_pg_cur,
939 		    MODEPAGE_CTRL_MODE_LEN, &hdrs, &skey, &asc, &ascq);
940 		if (result != 0)
941 			dprintf("failed to enable GLTSD (KEY=0x%x "
942 			    "ASC=0x%x ASCQ=0x%x\n", skey, asc, ascq);
943 		if (nvlist_add_boolean_value(sip->si_dsp->ds_state,
944 		    "gltsd", control_pg_cur.gltsd) != 0)
945 			return (scsi_set_errno(sip, EDS_NOMEM));
946 	}
947 
948 	return (0);
949 }
950 
951 /*
952  * Fetch the contents of the logpage, and then call the logpage-specific
953  * analysis function.  The analysis function is responsible for detecting any
954  * faults and filling in the details.
955  */
956 static int
957 analyze_one_logpage(ds_scsi_info_t *sip, logpage_validation_entry_t *entry)
958 {
959 	scsi_log_header_t *lhp;
960 	scsi_log_parameter_header_t *lphp;
961 	int buflen;
962 	int log_length;
963 	uint_t skey, asc, ascq;
964 	int result;
965 
966 	buflen = MAX_BUFLEN(scsi_log_header_t);
967 	if ((lhp = calloc(buflen, 1)) == NULL)
968 		return (scsi_set_errno(sip, EDS_NOMEM));
969 
970 	result = scsi_log_sense(sip, entry->ve_code,
971 	    PC_CUMULATIVE, (caddr_t)lhp, buflen, &skey, &asc, &ascq);
972 
973 	if (result == 0) {
974 		log_length = BE_16(lhp->lh_length);
975 		lphp = (scsi_log_parameter_header_t *)(((uchar_t *)lhp) +
976 		    sizeof (scsi_log_header_t));
977 
978 		result = entry->ve_analyze(sip, lphp, log_length);
979 	} else {
980 		result = scsi_set_errno(sip, EDS_IO);
981 	}
982 
983 	free(lhp);
984 	return (result);
985 }
986 
987 /*
988  * Analyze the IE logpage.  If we find an IE log record with a non-zero 'asc',
989  * then we have a fault.
990  */
991 static int
992 logpage_ie_analyze(ds_scsi_info_t *sip, scsi_log_parameter_header_t *lphp,
993     int log_length)
994 {
995 	int i, plen = 0;
996 	scsi_ie_log_param_t *iep = (scsi_ie_log_param_t *)lphp;
997 	nvlist_t *nvl;
998 
999 	assert(sip->si_dsp->ds_predfail == NULL);
1000 	if (nvlist_alloc(&sip->si_dsp->ds_predfail, NV_UNIQUE_NAME, 0) != 0)
1001 		return (scsi_set_errno(sip, EDS_NOMEM));
1002 	nvl = sip->si_dsp->ds_predfail;
1003 
1004 	for (i = 0; i < log_length; i += plen) {
1005 		iep = (scsi_ie_log_param_t *)((char *)iep + plen);
1006 
1007 		/*
1008 		 * Even though we validated the length during the initial phase,
1009 		 * never trust the device.
1010 		 */
1011 		if (BE_16(iep->ie_hdr.lph_param) == LOGPARAM_IE &&
1012 		    iep->ie_hdr.lph_length >= LOGPARAM_IE_MIN_LEN) {
1013 			if (nvlist_add_uint8(nvl, FM_EREPORT_PAYLOAD_SCSI_ASC,
1014 			    iep->ie_asc) != 0 ||
1015 			    nvlist_add_uint8(nvl, FM_EREPORT_PAYLOAD_SCSI_ASCQ,
1016 			    iep->ie_ascq) != 0)
1017 				return (scsi_set_errno(sip, EDS_NOMEM));
1018 
1019 			if (iep->ie_asc != 0)
1020 				sip->si_dsp->ds_faults |=
1021 				    DS_FAULT_PREDFAIL;
1022 			break;
1023 		}
1024 		plen = iep->ie_hdr.lph_length +
1025 		    sizeof (scsi_log_parameter_header_t);
1026 	}
1027 
1028 	return (0);
1029 }
1030 
1031 static int
1032 logpage_temp_analyze(ds_scsi_info_t *sip, scsi_log_parameter_header_t *lphp,
1033     int log_length)
1034 {
1035 	int i, plen = 0;
1036 	uint8_t reftemp, curtemp;
1037 	ushort_t param_code;
1038 	scsi_temp_log_param_t *temp;
1039 	nvlist_t *nvl;
1040 
1041 	assert(sip->si_dsp->ds_overtemp == NULL);
1042 	if (nvlist_alloc(&sip->si_dsp->ds_overtemp, NV_UNIQUE_NAME, 0) != 0)
1043 		return (scsi_set_errno(sip, EDS_NOMEM));
1044 	nvl = sip->si_dsp->ds_overtemp;
1045 
1046 	reftemp = curtemp = INVALID_TEMPERATURE;
1047 	for (i = 0; i < log_length; i += plen) {
1048 		lphp = (scsi_log_parameter_header_t *)((char *)lphp + plen);
1049 		param_code = BE_16(lphp->lph_param);
1050 		temp = (scsi_temp_log_param_t *)lphp;
1051 
1052 		switch (param_code) {
1053 		case LOGPARAM_TEMP_CURTEMP:
1054 			if (lphp->lph_length != LOGPARAM_TEMP_LEN)
1055 				break;
1056 
1057 			if (nvlist_add_uint8(nvl,
1058 			    FM_EREPORT_PAYLOAD_SCSI_CURTEMP,
1059 			    temp->t_temp) != 0)
1060 				return (scsi_set_errno(sip, EDS_NOMEM));
1061 			curtemp = temp->t_temp;
1062 			break;
1063 
1064 		case LOGPARAM_TEMP_REFTEMP:
1065 			if (lphp->lph_length != LOGPARAM_TEMP_LEN)
1066 				break;
1067 
1068 			if (nvlist_add_uint8(nvl,
1069 			    FM_EREPORT_PAYLOAD_SCSI_THRESHTEMP,
1070 			    temp->t_temp) != 0)
1071 				return (scsi_set_errno(sip, EDS_NOMEM));
1072 			reftemp = temp->t_temp;
1073 			break;
1074 		}
1075 
1076 		plen = lphp->lph_length +
1077 		    sizeof (scsi_log_parameter_header_t);
1078 	}
1079 
1080 	if (reftemp != INVALID_TEMPERATURE && curtemp != INVALID_TEMPERATURE &&
1081 	    curtemp > reftemp)
1082 		sip->si_dsp->ds_faults |= DS_FAULT_OVERTEMP;
1083 
1084 	return (0);
1085 }
1086 
1087 static int
1088 logpage_selftest_analyze(ds_scsi_info_t *sip, scsi_log_parameter_header_t *lphp,
1089     int log_length)
1090 {
1091 	int i, plen = 0;
1092 	int entries = 0;
1093 	ushort_t param_code;
1094 	scsi_selftest_log_param_t *stp;
1095 	nvlist_t *nvl;
1096 
1097 	assert(sip->si_dsp->ds_testfail == NULL);
1098 	if (nvlist_alloc(&sip->si_dsp->ds_testfail, NV_UNIQUE_NAME, 0) != 0)
1099 		return (scsi_set_errno(sip, EDS_NOMEM));
1100 	nvl = sip->si_dsp->ds_testfail;
1101 
1102 	for (i = 0; i < log_length; i += plen, entries++) {
1103 		lphp = (scsi_log_parameter_header_t *)((char *)lphp + plen);
1104 		param_code = BE_16(lphp->lph_param);
1105 		stp = (scsi_selftest_log_param_t *)lphp;
1106 
1107 		if (param_code >= LOGPAGE_SELFTEST_MIN_PARAM_CODE &&
1108 		    param_code <= LOGPAGE_SELFTEST_MAX_PARAM_CODE &&
1109 		    lphp->lph_length >= LOGPAGE_SELFTEST_PARAM_LEN) {
1110 			/*
1111 			 * We always log the last result, or the result of the
1112 			 * last completed test.
1113 			 */
1114 			if ((param_code == 1 ||
1115 			    SELFTEST_COMPLETE(stp->st_results))) {
1116 				if (nvlist_add_uint8(nvl,
1117 				    FM_EREPORT_PAYLOAD_SCSI_RESULTCODE,
1118 				    stp->st_results) != 0 ||
1119 				    nvlist_add_uint16(nvl,
1120 				    FM_EREPORT_PAYLOAD_SCSI_TIMESTAMP,
1121 				    BE_16(stp->st_timestamp)) != 0 ||
1122 				    nvlist_add_uint8(nvl,
1123 				    FM_EREPORT_PAYLOAD_SCSI_SEGMENT,
1124 				    stp->st_number) != 0 ||
1125 				    nvlist_add_uint64(nvl,
1126 				    FM_EREPORT_PAYLOAD_SCSI_ADDRESS,
1127 				    BE_64(stp->st_lba)) != 0)
1128 					return (scsi_set_errno(sip,
1129 					    EDS_NOMEM));
1130 
1131 				if (SELFTEST_COMPLETE(stp->st_results)) {
1132 					if (stp->st_results != SELFTEST_OK)
1133 						sip->si_dsp->ds_faults |=
1134 						    DS_FAULT_TESTFAIL;
1135 					return (0);
1136 				}
1137 			}
1138 		}
1139 
1140 		plen = lphp->lph_length +
1141 		    sizeof (scsi_log_parameter_header_t);
1142 	}
1143 
1144 	return (0);
1145 }
1146 
1147 /*
1148  * Analyze the IE mode sense page explicitly.  This is only needed if the IE log
1149  * page is not supported.
1150  */
1151 static int
1152 analyze_ie_sense(ds_scsi_info_t *sip)
1153 {
1154 	uint_t skey, asc, ascq;
1155 	nvlist_t *nvl;
1156 
1157 	/*
1158 	 * Don't bother checking if we weren't able to set our MRIE correctly.
1159 	 */
1160 	if (sip->si_iec_current.ie_mrie != IE_REPORT_ON_REQUEST)
1161 		return (0);
1162 
1163 	if (scsi_request_sense(sip, &skey, &asc, &ascq) != 0) {
1164 		dprintf("failed to request IE page (KEY=0x%x ASC=0x%x "
1165 		    "ASCQ=0x%x)\n", skey, asc, ascq);
1166 		return (scsi_set_errno(sip, EDS_IO));
1167 	} else if (skey == KEY_NO_SENSE) {
1168 		assert(sip->si_dsp->ds_predfail == NULL);
1169 		if (nvlist_alloc(&sip->si_dsp->ds_predfail,
1170 		    NV_UNIQUE_NAME, 0) != 0)
1171 			return (scsi_set_errno(sip, EDS_NOMEM));
1172 		nvl = sip->si_dsp->ds_predfail;
1173 
1174 		if (nvlist_add_uint8(nvl,
1175 		    FM_EREPORT_PAYLOAD_SCSI_ASC, asc) != 0 ||
1176 		    nvlist_add_uint8(nvl,
1177 		    FM_EREPORT_PAYLOAD_SCSI_ASCQ, ascq) != 0) {
1178 			nvlist_free(nvl);
1179 			return (scsi_set_errno(sip, EDS_NOMEM));
1180 		}
1181 
1182 		if (asc != 0)
1183 			sip->si_dsp->ds_faults |= DS_FAULT_PREDFAIL;
1184 	}
1185 
1186 	return (0);
1187 }
1188 
1189 /*
1190  * Clean up the scsi-specific information structure.
1191  */
1192 static void
1193 ds_scsi_close(void *arg)
1194 {
1195 	ds_scsi_info_t *sip = arg;
1196 	if (sip->si_sim)
1197 		(void) dlclose(sip->si_sim);
1198 
1199 	free(sip);
1200 }
1201 
1202 /*
1203  * Initialize a single disk.  Initialization consists of:
1204  *
1205  * 1. Check to see if the IE mechanism is enabled via MODE SENSE for the IE
1206  *    Control page (page 0x1C).
1207  *
1208  * 2. If the IE page is available, try to set the following parameters:
1209  *
1210  *    	DEXCPT		0	Enable exceptions
1211  *    	MRIE		6	Only report IE information on request
1212  *    	EWASC		1	Enable warning reporting
1213  *    	REPORT COUNT	1	Only report an IE exception once
1214  *    	LOGERR		1	Enable logging of errors
1215  *
1216  *    The remaining fields are left as-is, preserving the current values.  If we
1217  *    cannot set some of these fields, then we do our best.  Some drives may
1218  *    have a static configuration which still allows for some monitoring.
1219  *
1220  * 3. Check to see if the IE log page (page 0x2F) is supported by issuing a
1221  *    LOG SENSE command.
1222  *
1223  * 4. Check to see if the self-test log page (page 0x10) is supported.
1224  *
1225  * 5. Check to see if the temperature log page (page 0x0D) is supported, and
1226  *    contains a reference temperature.
1227  *
1228  * 6. Clear the GLTSD bit in control mode page 0xA.  This will allow the drive
1229  *    to save each of the log pages described above to nonvolatile storage.
1230  *    This is essential if the drive is to remember its failures across
1231  *    loss of power.
1232  */
1233 static void *
1234 ds_scsi_open_common(disk_status_t *dsp, ds_scsi_info_t *sip)
1235 {
1236 	boolean_t changed;
1237 
1238 	sip->si_dsp = dsp;
1239 
1240 	/* Load and validate mode pages */
1241 	if (load_modepages(sip) != 0) {
1242 		ds_scsi_close(sip);
1243 		return (NULL);
1244 	}
1245 
1246 	/* Load and validate log pages */
1247 	if (load_logpages(sip) != 0) {
1248 		ds_scsi_close(sip);
1249 		return (NULL);
1250 	}
1251 
1252 	/* Load IE state */
1253 	if (load_ie_modepage(sip) != 0 ||
1254 	    scsi_enable_ie(sip, &changed) != 0 ||
1255 	    (changed && load_ie_modepage(sip) != 0)) {
1256 		ds_scsi_close(sip);
1257 		return (NULL);
1258 	}
1259 
1260 	/* Clear the GLTSD bit in the control page */
1261 	if (sip->si_supp_log != 0 && clear_gltsd(sip) != 0) {
1262 		ds_scsi_close(sip);
1263 		return (NULL);
1264 	}
1265 
1266 	return (sip);
1267 }
1268 
1269 static void *
1270 ds_scsi_open_uscsi(disk_status_t *dsp)
1271 {
1272 	ds_scsi_info_t *sip;
1273 
1274 	if ((sip = calloc(sizeof (ds_scsi_info_t), 1)) == NULL) {
1275 		(void) ds_set_errno(dsp, EDS_NOMEM);
1276 		return (NULL);
1277 	}
1278 
1279 	return (ds_scsi_open_common(dsp, sip));
1280 }
1281 
1282 static void *
1283 ds_scsi_open_sim(disk_status_t *dsp)
1284 {
1285 	ds_scsi_info_t *sip;
1286 
1287 	if ((sip = calloc(sizeof (ds_scsi_info_t), 1)) == NULL) {
1288 		(void) ds_set_errno(dsp, EDS_NOMEM);
1289 		return (NULL);
1290 	}
1291 
1292 	if ((sip->si_sim = dlopen(dsp->ds_path, RTLD_LAZY)) == NULL) {
1293 		(void) ds_set_errno(dsp, EDS_NO_TRANSPORT);
1294 		free(sip);
1295 		return (NULL);
1296 	}
1297 
1298 	return (ds_scsi_open_common(dsp, sip));
1299 }
1300 
1301 
1302 /*
1303  * Scan for any faults.  The following steps are performed:
1304  *
1305  * 1. If the temperature log page is supported, check the current temperature
1306  *    and threshold.  If the current temperature exceeds the threshold, report
1307  *    and overtemp fault.
1308  *
1309  * 2. If the selftest log page is supported, check to the last completed self
1310  *    test.  If the last completed test resulted in failure, report a selftest
1311  *    fault.
1312  *
1313  * 3. If the IE log page is supported, check to see if failure is predicted.  If
1314  *    so, indicate a predictive failure fault.
1315  *
1316  * 4. If the IE log page is not supported, but the mode page supports report on
1317  *    request mode, then issue a REQUEST SENSE for the mode page.  Indicate a
1318  *    predictive failure fault if necessary.
1319  */
1320 static int
1321 ds_scsi_scan(void *arg)
1322 {
1323 	ds_scsi_info_t *sip = arg;
1324 	int i;
1325 
1326 	for (i = 0; i < NLOG_VALIDATION; i++) {
1327 		if ((sip->si_supp_log & log_validation[i].ve_supported) == 0)
1328 			continue;
1329 
1330 		if (analyze_one_logpage(sip, &log_validation[i]) != 0)
1331 			return (-1);
1332 	}
1333 
1334 	if (!(sip->si_supp_log & LOGPAGE_SUPP_IE) &&
1335 	    (sip->si_supp_mode & MODEPAGE_SUPP_IEC) &&
1336 	    analyze_ie_sense(sip) != 0)
1337 		return (-1);
1338 
1339 	return (0);
1340 }
1341 
1342 ds_transport_t ds_scsi_uscsi_transport = {
1343 	ds_scsi_open_uscsi,
1344 	ds_scsi_close,
1345 	ds_scsi_scan
1346 };
1347 
1348 ds_transport_t ds_scsi_sim_transport = {
1349 	ds_scsi_open_sim,
1350 	ds_scsi_close,
1351 	ds_scsi_scan
1352 };
1353