11708Sstevel /*
21708Sstevel  * CDDL HEADER START
31708Sstevel  *
41708Sstevel  * The contents of this file are subject to the terms of the
5*5889Szk194757  * Common Development and Distribution License (the "License").
6*5889Szk194757  * You may not use this file except in compliance with the License.
71708Sstevel  *
81708Sstevel  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
91708Sstevel  * or http://www.opensolaris.org/os/licensing.
101708Sstevel  * See the License for the specific language governing permissions
111708Sstevel  * and limitations under the License.
121708Sstevel  *
131708Sstevel  * When distributing Covered Code, include this CDDL HEADER in each
141708Sstevel  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
151708Sstevel  * If applicable, add the following below this CDDL HEADER, with the
161708Sstevel  * fields enclosed by brackets "[]" replaced with your own identifying
171708Sstevel  * information: Portions Copyright [yyyy] [name of copyright owner]
181708Sstevel  *
191708Sstevel  * CDDL HEADER END
201708Sstevel  */
211708Sstevel /*
22*5889Szk194757  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
231708Sstevel  * Use is subject to license terms.
241708Sstevel  *
251708Sstevel  * Littleneck Platform specific functions.
261708Sstevel  *
271708Sstevel  * 	called when :
281708Sstevel  *      machine_type ==  MTYPE_LITTLENECK
291708Sstevel  *
301708Sstevel  */
311708Sstevel 
321708Sstevel #pragma ident	"%Z%%M%	%I%	%E% SMI"
331708Sstevel 
341708Sstevel #include <stdio.h>
351708Sstevel #include <stdlib.h>
361708Sstevel #include <unistd.h>
371708Sstevel #include <kstat.h>
381708Sstevel #include <string.h>
391708Sstevel #include <assert.h>
401708Sstevel #include <libintl.h>
411708Sstevel #include <note.h>
421708Sstevel #include <syslog.h>
431708Sstevel 
441708Sstevel #include <sys/openpromio.h>
451708Sstevel #include <sys/sysmacros.h>
461708Sstevel 
471708Sstevel #include <pdevinfo.h>
481708Sstevel #include <display.h>
491708Sstevel #include <pdevinfo_sun4u.h>
501708Sstevel #include <display_sun4u.h>
511708Sstevel #include <libprtdiag.h>
521708Sstevel 
531708Sstevel #include <picl.h>
541708Sstevel #include "workfile.c"
551708Sstevel 
561708Sstevel #define	LNECK_MAX_PS		2
571708Sstevel #define	LNECK_MAX_DISKS		2
581708Sstevel #define	LNECK_MAX_FANS		1
591708Sstevel 
601708Sstevel #ifndef	SCHIZO_COMPAT_PROP
611708Sstevel #define	SCHIZO_COMPAT_PROP	"pci108e,8001"
621708Sstevel #endif
631708Sstevel 
641708Sstevel /* Count of failed PSU's found */
651708Sstevel int ps_failure = 0;
661708Sstevel 
671708Sstevel /*
681708Sstevel  * Ignore first entry into disp_envc_status()
691708Sstevel  * from libprtdiag/common/display_sun4u.c
701708Sstevel  */
711708Sstevel int print_flag = 0;
721708Sstevel 
731708Sstevel /*
741708Sstevel  * these functions will overlay the symbol table of libprtdiag
751708Sstevel  * at runtime (workgroup server systems only)
761708Sstevel  */
771708Sstevel int	error_check(Sys_tree *tree, struct system_kstat_data *kstats);
781708Sstevel void	display_cpu_devices(Sys_tree *tree);
791708Sstevel void	display_pci(Board_node *board);
801708Sstevel void	display_io_cards(struct io_card *list);
811708Sstevel void	display_diaginfo(int flag, Prom_node *root, Sys_tree *tree,
821708Sstevel 				struct system_kstat_data *kstats);
831708Sstevel void	display_ffb(Board_node *board, int table);
841708Sstevel void	display_memoryconf(Sys_tree *tree, struct grp_info *grps);
851708Sstevel 
861708Sstevel /* local functions */
871708Sstevel static	int disp_envc_status(void);
881708Sstevel static	int lneck_env_print_temps(picl_nodehdl_t);
891708Sstevel static	int lneck_env_print_keyswitch(picl_nodehdl_t);
901708Sstevel static	int lneck_env_print_FSP_LEDS(picl_nodehdl_t);
911708Sstevel static	int lneck_env_print_disk(picl_nodehdl_t);
921708Sstevel static	int lneck_env_print_fans(picl_nodehdl_t);
931708Sstevel static	int lneck_env_print_ps(picl_nodehdl_t);
941708Sstevel 
951708Sstevel static void lneck_display_hw_revisions(Prom_node *root,
961708Sstevel 					Board_node *bnode);
971708Sstevel static void display_schizo_revisions(Board_node *bdlist);
981708Sstevel 
991708Sstevel /*
1001708Sstevel  * Defining the error_check function in order to return the
1011708Sstevel  * appropriate error code.
1021708Sstevel  */
1031708Sstevel /*ARGSUSED0*/
1041708Sstevel int
error_check(Sys_tree * tree,struct system_kstat_data * kstats)1051708Sstevel error_check(Sys_tree *tree, struct system_kstat_data *kstats)
1061708Sstevel {
1071708Sstevel 	int exit_code = 0;	/* init to all OK */
1081708Sstevel 	/* silently check for any types of machine errors */
1091708Sstevel 	print_flag = 0;
1101708Sstevel 	if (disp_fail_parts(tree) || disp_envc_status())
1111708Sstevel 		/* set exit_code to show failures */
1121708Sstevel 		exit_code = 1;
1131708Sstevel 
1141708Sstevel 	print_flag = 1;
1151708Sstevel 
1161708Sstevel 	return (exit_code);
1171708Sstevel }
1181708Sstevel 
1191708Sstevel void
display_cpu_devices(Sys_tree * tree)1201708Sstevel display_cpu_devices(Sys_tree *tree)
1211708Sstevel {
1221708Sstevel 	Board_node *bnode;
1231708Sstevel 
1241708Sstevel 	/*
1251708Sstevel 	 * Display the table header for CPUs . Then display the CPU
1261708Sstevel 	 * frequency, cache size, and processor revision of all cpus.
1271708Sstevel 	 */
1281708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
129*5889Szk194757 	    "\n"
130*5889Szk194757 	    "========================= CPUs "
131*5889Szk194757 	    "==============================================="
132*5889Szk194757 	    "\n"
133*5889Szk194757 	    "\n"
134*5889Szk194757 	    "          Run    E$    CPU     CPU  \n"
135*5889Szk194757 	    "Brd  CPU  MHz    MB   Impl.    Mask \n"
136*5889Szk194757 	    "---  ---  ----  ----  -------  ---- \n"));
1371708Sstevel 
1381708Sstevel 	/* Now display all of the cpus on each board */
1391708Sstevel 	bnode = tree->bd_list;
1401708Sstevel 	while (bnode != NULL) {
1411708Sstevel 		display_cpus(bnode);
1421708Sstevel 		bnode = bnode->next;
1431708Sstevel 	}
1441708Sstevel 
1451708Sstevel 	log_printf("\n");
1461708Sstevel }
1471708Sstevel 
1481708Sstevel 
1491708Sstevel /*
1501708Sstevel  * Display the CPUs present on this board.
1511708Sstevel  */
1521708Sstevel void
display_cpus(Board_node * board)1531708Sstevel display_cpus(Board_node *board)
1541708Sstevel {
1551708Sstevel 	Prom_node 	*cpu;
1561708Sstevel 	char		cpu_name[] = "cpu";
1571708Sstevel 
1581708Sstevel 	/*
1591708Sstevel 	 * display the CPUs' operating frequency, cache size, impl. field
1601708Sstevel 	 * and mask revision.
1611708Sstevel 	 */
1621708Sstevel 
1631708Sstevel 	for (cpu = dev_find_type(board->nodes, cpu_name); cpu != NULL;
1641708Sstevel 	    cpu = dev_next_type(cpu, cpu_name)) {
165*5889Szk194757 		uint_t freq;	 /* CPU clock frequency */
1661708Sstevel 		int ecache_size; /* External cache size */
1671708Sstevel 		int *mid;
1681708Sstevel 		int *impl;
1691708Sstevel 		int *mask;
1701708Sstevel 
1711708Sstevel 		mid = (int *)get_prop_val(find_prop(cpu, "portid"));
1721708Sstevel 		freq = LNECK_CLK_FREQ_TO_MHZ(get_cpu_freq(cpu));
1731708Sstevel 		ecache_size = get_ecache_size(cpu);
1741708Sstevel 		impl = (int *)get_prop_val(find_prop(cpu, "implementation#"));
1751708Sstevel 		mask = (int *)get_prop_val(find_prop(cpu, "mask#"));
1761708Sstevel 
1771708Sstevel 		/* Do not display a failed CPU node */
1781708Sstevel 		if ((freq != 0) && (node_failed(cpu) == 0)) {
1791708Sstevel 			/* Board number */
1801708Sstevel 			switch (*mid) {
1811708Sstevel 			case 1:
1821708Sstevel 				log_printf(dgettext(TEXT_DOMAIN,
1831708Sstevel 				" B   "));
1841708Sstevel 				break;
1851708Sstevel 			case 0:
1861708Sstevel 				log_printf(dgettext(TEXT_DOMAIN,
1871708Sstevel 				" A   "));
1881708Sstevel 				break;
1891708Sstevel 			default:
1901708Sstevel 				log_printf(dgettext(TEXT_DOMAIN, "X    "));
1911708Sstevel 			}
1921708Sstevel 
1931708Sstevel 			/* CPU MID */
1941708Sstevel 			log_printf("%2d   ", *mid);
1951708Sstevel 
1961708Sstevel 			/* Module number */
1971708Sstevel 
1981708Sstevel 			/* Running frequency */
199*5889Szk194757 			log_printf("%4u  ", freq);
2001708Sstevel 
2011708Sstevel 			/* Ecache size */
2021708Sstevel 			if (ecache_size == 0)
2031708Sstevel 				log_printf("N/A  ");
2041708Sstevel 			else
2051708Sstevel 				log_printf("%4.1f  ",
206*5889Szk194757 				    (float)ecache_size / (float)(1<<20));
2071708Sstevel 
2081708Sstevel 			/* Implementation */
2091708Sstevel 			if (impl == NULL) {
2101708Sstevel 				log_printf(dgettext(TEXT_DOMAIN, "%6s  "),
2111708Sstevel 				" N/A");
2121708Sstevel 			} else {
2131708Sstevel 				if (IS_CHEETAH(*impl))
2141708Sstevel 					log_printf("%-7s ", "US-III", 0);
2151708Sstevel 				else if (IS_CHEETAH_PLUS(*impl))
2161708Sstevel 					log_printf("%-7s ", "US-III+", 0);
2171708Sstevel 				else
2181708Sstevel 					log_printf("%-7x ", *impl, 0);
2191708Sstevel 			}
2201708Sstevel 
2211708Sstevel 			/* CPU Mask */
2221708Sstevel 			if (mask == NULL) {
2231708Sstevel 				log_printf(dgettext(TEXT_DOMAIN, " N/A   "));
2241708Sstevel 			} else {
2251708Sstevel 				log_printf(dgettext(TEXT_DOMAIN, " %d.%d   "),
2261708Sstevel 				    (*mask >> 4) & 0xf, *mask & 0xf);
2271708Sstevel 			}
2281708Sstevel 
2291708Sstevel 			log_printf("\n");
2301708Sstevel 		}
2311708Sstevel 	}
2321708Sstevel }
2331708Sstevel 
2341708Sstevel /*ARGSUSED0*/
2351708Sstevel void
display_memoryconf(Sys_tree * tree,struct grp_info * grps)2361708Sstevel display_memoryconf(Sys_tree *tree, struct grp_info *grps)
2371708Sstevel {
2381708Sstevel 	Board_node	*bnode = tree->bd_list;
2391708Sstevel 
2401708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
241*5889Szk194757 	    "========================= Memory Configuration"
242*5889Szk194757 	    " ===============================\n"
243*5889Szk194757 	    "\n           Logical  Logical  Logical "
244*5889Szk194757 	    "\n      MC   Bank     Bank     Bank         DIMM    "
245*5889Szk194757 	    "Interleave  Interleaved"
246*5889Szk194757 	    "\n Brd  ID   num      size     Status       Size    "
247*5889Szk194757 	    "Factor      with"
248*5889Szk194757 	    "\n----  ---  ----     ------   -----------  ------  "
249*5889Szk194757 	    "----------  -----------"));
2501708Sstevel 
2511708Sstevel 	while (bnode != NULL) {
2521708Sstevel 		if (get_us3_mem_regs(bnode)) {
2531708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
2541708Sstevel 			    "\nFailed to get memory information.\n"));
2551708Sstevel 			return;
2561708Sstevel 		}
2571708Sstevel 		bnode = bnode->next;
2581708Sstevel 	}
2591708Sstevel 
2601708Sstevel 	/* Display what we have found */
2611708Sstevel 	display_us3_banks();
2621708Sstevel }
2631708Sstevel 
2641708Sstevel /*ARGSUSED2*/
2651708Sstevel void
display_diaginfo(int flag,Prom_node * root,Sys_tree * tree,struct system_kstat_data * kstats)2661708Sstevel display_diaginfo(int flag, Prom_node *root, Sys_tree *tree,
2671708Sstevel 	struct system_kstat_data *kstats)
2681708Sstevel {
2691708Sstevel 	/*
2701708Sstevel 	 * Now display the last powerfail time and the fatal hardware
2711708Sstevel 	 * reset information. We do this under a couple of conditions.
2721708Sstevel 	 * First if the user asks for it. The second is iof the user
2731708Sstevel 	 * told us to do logging, and we found a system failure.
2741708Sstevel 	 */
2751708Sstevel 
2761708Sstevel 	if (flag) {
2771708Sstevel 		/*
2781708Sstevel 		 * display time of latest powerfail. Not all systems
2791708Sstevel 		 * have this capability. For those that do not, this
2801708Sstevel 		 * is just a no-op.
2811708Sstevel 		 */
2821708Sstevel 		disp_powerfail(root);
2831708Sstevel 
2841708Sstevel 		(void) disp_envc_status();
2851708Sstevel 
2861708Sstevel 		/* Hardware revision function calls */
2871708Sstevel 		lneck_display_hw_revisions(root, tree->bd_list);
2881708Sstevel 		log_printf("\n");
2891708Sstevel 	}
2901708Sstevel 	return;
2911708Sstevel 
2921708Sstevel }
2931708Sstevel 
2941708Sstevel /*
2951708Sstevel  * display_pci
2961708Sstevel  * Display all the PCI IO cards on this board.
2971708Sstevel  */
2981708Sstevel void
display_pci(Board_node * board)2991708Sstevel display_pci(Board_node *board)
3001708Sstevel {
3011708Sstevel 	struct io_card	*card_list = NULL;
3021708Sstevel 	struct io_card	card;
3031708Sstevel 	void		*value;
3041708Sstevel 	Prom_node	*pci;
3051708Sstevel 	Prom_node	*card_node;
3061708Sstevel 
3071708Sstevel 	char		*slot_name_arr[LNECK_MAX_SLOTS_PER_IO_BD] = {NULL};
3081708Sstevel 	int		i;
3091708Sstevel 
3101708Sstevel 	if (board == NULL)
3111708Sstevel 		return;
3121708Sstevel 
3131708Sstevel 	memset(&card, 0, sizeof (struct io_card));
3141708Sstevel 	/* Initialize all the common information */
3151708Sstevel 	card.display = TRUE;
3161708Sstevel 	card.board = board->board_num;
3171708Sstevel 
3181708Sstevel 	/*
3191708Sstevel 	 * Search for each pci instance, then find/display all nodes under
3201708Sstevel 	 * each instance node found.
3211708Sstevel 	 */
3221708Sstevel 	for (pci = dev_find_node_by_compat(board->nodes, SCHIZO_COMPAT_PROP);
323*5889Szk194757 	    pci != NULL;
324*5889Szk194757 	    pci = dev_next_node_by_compat(pci, SCHIZO_COMPAT_PROP)) {
3251708Sstevel 		(void) snprintf(card.bus_type, MAXSTRLEN,
326*5889Szk194757 		    dgettext(TEXT_DOMAIN, "PCI"));
3271708Sstevel 		/*
3281708Sstevel 		 * Get slot-name properties from parent node and
3291708Sstevel 		 * store them in an array.
3301708Sstevel 		 */
3311708Sstevel 		value = (char *)get_prop_val(
332*5889Szk194757 		    find_prop(pci, "slot-names"));
3331708Sstevel 
3341708Sstevel 		if (value != NULL) {
3351708Sstevel 			/* array starts after first int */
3361708Sstevel 			slot_name_arr[0] = (char *)value + sizeof (int);
3371708Sstevel 			for (i = 1; i < LNECK_MAX_SLOTS_PER_IO_BD; i++) {
3381708Sstevel 				slot_name_arr[i] = (char *)slot_name_arr[i - 1]
339*5889Szk194757 				    + strlen(slot_name_arr[i - 1]) +1;
3401708Sstevel 			}
3411708Sstevel 		}
3421708Sstevel 		/*
3431708Sstevel 		 * Search for Children of this node ie. Cards.
3441708Sstevel 		 * Note: any of these cards can be a pci-bridge
3451708Sstevel 		 *	that itself has children. If we find a
3461708Sstevel 		 *	pci-bridge we need to handle it specially.
3471708Sstevel 		 */
3481708Sstevel 		card_node = pci->child;
3491708Sstevel 		/* Generate the list of pci cards on pci instance: pci */
3501708Sstevel 		fill_pci_card_list(pci, card_node, &card, &card_list,
351*5889Szk194757 		    slot_name_arr);
3521708Sstevel 	} /* end-for */
3531708Sstevel 
3541708Sstevel 	display_io_cards(card_list);
3551708Sstevel 	free_io_cards(card_list);
3561708Sstevel 	log_printf("\n");
3571708Sstevel }
3581708Sstevel 
3591708Sstevel /*
3601708Sstevel  * Print out all the io cards in the list.  Also print the column
3611708Sstevel  * headers if told to do so.
3621708Sstevel  */
3631708Sstevel void
display_io_cards(struct io_card * list)3641708Sstevel display_io_cards(struct io_card *list)
3651708Sstevel {
3661708Sstevel 	static int banner = 0; /* Have we printed the column headings? */
3671708Sstevel 	struct io_card *p;
3681708Sstevel 
3691708Sstevel 	if (list == NULL) {
3701708Sstevel 		return;
3711708Sstevel 	}
3721708Sstevel 
3731708Sstevel 	if (banner == FALSE) {
3741708Sstevel 		log_printf(dgettext(TEXT_DOMAIN,
375*5889Szk194757 		    "                         Bus  Max\n"
376*5889Szk194757 		    "     IO   Port Bus       Freq Bus  Dev,\n"
377*5889Szk194757 		    "Brd  Type  ID  Side Slot MHz  Freq Func State "
378*5889Szk194757 		    "Name                              "));
3791708Sstevel #ifdef DEBUG
380*5889Szk194757 		log_printf(dgettext(TEXT_DOMAIN,
381*5889Szk194757 		    "Model                   Notes\n"));
3821708Sstevel #else
383*5889Szk194757 		log_printf(dgettext(TEXT_DOMAIN, "Model\n"));
3841708Sstevel #endif
3851708Sstevel 		/* ---------Node Brd  IO   Port Bus  Slot Bus  Max  Dev  Stat */
3861708Sstevel 		log_printf(dgettext(TEXT_DOMAIN,
387*5889Szk194757 		    "---- ---- ---- ---- ---- ---- ---- ---- ----- "
388*5889Szk194757 		    "--------------------------------  "
3891708Sstevel #ifdef DEBUG
390*5889Szk194757 		    "----------------------  "
3911708Sstevel #endif
392*5889Szk194757 		    "----------------------\n"));
3931708Sstevel 		banner = TRUE;
3941708Sstevel 	}
3951708Sstevel 
3961708Sstevel 	for (p = list; p != NULL; p = p -> next) {
3971708Sstevel 		log_printf(dgettext(TEXT_DOMAIN, "I/O   "));
3981708Sstevel 		log_printf(dgettext(TEXT_DOMAIN, "%-4s  "), p->bus_type);
3991708Sstevel 		log_printf(dgettext(TEXT_DOMAIN, "%-3d  "),
400*5889Szk194757 		    p->schizo_portid);
4011708Sstevel 		log_printf(dgettext(TEXT_DOMAIN, "%c    "), p->pci_bus);
4021708Sstevel 		log_printf(dgettext(TEXT_DOMAIN, "%-1s    "), p->slot_str);
4031708Sstevel 		log_printf(dgettext(TEXT_DOMAIN, "%-3d "), p->freq);
4041708Sstevel 		switch (p->pci_bus) {
4051708Sstevel 		case 'A':
4061708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, " 66  "));
4071708Sstevel 			break;
4081708Sstevel 		case 'B':
4091708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, " 33  "));
4101708Sstevel 			break;
4111708Sstevel 		default:
4121708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "  -  "));
4131708Sstevel 			break;
4141708Sstevel 		}
4151708Sstevel 
4161708Sstevel 		log_printf(dgettext(TEXT_DOMAIN, "%-1d,%-1d  "),
4171708Sstevel 		    p->dev_no, p->func_no);
4181708Sstevel 		log_printf(dgettext(TEXT_DOMAIN, "%-5s "), p->status);
4191708Sstevel 		log_printf(dgettext(TEXT_DOMAIN, "%-32.32s"), p->name);
4201708Sstevel 		if (strlen(p->name) > 32)
4211708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "+ "));
4221708Sstevel 		else
4231708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "  "));
4241708Sstevel 		log_printf(dgettext(TEXT_DOMAIN, "%-22.22s"), p->model);
4251708Sstevel 		if (strlen(p->model) > 22)
4261708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "+"));
4271708Sstevel #ifdef DEBUG
4281708Sstevel 		log_printf("%s  ", p->notes);
4291708Sstevel #endif
4301708Sstevel 		log_printf("\n");
4311708Sstevel 	}
4321708Sstevel }
4331708Sstevel 
4341708Sstevel /*
4351708Sstevel  * display_ffb
4361708Sstevel  *
4371708Sstevel  * There are no FFB's on a Littleneck, however in the generic library,
4381708Sstevel  * the display_ffb() function is implemented so we have to define an
4391708Sstevel  * empty function here.
4401708Sstevel  */
4411708Sstevel /*ARGSUSED0*/
4421708Sstevel void
display_ffb(Board_node * board,int table)4431708Sstevel display_ffb(Board_node *board, int table)
4441708Sstevel {}
4451708Sstevel 
4461708Sstevel 
4471708Sstevel /*
4481708Sstevel  * local functions
4491708Sstevel  */
4501708Sstevel 
4511708Sstevel /*
4521708Sstevel  * disp_fail_parts
4531708Sstevel  *
4541708Sstevel  * Display the failed parts in the system. This function looks for
4551708Sstevel  * the status property in all PROM nodes. On systems where
4561708Sstevel  * the PROM does not support passing diagnostic information
4571708Sstevel  * through the device tree, this routine will be silent.
4581708Sstevel  */
4591708Sstevel int
disp_fail_parts(Sys_tree * tree)4601708Sstevel disp_fail_parts(Sys_tree *tree)
4611708Sstevel {
4621708Sstevel 	int exit_code = 0;
4631708Sstevel 	int system_failed = 0;
4641708Sstevel 	Board_node *bnode = tree->bd_list;
4651708Sstevel 	Prom_node *pnode;
4661708Sstevel 
4671708Sstevel 	/* go through all of the boards looking for failed units. */
4681708Sstevel 	while (bnode != NULL) {
4691708Sstevel 		/* find failed chips */
4701708Sstevel 		pnode = find_failed_node(bnode->nodes);
4711708Sstevel 		if ((pnode != NULL) && !system_failed) {
4721708Sstevel 			system_failed = 1;
4731708Sstevel 			exit_code = 1;
4741708Sstevel 			if (print_flag == 0) {
4751708Sstevel 				return (exit_code);
4761708Sstevel 			}
4771708Sstevel 			log_printf("\n");
4781708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "Failed Field "
479*5889Szk194757 			    "Replaceable Units (FRU) in System:\n"));
4801708Sstevel 			log_printf("=========================="
481*5889Szk194757 			    "====================\n");
4821708Sstevel 		}
4831708Sstevel 		while (pnode != NULL) {
4841708Sstevel 			void *value;
4851708Sstevel 			char *name;		/* node name string */
4861708Sstevel 			char *type;		/* node type string */
4871708Sstevel 			char *board_type = NULL;
4881708Sstevel 
4891708Sstevel 			value = get_prop_val(find_prop(pnode, "status"));
4901708Sstevel 			name = get_node_name(pnode);
4911708Sstevel 
4921708Sstevel 			/* sanity check of data retrieved from PROM */
4931708Sstevel 			if ((value == NULL) || (name == NULL)) {
4941708Sstevel 				pnode = next_failed_node(pnode);
4951708Sstevel 				continue;
4961708Sstevel 			}
4971708Sstevel 
4981708Sstevel 			/* Find the board type of this board */
4991708Sstevel 			if (bnode->board_type == CPU_BOARD) {
5001708Sstevel 				board_type = "CPU";
5011708Sstevel 			} else {
5021708Sstevel 				board_type = "IO";
5031708Sstevel 			}
5041708Sstevel 
5051708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "%s unavailable "
506*5889Szk194757 			    "on %s Board #%d\n"), name, board_type,
507*5889Szk194757 			    bnode->board_num);
5081708Sstevel 
5091708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
510*5889Szk194757 			    "\tPROM fault string: %s\n"), value);
5111708Sstevel 
5121708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
513*5889Szk194757 			    "\tFailed Field Replaceable Unit is "));
5141708Sstevel 
5151708Sstevel 			/*
5161708Sstevel 			 * Determine whether FRU is CPU module, system
5171708Sstevel 			 * board, or SBus card.
5181708Sstevel 			 */
5191708Sstevel 			if ((name != NULL) && (strstr(name, "sbus"))) {
5201708Sstevel 
5211708Sstevel 				log_printf(dgettext(TEXT_DOMAIN,
522*5889Szk194757 				    "SBus Card %d\n"),
523*5889Szk194757 				    get_sbus_slot(pnode));
5241708Sstevel 
5251708Sstevel 			} else if (((name = get_node_name(pnode->parent)) !=
526*5889Szk194757 			    NULL) && (strstr(name, "pci"))) {
5271708Sstevel 
5281708Sstevel 				log_printf(dgettext(TEXT_DOMAIN,
529*5889Szk194757 				    "PCI Card %d"),
530*5889Szk194757 				    get_pci_device(pnode));
5311708Sstevel 
5321708Sstevel 			} else if (((type = get_node_type(pnode)) != NULL) &&
533*5889Szk194757 			    (strstr(type, "cpu"))) {
5341708Sstevel 
5351708Sstevel 				log_printf(dgettext(TEXT_DOMAIN, "UltraSPARC "
536*5889Szk194757 				    "module Board %d Module %d\n"), 0,
537*5889Szk194757 				    get_id(pnode));
5381708Sstevel 
5391708Sstevel 			} else {
5401708Sstevel 				log_printf(dgettext(TEXT_DOMAIN,
541*5889Szk194757 				    "%s board %d\n"), board_type,
542*5889Szk194757 				    bnode->board_num);
5431708Sstevel 			}
5441708Sstevel 			pnode = next_failed_node(pnode);
5451708Sstevel 		}
5461708Sstevel 		bnode = bnode->next;
5471708Sstevel 	}
5481708Sstevel 
5491708Sstevel 	if (!system_failed) {
5501708Sstevel 		log_printf(dgettext(TEXT_DOMAIN,
551*5889Szk194757 		    "No failures found in System\n"));
5521708Sstevel 		log_printf("===========================\n\n");
5531708Sstevel 		return (0);
5541708Sstevel 	} else {
5551708Sstevel 		return (1);
5561708Sstevel 	}
5571708Sstevel }
5581708Sstevel 
5591708Sstevel 
5601708Sstevel /*
5611708Sstevel  * disp_envc_status
5621708Sstevel  *
5631708Sstevel  * This routine displays the environmental status passed up from
5641708Sstevel  * device drivers via the envlibobj.so library.
5651708Sstevel  * This is a Littleneck specific environmental information display routine.
5661708Sstevel  */
5671708Sstevel static int
disp_envc_status(void)5681708Sstevel disp_envc_status(void)
5691708Sstevel {
5701708Sstevel 	int err;
5711708Sstevel 	char *system = "SYSTEM";
5721708Sstevel 	picl_nodehdl_t system_node, root;
5731708Sstevel 
5741708Sstevel 	log_printf("\n");
5751708Sstevel 	log_printf(dgettext(TEXT_DOMAIN, "========================="
576*5889Szk194757 	    " Environmental Status =========================\n\n"));
5771708Sstevel 
5781708Sstevel 	err = picl_initialize();
5791708Sstevel 	if (err != PICL_SUCCESS) {
5801708Sstevel 		log_printf(dgettext(TEXT_DOMAIN,
581*5889Szk194757 		    "Cannot print environmental information\n"
582*5889Szk194757 		    "picl_initialize failed\n"
583*5889Szk194757 		    "%s\n"), picl_strerror(err));
5841708Sstevel 	}
5851708Sstevel 
5861708Sstevel 	if (err == PICL_SUCCESS) {
5871708Sstevel 		err = picl_get_root(&root);
5881708Sstevel 		err = find_child_device(root, system, &system_node);
5891708Sstevel 		if (err != PICL_SUCCESS) {
5901708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
591*5889Szk194757 			    "Cannot print environmental information\n"
592*5889Szk194757 			    "find_child_device for the SYSTEM node "
593*5889Szk194757 			    "failed\n"
594*5889Szk194757 			    "%s\n"), picl_strerror(err));
5951708Sstevel 		}
5961708Sstevel 
5971708Sstevel 		if ((err = lneck_env_print_temps(system_node)) !=
5981708Sstevel 		    PICL_SUCCESS) {
5991708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
600*5889Szk194757 			    "Temperature Checking failed: %s\n"),
601*5889Szk194757 			    picl_strerror(err));
6021708Sstevel 		}
6031708Sstevel 		if ((err = lneck_env_print_keyswitch(system_node)) !=
6041708Sstevel 		    PICL_SUCCESS) {
6051708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
606*5889Szk194757 			    "Keyswitch information checking failed: %s\n"),
607*5889Szk194757 			    picl_strerror(err));
6081708Sstevel 		}
6091708Sstevel 		if ((err = lneck_env_print_FSP_LEDS(system_node)) !=
6101708Sstevel 		    PICL_SUCCESS) {
6111708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
612*5889Szk194757 			    "FSP LED information checking failed: %s\n"),
613*5889Szk194757 			    picl_strerror(err));
6141708Sstevel 		}
6151708Sstevel 		if ((err = lneck_env_print_disk(system_node)) !=
6161708Sstevel 		    PICL_SUCCESS) {
6171708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
618*5889Szk194757 			    "Disk information checking failed: %s\n"),
619*5889Szk194757 			    picl_strerror(err));
6201708Sstevel 		}
6211708Sstevel 		if ((err = lneck_env_print_fans(system_node)) !=
6221708Sstevel 		    PICL_SUCCESS) {
6231708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
624*5889Szk194757 			    "Fan information checking failed: %s\n"),
625*5889Szk194757 			    picl_strerror(err));
6261708Sstevel 		}
6271708Sstevel 		if ((err = lneck_env_print_ps(system_node)) !=
6281708Sstevel 		    PICL_SUCCESS) {
6291708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
630*5889Szk194757 			    "Power Supply information checking failed: "
631*5889Szk194757 			    "%s\n"), picl_strerror(err));
6321708Sstevel 		} else if (ps_failure != 0)
6331708Sstevel 			err = PICL_FAILURE;
6341708Sstevel 	}
6351708Sstevel 	return (err);
6361708Sstevel }
6371708Sstevel 
6381708Sstevel int
lneck_env_print_ps(picl_nodehdl_t system_node)6391708Sstevel lneck_env_print_ps(picl_nodehdl_t system_node)
6401708Sstevel {
6411708Sstevel 	int		i, err = 0;
6421708Sstevel 	int32_t		number;
6431708Sstevel 	picl_nodehdl_t	*ps;
6441708Sstevel 	picl_nodehdl_t	ps_fail[2], ps_type[2];
6451708Sstevel 	char		name[PICL_PROPNAMELEN_MAX];
6461708Sstevel 	boolean_t	type;
6471708Sstevel 	char		fault_state[PICL_PROPNAMELEN_MAX];
6481708Sstevel 
6491708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
650*5889Szk194757 	    "Power Supplies:\n"
651*5889Szk194757 	    "---------------\n"
652*5889Szk194757 	    "Supply     Status         PS Type\n"
653*5889Szk194757 	    "------     ------      ---------------\n"));
6541708Sstevel 	err = fill_device_array_from_id(system_node, "PSVC_PS", &number,
655*5889Szk194757 	    &ps);
6561708Sstevel 	if (err != PICL_SUCCESS) {
6571708Sstevel 		return (err);
6581708Sstevel 	}
6591708Sstevel 
6601708Sstevel 	for (i = 0; i < LNECK_MAX_PS; i++) {
6611708Sstevel 		err = picl_get_propval_by_name(ps[i], PICL_PROP_NAME, name,
662*5889Szk194757 		    PICL_PROPNAMELEN_MAX);
6631708Sstevel 		if (err == PICL_SUCCESS) {
6641708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "%6-s"), name);
6651708Sstevel 		} else continue;
6661708Sstevel 
6671708Sstevel 		err = picl_get_propval_by_name(ps[i], "FaultInformation",
668*5889Szk194757 		    fault_state, PICL_PROPNAMELEN_MAX);
6691708Sstevel 		if (err == PICL_SUCCESS) {
6701708Sstevel 			if ((strlen(fault_state) == 0) ||
671*5889Szk194757 			    (strcmp(fault_state, "NO_FAULT") == 0)) {
6721708Sstevel 				strcpy(fault_state, "OK");
6731708Sstevel 			} else
6741708Sstevel 				/*
6751708Sstevel 				 * Bump up count if fault_state	 !OK
6761708Sstevel 				 */
6771708Sstevel 				ps_failure++;
6781708Sstevel 
6791708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "    [%-6s] "),
680*5889Szk194757 			    fault_state);
6811708Sstevel 		} else {
6821708Sstevel 			return (err);
6831708Sstevel 		}
6841708Sstevel 
6851708Sstevel 		err = fill_device_from_id(ps[i], "PSVC_DEV_FAULT_SENSOR",
686*5889Szk194757 		    &ps_fail[i]);
6871708Sstevel 		if (err != PICL_SUCCESS) {
6881708Sstevel 			return (err);
6891708Sstevel 		}
6901708Sstevel 
6911708Sstevel 		err = fill_device_from_id(ps[i], "PSVC_DEV_TYPE_SENSOR",
692*5889Szk194757 		    &ps_type[i]);
6931708Sstevel 		if (err != PICL_SUCCESS) {
6941708Sstevel 			return (err);
6951708Sstevel 		}
6961708Sstevel 		err = picl_get_propval_by_name(ps_type[i], "Gpio-value", &type,
697*5889Szk194757 		    sizeof (boolean_t));
6981708Sstevel 		if (err == PICL_SUCCESS) {
6991708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "    [%13s]"),
700*5889Szk194757 			    type == 0 ? "Quahog/Razor" : "Sun-Fire-280R");
7011708Sstevel 			if (type == 0) {
7021708Sstevel 				log_printf(dgettext(TEXT_DOMAIN,
703*5889Szk194757 				    "WARNING: PS is of the wrong type\n"));
7041708Sstevel 			} else log_printf("\n");
7051708Sstevel 		} else {
7061708Sstevel 			return (err);
7071708Sstevel 		}
7081708Sstevel 
7091708Sstevel 	}
7101708Sstevel 
7111708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
712*5889Szk194757 	    "\n"
713*5889Szk194757 	    "================================="
714*5889Szk194757 	    "\n"
715*5889Szk194757 	    "\n"));
7161708Sstevel 
7171708Sstevel 	/*
7181708Sstevel 	 * Do not display an error message just because PS1 is
7191708Sstevel 	 * not present.
7201708Sstevel 	 */
7211708Sstevel 	if (err == PICL_INVALIDHANDLE) {
7221708Sstevel 		err = PICL_SUCCESS;
7231708Sstevel 	}
7241708Sstevel 
7251708Sstevel 	return (err);
7261708Sstevel }
7271708Sstevel 
7281708Sstevel int
lneck_env_print_fans(picl_nodehdl_t system_node)7291708Sstevel lneck_env_print_fans(picl_nodehdl_t system_node) {
7301708Sstevel 	int		i, err = 0;
7311708Sstevel 	int32_t		number;
7321708Sstevel 	picl_nodehdl_t	*fans;
7331708Sstevel 	picl_nodehdl_t	fan_fault[1];
7341708Sstevel 	char		fault_state[PICL_PROPNAMELEN_MAX];
7351708Sstevel 	char		name[PICL_PROPNAMELEN_MAX];
7361708Sstevel 
7371708Sstevel 	err = fill_device_array_from_id(system_node, "PSVC_FAN", &number,
7381708Sstevel 				&fans);
7391708Sstevel 	if (err != PICL_SUCCESS) {
7401708Sstevel 		return (err);
7411708Sstevel 	}
7421708Sstevel 
7431708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
7441708Sstevel 		"\n"
7451708Sstevel 		"=================================\n"
7461708Sstevel 		"\n"
7471708Sstevel 		"Fan Bank :\n"
7481708Sstevel 		"----------\n"
7491708Sstevel 		"\n"
7501708Sstevel 		"Bank                        Status\n"
7511708Sstevel 		"----                        -------\n"));
7521708Sstevel 
7531708Sstevel 	for (i = 0; i < LNECK_MAX_FANS; i++) {
7541708Sstevel 		err = picl_get_propval_by_name(fans[i], PICL_PROP_NAME, name,
7551708Sstevel 				PICL_PROPNAMELEN_MAX);
7561708Sstevel 		if (err == PICL_SUCCESS) {
7571708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "%16-s"), name);
7581708Sstevel 		} else continue;
7591708Sstevel 
7601708Sstevel 		err = fill_device_from_id(fans[i], "PSVC_DEV_FAULT_SENSOR",
7611708Sstevel 				&fan_fault[i]);
7621708Sstevel 		if (err != PICL_SUCCESS) {
7631708Sstevel 			return (err);
7641708Sstevel 		}
7651708Sstevel 
7661708Sstevel 		err = picl_get_propval_by_name(fans[i], "FaultInformation",
7671708Sstevel 			&fault_state, PICL_PROPNAMELEN_MAX);
7681708Sstevel 
7691708Sstevel 		if (err == PICL_SUCCESS) {
7701708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "            [%3s]\n"),
7711708Sstevel 				fault_state);
7721708Sstevel 		} else {
7731708Sstevel 		    return (err);
7741708Sstevel 		}
7751708Sstevel 	}
7761708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
7771708Sstevel 		"\n"
7781708Sstevel 		"================================="
7791708Sstevel 		"\n"
7801708Sstevel 		"\n"));
7811708Sstevel 
7821708Sstevel 	return (err);
7831708Sstevel }
7841708Sstevel 
7851708Sstevel int
lneck_env_print_disk(picl_nodehdl_t system_node)7861708Sstevel lneck_env_print_disk(picl_nodehdl_t system_node) {
7871708Sstevel 	int		i, err = 0;
7881708Sstevel 	int32_t		number;
7891708Sstevel 	picl_nodehdl_t	*disks;
7901708Sstevel 	char		fault_state[PICL_PROPNAMELEN_MAX];
7911708Sstevel 	char		name[PICL_PROPNAMELEN_MAX];
7921708Sstevel 
7931708Sstevel 	err = fill_device_array_from_id(system_node, "PSVC_DISK", &number,
7941708Sstevel 				&disks);
7951708Sstevel 	if (err != PICL_SUCCESS) {
7961708Sstevel 		return (err);
7971708Sstevel 	}
7981708Sstevel 
7991708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
8001708Sstevel 		"Disk Status:\n"
8011708Sstevel 		"          Presence      Fault Value\n"
8021708Sstevel 		"          --------      -----------\n"));
8031708Sstevel 
8041708Sstevel 	for (i = 0; i < LNECK_MAX_DISKS; i++) {
8051708Sstevel 		err = picl_get_propval_by_name(disks[i], PICL_PROP_NAME, name,
8061708Sstevel 				PICL_PROPNAMELEN_MAX);
8071708Sstevel 		switch (err) {
8081708Sstevel 		case PICL_SUCCESS:
8091708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
8101708Sstevel 				"DISK  %2d: [PRESENT]"), i);
8111708Sstevel 			break;
8121708Sstevel 		case PICL_INVALIDHANDLE:
8131708Sstevel 			log_printf(dgettext(TEXT_DOMAIN,
8141708Sstevel 				"DISK  %2d: [EMPTY  ]\n"), i);
8151708Sstevel 			continue;
8161708Sstevel 		default:
8171708Sstevel 		    return (err);
8181708Sstevel 		}
8191708Sstevel 		err = picl_get_propval_by_name(disks[i], "FaultInformation",
8201708Sstevel 			&fault_state, PICL_PROPNAMELEN_MAX);
8211708Sstevel 		if (err == PICL_SUCCESS) {
8221708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "     [%3s]"),
8231708Sstevel 				fault_state);
8241708Sstevel 		} else {
8251708Sstevel 			if (err != PICL_INVALIDHANDLE)
8261708Sstevel 				return (err);
8271708Sstevel 		}
8281708Sstevel 		log_printf("\n");
8291708Sstevel 	}
8301708Sstevel 
8311708Sstevel 	if (err == PICL_INVALIDHANDLE) {
8321708Sstevel 		err = PICL_SUCCESS;
8331708Sstevel 	}
8341708Sstevel 
8351708Sstevel 	return (err);
8361708Sstevel }
8371708Sstevel 
8381708Sstevel int
lneck_env_print_FSP_LEDS(picl_nodehdl_t system_node)8391708Sstevel lneck_env_print_FSP_LEDS(picl_nodehdl_t system_node) {
8401708Sstevel 	int		err;
8411708Sstevel 	int32_t		number;
8421708Sstevel 	picl_nodehdl_t	*fsp_led;
8431708Sstevel 	char		fault_state[PICL_PROPNAMELEN_MAX];
8441708Sstevel 
8451708Sstevel 	err = fill_device_array_from_id(system_node, "PSVC_FSP_LED", &number,
8461708Sstevel 				&fsp_led);
8471708Sstevel 	if (err != PICL_SUCCESS) {
8481708Sstevel 		return (err);
8491708Sstevel 	}
8501708Sstevel 
8511708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
8521708Sstevel 		"System LED Status: POWER                   GEN FAULT\n"
8531708Sstevel 		"                   [ ON]"));
8541708Sstevel 	err = picl_get_propval_by_name(fsp_led[0], "State", &fault_state,
8551708Sstevel 		PICL_PROPNAMELEN_MAX);
8561708Sstevel 	if (err == PICL_SUCCESS) {
8571708Sstevel 		log_printf("                    [%3s]", fault_state);
8581708Sstevel 	} else {
8591708Sstevel 		return (err);
8601708Sstevel 	}
8611708Sstevel 
8621708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
8631708Sstevel 		"\n"
8641708Sstevel 		"\n"
8651708Sstevel 		"================================="
8661708Sstevel 		"\n"
8671708Sstevel 		"\n"));
8681708Sstevel 
8691708Sstevel 	return (err);
8701708Sstevel }
8711708Sstevel 
8721708Sstevel int
lneck_env_print_keyswitch(picl_nodehdl_t system_node)8731708Sstevel lneck_env_print_keyswitch(picl_nodehdl_t system_node) {
8741708Sstevel 	int		err = 0;
8751708Sstevel 	picl_nodehdl_t	*keyswitch;
8761708Sstevel 	int32_t		number;
8771708Sstevel 	char		ks_pos[PICL_PROPNAMELEN_MAX];
8781708Sstevel 
8791708Sstevel 	err = fill_device_array_from_id(system_node, "PSVC_KEYSWITCH", &number,
8801708Sstevel 				&keyswitch);
8811708Sstevel 	if (err != PICL_SUCCESS) {
8821708Sstevel 		return (err);
8831708Sstevel 	}
8841708Sstevel 	err = picl_get_propval_by_name(keyswitch[0], "State", ks_pos,
8851708Sstevel 		PICL_PROPNAMELEN_MAX);
8861708Sstevel 	if (err != PICL_SUCCESS) {
8871708Sstevel 		return (err);
8881708Sstevel 	}
8891708Sstevel 
8901708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
8911708Sstevel 		"Front Status Panel:\n"
8921708Sstevel 		"-------------------\n"
8931708Sstevel 		"Keyswitch position: %s\n"), ks_pos);
8941708Sstevel 	log_printf("\n");
8951708Sstevel 
8961708Sstevel 	return (err);
8971708Sstevel }
8981708Sstevel 
8991708Sstevel int
lneck_env_print_temps(picl_nodehdl_t system_node)9001708Sstevel lneck_env_print_temps(picl_nodehdl_t system_node) {
9011708Sstevel 	int		i, err = 0;
9021708Sstevel 	picl_nodehdl_t	*system_ts_nodes;
9031708Sstevel 	int32_t		temp, number;
9041708Sstevel 
9051708Sstevel 	err = fill_device_array_from_id(system_node, "PSVC_TS", &number,
9061708Sstevel 				&system_ts_nodes);
9071708Sstevel 	if (err != PICL_SUCCESS) {
9081708Sstevel 		return (err);
9091708Sstevel 	}
9101708Sstevel 
9111708Sstevel 
9121708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
9131708Sstevel 		"System Temperatures (Celsius):\n"
9141708Sstevel 		"------------------------------\n"
9151708Sstevel 		"cpu0   1 \n"
9161708Sstevel 		"---------\n"));
9171708Sstevel 
9181708Sstevel 	for (i = 0; i < 2; i++) {
9191708Sstevel 		err = picl_get_propval_by_name(system_ts_nodes[i],
9201708Sstevel 				"Temperature", &temp, sizeof (temp));
9211708Sstevel 		if (err == PICL_SUCCESS) {
9221708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "  %02d"), temp);
9231708Sstevel 		} else {
9241708Sstevel 			if (err == PICL_INVALIDHANDLE) {
9251708Sstevel 				err = PICL_SUCCESS;
9261708Sstevel 				log_printf(dgettext(TEXT_DOMAIN, "  xx"));
9271708Sstevel 			} else {
9281708Sstevel 				return (err);
9291708Sstevel 			}
9301708Sstevel 		}
9311708Sstevel 	}
9321708Sstevel 
9331708Sstevel 	log_printf("\n");
9341708Sstevel 	log_printf("\n");
9351708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
9361708Sstevel 	"=================================\n"));
9371708Sstevel 	log_printf("\n");
9381708Sstevel 
9391708Sstevel 	return (err);
9401708Sstevel }
9411708Sstevel 
9421708Sstevel static void
lneck_display_hw_revisions(Prom_node * root,Board_node * bdlist)9431708Sstevel lneck_display_hw_revisions(Prom_node *root, Board_node *bdlist)
9441708Sstevel {
9451708Sstevel 	Prom_node	*pnode;
9461708Sstevel 	char		*value;
9471708Sstevel 
9481708Sstevel 	log_printf(dgettext(TEXT_DOMAIN, "\n"
949*5889Szk194757 	    "========================= HW Revisions "
950*5889Szk194757 	    "=======================================\n\n"));
9511708Sstevel 
9521708Sstevel 	log_printf(dgettext(TEXT_DOMAIN,
953*5889Szk194757 	    "System PROM revisions:\n"
954*5889Szk194757 	    "----------------------\n"));
9551708Sstevel 
9561708Sstevel 	pnode = dev_find_node(root, "openprom");
9571708Sstevel 	if (pnode != NULL) {
958*5889Szk194757 		value = (char *)get_prop_val(find_prop(pnode, "version"));
959*5889Szk194757 		log_printf(value);
9601708Sstevel 	}
9611708Sstevel 
9621708Sstevel 	log_printf(dgettext(TEXT_DOMAIN, "\n\n"
963*5889Szk194757 	    "IO ASIC revisions:\n"
964*5889Szk194757 	    "------------------\n"
965*5889Szk194757 	    "                     Port\n"
966*5889Szk194757 	    "Model     ID  Status Version\n"
967*5889Szk194757 	    "-------- ---- ------ -------\n"));
9681708Sstevel 
9691708Sstevel 	display_schizo_revisions(bdlist);
9701708Sstevel }
9711708Sstevel 
9721708Sstevel static void
display_schizo_revisions(Board_node * bdlist)9731708Sstevel display_schizo_revisions(Board_node *bdlist)
9741708Sstevel {
9751708Sstevel 	Prom_node	*pnode;
9761708Sstevel 	int		*int_val;
9771708Sstevel 	int		portid;
9781708Sstevel 	int		prev_portid = -1;
9791708Sstevel 	char		*status_a = NULL;
9801708Sstevel 	char		*status_b = NULL;
9811708Sstevel 	int		revision;
9821708Sstevel #ifdef DEBUG
9831708Sstevel 	uint32_t	a_notes, b_notes;
9841708Sstevel #endif
9851708Sstevel 	int		pci_bus;
9861708Sstevel 	Board_node	*bnode;
9871708Sstevel 	bnode = bdlist;
9881708Sstevel 
9891708Sstevel 	while (bnode != NULL) {
9901708Sstevel 		/*
9911708Sstevel 		 * search this board node for all Schizos
9921708Sstevel 		 */
9931708Sstevel 
9941708Sstevel 		for (pnode = dev_find_node_by_compat(bnode->nodes,
995*5889Szk194757 		    SCHIZO_COMPAT_PROP); pnode != NULL;
996*5889Szk194757 		    pnode = dev_next_node_by_compat(pnode,
997*5889Szk194757 		    SCHIZO_COMPAT_PROP)) {
9981708Sstevel 
9991708Sstevel 			/*
10001708Sstevel 			 * get the reg property to determine
10011708Sstevel 			 * whether we are looking at side A or B
10021708Sstevel 			 */
10031708Sstevel 
10041708Sstevel 			int_val = (int *)get_prop_val
1005*5889Szk194757 			    (find_prop(pnode, "reg"));
10061708Sstevel 			if (int_val != NULL) {
10071708Sstevel 				int_val ++; /* second integer in array */
10081708Sstevel 				pci_bus = ((*int_val) & 0x7f0000);
10091708Sstevel 			}
10101708Sstevel 
10111708Sstevel 			/* get portid */
10121708Sstevel 			int_val = (int *)get_prop_val
1013*5889Szk194757 			    (find_prop(pnode, "portid"));
10141708Sstevel 			if (int_val == NULL)
10151708Sstevel 				continue;
10161708Sstevel 
10171708Sstevel 			portid = *int_val;
10181708Sstevel 
10191708Sstevel 			/*
10201708Sstevel 			 * If this is a new portid and it is PCI bus B,
10211708Sstevel 			 * we skip onto the PCI bus A.
10221708Sstevel 			 */
10231708Sstevel 			if ((portid != prev_portid) && (pci_bus == 0x700000)) {
10241708Sstevel 				prev_portid = portid;
10251708Sstevel 				/* status */
10261708Sstevel 				status_b = (char *)get_prop_val
10271708Sstevel 				    (find_prop(pnode, "status"));
10281708Sstevel #ifdef DEBUG
10291708Sstevel 				b_notes = pci_bus;
10301708Sstevel #endif
10311708Sstevel 				continue; /* skip to the next schizo */
10321708Sstevel 			}
10331708Sstevel 
10341708Sstevel 			/*
10351708Sstevel 			 * This must be side A of the same Schizo.
10361708Sstevel 			 * Gather all its props and display them.
10371708Sstevel 			 */
10381708Sstevel #ifdef DEBUG
10391708Sstevel 			a_notes = pci_bus;
10401708Sstevel #endif
10411708Sstevel 
10421708Sstevel 			prev_portid = portid;
10431708Sstevel 
10441708Sstevel 			int_val = (int *)get_prop_val
1045*5889Szk194757 			    (find_prop(pnode, "version#"));
10461708Sstevel 			if (int_val != NULL)
10471708Sstevel 				revision = *int_val;
10481708Sstevel 			else
10491708Sstevel 				revision = -1;
10501708Sstevel 
10511708Sstevel 			status_a = (char *)get_prop_val(find_prop
1052*5889Szk194757 			    (pnode, "status"));
10531708Sstevel 
10541708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "Schizo    "));
10551708Sstevel 
10561708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, "%-3d "), portid, 0);
10571708Sstevel 
10581708Sstevel 
10591708Sstevel 			log_printf((status_a == NULL && status_b == NULL) ?
1060*5889Szk194757 			    dgettext(TEXT_DOMAIN, "  ok  ") :
1061*5889Szk194757 			    dgettext(TEXT_DOMAIN, " fail "));
10621708Sstevel 
10631708Sstevel 			log_printf(dgettext(TEXT_DOMAIN, " %4d   "),
10641708Sstevel 			    revision);
10651708Sstevel #ifdef DEBUG
10661708Sstevel 			log_printf(" 0x%x 0x%x", a_notes, b_notes);
10671708Sstevel #endif
10681708Sstevel 			log_printf("\n");
10691708Sstevel 		}
10701708Sstevel 		bnode = bnode->next;
10711708Sstevel 	}
10721708Sstevel }
1073