xref: /illumos-gate/usr/src/cmd/sasinfo/printAttrs.c (revision e9db39cef1f968a982994f50c05903cc988a3dd3)
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 2009 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 #include <ctype.h>
27 #include <printAttrs.h>
28 
29 static SAS_STATE hbastatus_string[] = {
30 	HBA_STATUS_OK,				"Okay",
31 	HBA_STATUS_ERROR,			"Error",
32 	HBA_STATUS_ERROR_NOT_SUPPORTED,		"Not Supported",
33 	HBA_STATUS_ERROR_INVALID_HANDLE,	"Invalid Handle",
34 	HBA_STATUS_ERROR_ARG,			"Argument Error",
35 	HBA_STATUS_ERROR_ILLEGAL_WWN,		"Illegal WWN",
36 	HBA_STATUS_ERROR_ILLEGAL_INDEX,		"Illegal Index",
37 	HBA_STATUS_ERROR_MORE_DATA,		"Not Enough Buffer for Data",
38 	HBA_STATUS_ERROR_STALE_DATA,		"Stale Data",
39 	HBA_STATUS_SCSI_CHECK_CONDITION,	"SCSI Check Condition",
40 	HBA_STATUS_ERROR_BUSY,			"Busy",
41 	HBA_STATUS_ERROR_TRY_AGAIN,		"Try Again",
42 	HBA_STATUS_ERROR_UNAVAILABLE,		"Unavailable",
43 	HBA_STATUS_ERROR_ELS_REJECT,		"ELS Reject",
44 	HBA_STATUS_ERROR_INVALID_LUN,		"Invalid LUN",
45 	HBA_STATUS_ERROR_INCOMPATIBLE,		"Request Incompatible",
46 	HBA_STATUS_ERROR_AMBIGUOUS_WWN,		"Ambiguous WWN",
47 	HBA_STATUS_ERROR_LOCAL_BUS,		"Local Bus Error",
48 	HBA_STATUS_ERROR_LOCAL_TARGET,		"Local Target Error",
49 	HBA_STATUS_ERROR_LOCAL_LUN,		"Local LUN Error",
50 	HBA_STATUS_ERROR_LOCAL_SCSIID_BOUND,	"Local SCSIID Bound",
51 	HBA_STATUS_ERROR_TARGET_FCID,		"Target FCID Error",
52 	HBA_STATUS_ERROR_TARGET_NODE_WWN,	"Target Node WWN Error",
53 	HBA_STATUS_ERROR_TARGET_PORT_WWN,	"Target Port WWN Error",
54 	HBA_STATUS_ERROR_TARGET_LUN,		"Target LUN Error",
55 	HBA_STATUS_ERROR_TARGET_LUID,		"Target LUID Error",
56 	HBA_STATUS_ERROR_NO_SUCH_BINDING,	"No Such Binding",
57 	HBA_STATUS_ERROR_NOT_A_TARGET,		"Not a Target",
58 	HBA_STATUS_ERROR_UNSUPPORTED_FC4,	"Unsupported FC4",
59 	HBA_STATUS_ERROR_INCAPABLE,		"Incapable",
60 	HBA_STATUS_ERROR_TARGET_BUSY,		"Target Busy",
61 	HBA_STATUS_ERROR_NOT_LOADED,		"Not Loaded",
62 	HBA_STATUS_ERROR_ALREADY_LOADED,	"Alreday Loaded",
63 	HBA_STATUS_ERROR_ILLEGAL_FCID,		"Illegal FCID",
64 	HBA_STATUS_ERROR_NOT_ASCSIDEVICE,	"Not a SCSI Device",
65 	HBA_STATUS_ERROR_INVALID_PROTOCOL_TYPE,	"Invalid Protocol Type",
66 	HBA_STATUS_ERROR_BAD_EVENT_TYPE,	"Bad Event Type",
67 	-1,					NULL
68 };
69 
70 SAS_STATE porttype_string[] = {
71 	HBA_PORTTYPE_UNKNOWN,		"UNKNOWN",
72 	HBA_PORTTYPE_OTHER,		"OTHER",
73 	HBA_PORTTYPE_NOTPRESENT,	"NOT Present",
74 	HBA_PORTTYPE_SASDEVICE,		"SAS Device",
75 	HBA_PORTTYPE_SATADEVICE,	"SATA Device",
76 	HBA_PORTTYPE_SASEXPANDER, 	"SAS Expander",
77 	-1,				NULL,
78 };
79 
80 SAS_STATE portstate_string[] = {
81 	HBA_PORTSTATE_UNKNOWN,		"unknown",
82 	HBA_PORTSTATE_ONLINE,		"online",
83 	HBA_PORTSTATE_OFFLINE,		"offline",
84 	HBA_PORTSTATE_BYPASSED,		"bypassed",
85 	HBA_PORTSTATE_DIAGNOSTICS,	"diagnostics",
86 	HBA_PORTSTATE_LINKDOWN,		"link Down",
87 	HBA_PORTSTATE_ERROR,		"port Error",
88 	HBA_PORTSTATE_LOOPBACK,		"loopback",
89 	HBA_PORTSTATE_DEGRADED,		"degraded",
90 	-1,				NULL,
91 };
92 
93 static SAS_STATE phystate_string[] = {
94 	HBA_SASSTATE_UNKNOWN,		"unknown",
95 	HBA_SASSTATE_DISABLED,		"disabled",
96 	HBA_SASSTATE_FAILED,		"failed",
97 	HBA_SASSTATE_SATASPINUP,	"sata-spinup",
98 	HBA_SASSTATE_SATAPORTSEL,	"sata-portselector",
99 	HBA_SASSPEED_1_5GBIT,		"1.5Gbit",
100 	HBA_SASSPEED_3GBIT,		"3Gbit",
101 	HBA_SASSPEED_6GBIT,		"6Gbit",
102 	HBA_SASSPEED_12GBIT,		"12Gbit",
103 	-1,				NULL,
104 };
105 
106 static SAS_STATE dtype_string[] = {
107 	DTYPE_DIRECT,			"Disk Device",
108 	DTYPE_SEQUENTIAL,		"Tape Device",
109 	DTYPE_PRINTER,			"Printer Device",
110 	DTYPE_PROCESSOR,		"Processor Device",
111 	DTYPE_WORM,			"WORM Device",
112 	DTYPE_RODIRECT,			"CD/DVD Device",
113 	DTYPE_SCANNER,			"Scanner Device",
114 	DTYPE_OPTICAL,			"Optical Memory Device",
115 	DTYPE_CHANGER,			"Medium Changer Device",
116 	DTYPE_COMM,			"Communications Device",
117 	DTYPE_ARRAY_CTRL,		"Storage Array Controller Device",
118 	DTYPE_ESI,			"Enclosure Services Device",
119 	DTYPE_RBC,			"Simplified Direct-access Device",
120 	DTYPE_OCRW,			"Optical Card Reader/Writer Device",
121 	DTYPE_BCC,			"Bridge Controller Commands",
122 	DTYPE_OSD,			"Object-based Storage Device",
123 	DTYPE_ADC,			"Automation/Drive Interface",
124 	DTYPE_WELLKNOWN,		"Well Known Logical Unit",
125 	DTYPE_UNKNOWN,			"Unknown Device",
126 	-1,				NULL
127 };
128 
129 static char *getPhyStateString(HBA_UINT32 key, phystat_type phyt);
130 
131 char *
132 getIndentSpaces(int number)
133 {
134 	int 		i = 0;
135 	/* the maximum indent with terminator '\0' */
136 	static char	ret[MAXINDENT+1];
137 
138 	if (number > MAXINDENT)
139 		number = MAXINDENT;
140 
141 	for (i = 0; i < number; i++) {
142 		ret[i] = ' ';
143 	}
144 	ret[i] = '\0';
145 	return (ret);
146 }
147 
148 char *
149 getStateString(HBA_UINT32 key, SAS_STATE *stat_string)
150 {
151 	static char ret[64];
152 	while (stat_string->key != -1) {
153 		if (stat_string->key == key) {
154 			return ((char *)stat_string->value);
155 		}
156 		stat_string++;
157 	}
158 	(void *) sprintf(ret, "Undefined value (%d)", key);
159 	return (ret);
160 }
161 
162 static char *
163 getPhyStateString(HBA_UINT32 key, phystat_type phyt)
164 {
165 	int i = 0, len = 0, match = 0;
166 	HBA_UINT32 physpeed[] = {
167 		HBA_SASSPEED_1_5GBIT,
168 		HBA_SASSPEED_3GBIT,
169 		HBA_SASSPEED_6GBIT,
170 		HBA_SASSPEED_12GBIT
171 	};
172 
173 	len = sizeof (physpeed) / sizeof (HBA_UINT32);
174 	for (i = 0; i < len; i++) {
175 		if (key == physpeed[i]) {
176 			match = 1;
177 			break;
178 		}
179 	}
180 
181 	if (match == 1) {
182 		if (phyt == PHY_STATE)
183 			return ("enabled");
184 		else
185 			return (getStateString(key, phystate_string));
186 	} else {
187 		if (phyt == PHY_STATE)
188 			return (getStateString(key, phystate_string));
189 		else
190 			return ("not available");
191 	}
192 }
193 
194 char *
195 getHBAStatus(HBA_STATUS key)
196 {
197 	return (getStateString(key, hbastatus_string));
198 }
199 
200 /*
201  * return device type description
202  *
203  * Arguments:
204  *	dType - Device type returned from Standard INQUIRY
205  * Returns:
206  *	char string description for device type
207  */
208 char *
209 getDTypeString(uchar_t dType)
210 {
211 	return (getStateString((dType & DTYPE_MASK), dtype_string));
212 }
213 
214 uint64_t
215 wwnConversion(uchar_t *wwn)
216 {
217 	uint64_t tmp;
218 	(void *) memcpy(&tmp, wwn, sizeof (uint64_t));
219 	return (ntohll(tmp));
220 }
221 
222 /*
223  * prints out HBA information
224  */
225 void
226 printHBAInfo(SMHBA_ADAPTERATTRIBUTES *attrs, int pflag, int numberOfPorts,
227     const char *adapterName)
228 {
229 
230 	(void *) fprintf(stdout, "%s %s\n", "HBA Name:", adapterName);
231 
232 	if (pflag & PRINT_VERBOSE) {
233 		(void *) fprintf(stdout, "%s%s %s\n",
234 		    getIndentSpaces(4), "Manufacturer:",
235 		    attrs->Manufacturer[0] == 0?
236 		    "not available":attrs->Manufacturer);
237 		(void *) fprintf(stdout, "%s%s %s\n",
238 		    getIndentSpaces(4), "Model: ",
239 		    attrs->Model[0] == 0? "not available":attrs->Model);
240 		(void *) fprintf(stdout, "%s%s %s\n",
241 		    getIndentSpaces(4),
242 		    "Firmware Version:",
243 		    attrs->FirmwareVersion[0] == 0? "not available":
244 		    attrs->FirmwareVersion);
245 		(void *) fprintf(stdout, "%s%s %s\n",
246 		    getIndentSpaces(4),
247 		    "FCode/BIOS Version:",
248 		    attrs->OptionROMVersion[0] == 0? "not available":
249 		    attrs->OptionROMVersion);
250 		(void *) fprintf(stdout, "%s%s %s\n",
251 		    getIndentSpaces(4),
252 		    "Serial Number:",
253 		    attrs->SerialNumber[0] == 0? "not available":
254 		    attrs->SerialNumber);
255 		(void *) fprintf(stdout, "%s%s %s\n",
256 		    getIndentSpaces(4),
257 		    "Driver Name:",
258 		    attrs->DriverName[0] == 0? "not available":
259 		    attrs->DriverName);
260 		(void *) fprintf(stdout, "%s%s %s\n",
261 		    getIndentSpaces(4),
262 		    "Driver Version:",
263 		    attrs->DriverVersion[0] == 0? "not available":
264 		    attrs->DriverVersion);
265 		(void *) fprintf(stdout, "%s%s %d\n",
266 		    getIndentSpaces(4),
267 		    "Number of HBA Ports:",
268 		    numberOfPorts);
269 	}
270 }
271 
272 /*
273  * prints out all the HBA port information
274  */
275 void
276 printHBAPortInfo(SMHBA_PORTATTRIBUTES *port,
277     SMHBA_ADAPTERATTRIBUTES *attrs, int pflag) {
278 
279 	if ((port == NULL) || (attrs == NULL)) {
280 		return;
281 	}
282 
283 	(void *) fprintf(stdout, "%s%s %s\n",
284 	    getIndentSpaces(2),
285 	    "HBA Port Name:",
286 	    port->OSDeviceName);
287 
288 	if (!(pflag & PRINT_VERBOSE)) {
289 		return;
290 	}
291 
292 	if (port->PortType != HBA_PORTTYPE_SASDEVICE)
293 		return;
294 
295 	(void *) fprintf(stdout, "%s%s %s\n",
296 	    getIndentSpaces(4),
297 	    "Type:",
298 	    getStateString(port->PortType, porttype_string));
299 	(void *) fprintf(stdout, "%s%s %s\n",
300 	    getIndentSpaces(4),
301 	    "State:",
302 	    getStateString(port->PortState, portstate_string));
303 
304 	(void *) fprintf(stdout, "%s%s %016llx\n",
305 	    getIndentSpaces(4),
306 	    "Local SAS Address:",
307 	    wwnConversion(port->PortSpecificAttribute.SASPort->\
308 	    LocalSASAddress.wwn));
309 
310 	(void *) fprintf(stdout, "%s%s %016llx\n",
311 	    getIndentSpaces(4),
312 	    "Attached SAS Address:",
313 	    wwnConversion(port->PortSpecificAttribute.SASPort->\
314 	    AttachedSASAddress.wwn));
315 
316 	(void *) fprintf(stdout, "%s%s %d\n",
317 	    getIndentSpaces(4),
318 	    "Number of Phys:",
319 	    port->PortSpecificAttribute.SASPort->NumberofPhys);
320 }
321 
322 void
323 printHBAPortPhyInfo(SMHBA_SAS_PHY *phyinfo)
324 {
325 	if (phyinfo == NULL)
326 		return;
327 
328 	(void *) fprintf(stdout, "%s%s %u\n",
329 	    getIndentSpaces(6),
330 	    "Identifier:",
331 	    phyinfo->PhyIdentifier);
332 
333 	(void *) fprintf(stdout, "%s%s %s\n",
334 	    getIndentSpaces(8),
335 	    "State: ",
336 	    getPhyStateString(phyinfo->NegotiatedLinkRate, PHY_STATE));
337 	(void *) fprintf(stdout, "%s%s %s/%s\n",
338 	    getIndentSpaces(8),
339 	    "HardwareLinkRate(Min/Max):",
340 	    getPhyStateString(phyinfo->HardwareMinLinkRate, PHY_SPEED),
341 	    getPhyStateString(phyinfo->HardwareMaxLinkRate, PHY_SPEED));
342 	(void *) fprintf(stdout, "%s%s %s/%s\n",
343 	    getIndentSpaces(8),
344 	    "ProgrammedLinkRate(Min/Max):",
345 	    getPhyStateString(phyinfo->ProgrammedMinLinkRate, PHY_SPEED),
346 	    getPhyStateString(phyinfo->ProgrammedMaxLinkRate, PHY_SPEED));
347 	(void *) fprintf(stdout, "%s%s %s\n",
348 	    getIndentSpaces(8),
349 	    "NegotiatedLinkRate:",
350 	    getPhyStateString(phyinfo->NegotiatedLinkRate, PHY_SPEED));
351 }
352 
353 void
354 printHBAPortPhyStatistics(SMHBA_SASPHYSTATISTICS *phystat)
355 {
356 	if (phystat == NULL)
357 		return;
358 
359 	(void *) fprintf(stdout, "%s%s\n",
360 	    getIndentSpaces(8),
361 	    "Link Error Statistics:");
362 	(void *) fprintf(stdout, "%s%s %llu\n",
363 	    getIndentSpaces(12),
364 	    "Invalid Dword:",
365 	    phystat->InvalidDwordCount);
366 	(void *) fprintf(stdout, "%s%s %llu\n",
367 	    getIndentSpaces(12),
368 	    "Running Disparity Error:",
369 	    phystat->RunningDisparityErrorCount);
370 	(void *) fprintf(stdout, "%s%s %llu\n",
371 	    getIndentSpaces(12),
372 	    "Loss of Dword Sync:",
373 	    phystat->LossofDwordSyncCount);
374 	(void *) fprintf(stdout, "%s%s %llu\n",
375 	    getIndentSpaces(12),
376 	    "Reset Problem:",
377 	    phystat->PhyResetProblemCount);
378 }
379 
380 /*
381  * print the OS device name for the logical-unit object
382  *
383  * Arguments:
384  *	devListWalk - OS device path info
385  *	verbose - boolean indicating whether to display additional info
386  *
387  * returns:
388  * 	0 - we're good.
389  * 	>0 - we met issues.
390  */
391 int
392 printTargetPortInfo(targetPortList_t *TPListWalk, int pflag)
393 {
394 	targetPortConfig_t	*configList;
395 	targetPortMappingData_t	*mapList;
396 	int			count, i;
397 	int			ret = 0;
398 
399 	(void *) fprintf(stdout, "Target Port SAS Address: %016llx\n",
400 	    wwnConversion(TPListWalk->sasattr.LocalSASAddress.wwn));
401 	if ((pflag & PRINT_VERBOSE) || (pflag & PRINT_TARGET_SCSI)) {
402 		(void *) fprintf(stdout, "%sType: %s\n", getIndentSpaces(4),
403 		    getStateString(TPListWalk->targetattr.PortType,
404 		    porttype_string));
405 		for (configList = TPListWalk->configEntry;
406 		    configList != NULL; configList = configList->next) {
407 			(void *) fprintf(stdout, "%sHBA Port Name: %s\n",
408 			    getIndentSpaces(4), configList->hbaPortName);
409 			if (wwnConversion(configList->expanderSASAddr.wwn) !=
410 			    0) {
411 				if (configList->expanderValid) {
412 					(void *) fprintf(stdout,
413 					    "%sExpander Device SAS Address:"
414 					    " %016llx",
415 					    getIndentSpaces(8),
416 					    wwnConversion(configList->
417 					    expanderSASAddr.wwn));
418 				} else {
419 					(void *) fprintf(stdout,
420 					    "%sExpander Device SAS Address:"
421 					    " %016llx (Failed to Validate"
422 					    " Attached Port.)",
423 					    getIndentSpaces(8),
424 					    wwnConversion(configList->
425 					    expanderSASAddr.wwn));
426 					ret++;
427 				}
428 			} else {
429 				if (configList->expanderValid) {
430 					(void *) fprintf(stdout,
431 					    "%sExpander Device SAS Address: %s",
432 					    getIndentSpaces(8),
433 					    "None (direct attached)");
434 				} else {
435 					(void *) fprintf(stdout,
436 					    "%sExpander Device SAS Address: %s",
437 					    getIndentSpaces(8),
438 					    "None (Failed to Get"
439 					    " Attached Port)");
440 				}
441 			}
442 			(void *) fprintf(stdout, "\n");
443 			if (pflag & PRINT_TARGET_SCSI) {
444 
445 				if (configList->reportLUNsFailed) {
446 					(void *) fprintf(stdout,
447 					    "%s %016llx\n",
448 					    gettext("Error: Failed to get "
449 					    "ReportLun Data on"),
450 					    wwnConversion(TPListWalk->
451 					    sasattr.LocalSASAddress.wwn));
452 					ret++;
453 					continue;
454 				}
455 
456 				for (mapList = configList->map;
457 				    mapList != NULL; mapList = mapList->next) {
458 					(void *) fprintf(stdout, "%sLUN : %d\n",
459 					    getIndentSpaces(12),
460 					    mapList->osLUN);
461 					if (mapList->mappingExist) {
462 						(void *) fprintf(stdout,
463 						    "%sOS Device Name : %s\n",
464 						    getIndentSpaces(14),
465 						    (mapList->osDeviceName[0] ==
466 						    '\0') ?  "Not avaialble" :
467 						    mapList->osDeviceName);
468 					} else {
469 						(void *) fprintf(stdout,
470 						    "%sOS Device Name : %s\n",
471 						    getIndentSpaces(14), "No "
472 						    "matching OS Device "
473 						    "found.");
474 						ret++;
475 					}
476 		/* indentation changed here */
477 		if (mapList->inquiryFailed) {
478 			(void *) fprintf(stdout, "%s %s LUN %d\n",
479 			    gettext("Error: Failed to get Inquiry Data on"),
480 			    mapList->osDeviceName, mapList->osLUN);
481 			ret++;
482 		} else {
483 			(void *) fprintf(stdout, "%sVendor: ",
484 			    getIndentSpaces(14));
485 			for (count = sizeof (mapList->inq_vid), i = 0;
486 			    i < count; i++) {
487 				if (isprint(mapList->inq_vid[i]))
488 					(void *) fprintf(stdout, "%c",
489 					    mapList->inq_vid[i]);
490 			}
491 
492 			(void *) fprintf(stdout, "\n%sProduct: ",
493 			    getIndentSpaces(14));
494 			for (count = sizeof (mapList->inq_pid), i = 0;
495 			    i < count; i++) {
496 				if (isprint(mapList->inq_pid[i]))
497 					(void *) fprintf(stdout, "%c",
498 					    mapList->inq_pid[i]);
499 			}
500 
501 			(void *) fprintf(stdout, "\n%sDevice Type: %s\n",
502 			    getIndentSpaces(14),
503 			    getDTypeString(mapList->inq_dtype));
504 		}
505 		/* indentation changed back */
506 				}
507 			}
508 		}
509 	}
510 	return (ret);
511 }
512 
513 /*
514  * print the OS device name for the logical-unit object
515  *
516  * Arguments:
517  *	devListWalk - OS device path info
518  *	verbose - boolean indicating whether to display additional info
519  *
520  * returns:
521  * 	0 - we're good.
522  * 	>0 - we met issues.
523  */
524 int
525 printOSDeviceNameInfo(discoveredDevice *devListWalk, boolean_t verbose)
526 {
527 	portList		*portElem;
528 	tgtPortWWNList		*tgtWWNList;
529 	int			i, count;
530 	int			ret = 0;
531 
532 	(void *) fprintf(stdout, "OS Device Name: %s\n",
533 	    devListWalk->OSDeviceName);
534 	if (verbose == B_TRUE) {
535 		for (portElem = devListWalk->HBAPortList;
536 		    portElem != NULL; portElem = portElem->next) {
537 			(void *) fprintf(stdout, "%sHBA Port Name: ",
538 			    getIndentSpaces(4));
539 			(void *) fprintf(stdout, "%s", portElem->portName);
540 			for (tgtWWNList = portElem->tgtPortWWN;
541 			    tgtWWNList != NULL; tgtWWNList = tgtWWNList->next) {
542 				(void *) fprintf(stdout,
543 				    "\n%sTarget Port SAS Address: ",
544 				    getIndentSpaces(8));
545 				(void *) fprintf(stdout, "%016llx",
546 				    wwnConversion(tgtWWNList->portWWN.wwn));
547 				(void *) fprintf(stdout, "\n%sLUN: %u",
548 				    getIndentSpaces(12),
549 				    tgtWWNList->scsiOSLun);
550 			}
551 			(void *) fprintf(stdout, "\n");
552 		}
553 
554 		if (devListWalk->inquiryFailed) {
555 			(void *) fprintf(stdout, "%s %s\n",
556 			    gettext("Error: Failed to get Inquiry data "
557 			    "on device"), devListWalk->OSDeviceName);
558 			ret++;
559 		} else {
560 			(void *) fprintf(stdout, "%sVendor: ",
561 			    getIndentSpaces(4));
562 			for (count = sizeof (devListWalk->VID), i = 0;
563 			    i < count; i++) {
564 				if (isprint(devListWalk->VID[i]))
565 					(void *) fprintf(stdout, "%c",
566 					    devListWalk->VID[i]);
567 			}
568 
569 			(void *) fprintf(stdout, "\n%sProduct: ",
570 			    getIndentSpaces(4));
571 			for (count = sizeof (devListWalk->PID), i = 0;
572 			    i < count; i++) {
573 				if (isprint(devListWalk->PID[i]))
574 					(void *) fprintf(stdout, "%c",
575 					    devListWalk->PID[i]);
576 			}
577 
578 			(void *) fprintf(stdout, "\n%sDevice Type: %s\n",
579 			    getIndentSpaces(4),
580 			    getDTypeString(devListWalk->dType));
581 		}
582 	}
583 	return (ret);
584 }
585