xref: /onnv-gate/usr/src/psm/stand/boot/sparc/common/machdep.c (revision 11498:2fc576d74742)
1*11498SJerry.Gilliam@Sun.COM /*
2*11498SJerry.Gilliam@Sun.COM  * CDDL HEADER START
3*11498SJerry.Gilliam@Sun.COM  *
4*11498SJerry.Gilliam@Sun.COM  * The contents of this file are subject to the terms of the
5*11498SJerry.Gilliam@Sun.COM  * Common Development and Distribution License (the "License").
6*11498SJerry.Gilliam@Sun.COM  * You may not use this file except in compliance with the License.
7*11498SJerry.Gilliam@Sun.COM  *
8*11498SJerry.Gilliam@Sun.COM  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9*11498SJerry.Gilliam@Sun.COM  * or http://www.opensolaris.org/os/licensing.
10*11498SJerry.Gilliam@Sun.COM  * See the License for the specific language governing permissions
11*11498SJerry.Gilliam@Sun.COM  * and limitations under the License.
12*11498SJerry.Gilliam@Sun.COM  *
13*11498SJerry.Gilliam@Sun.COM  * When distributing Covered Code, include this CDDL HEADER in each
14*11498SJerry.Gilliam@Sun.COM  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15*11498SJerry.Gilliam@Sun.COM  * If applicable, add the following below this CDDL HEADER, with the
16*11498SJerry.Gilliam@Sun.COM  * fields enclosed by brackets "[]" replaced with your own identifying
17*11498SJerry.Gilliam@Sun.COM  * information: Portions Copyright [yyyy] [name of copyright owner]
18*11498SJerry.Gilliam@Sun.COM  *
19*11498SJerry.Gilliam@Sun.COM  * CDDL HEADER END
20*11498SJerry.Gilliam@Sun.COM  */
21*11498SJerry.Gilliam@Sun.COM /*
22*11498SJerry.Gilliam@Sun.COM  * Copyright 2010 Sun Microsystems, Inc.  All rights reserved.
23*11498SJerry.Gilliam@Sun.COM  * Use is subject to license terms.
24*11498SJerry.Gilliam@Sun.COM  */
25*11498SJerry.Gilliam@Sun.COM 
26*11498SJerry.Gilliam@Sun.COM #include <sys/types.h>
27*11498SJerry.Gilliam@Sun.COM #include <sys/param.h>
28*11498SJerry.Gilliam@Sun.COM #include <sys/fcntl.h>
29*11498SJerry.Gilliam@Sun.COM #include <sys/promif.h>
30*11498SJerry.Gilliam@Sun.COM #include <sys/prom_plat.h>
31*11498SJerry.Gilliam@Sun.COM #include <sys/salib.h>
32*11498SJerry.Gilliam@Sun.COM 
33*11498SJerry.Gilliam@Sun.COM extern int is_sun4v;
34*11498SJerry.Gilliam@Sun.COM 
35*11498SJerry.Gilliam@Sun.COM /*
36*11498SJerry.Gilliam@Sun.COM  * Check if the CPU should default to 64-bit or not.
37*11498SJerry.Gilliam@Sun.COM  * UltraSPARC-1's default to 32-bit mode.
38*11498SJerry.Gilliam@Sun.COM  * Everything else defaults to 64-bit mode.
39*11498SJerry.Gilliam@Sun.COM  */
40*11498SJerry.Gilliam@Sun.COM 
41*11498SJerry.Gilliam@Sun.COM /*
42*11498SJerry.Gilliam@Sun.COM  * Manufacturer codes for the CPUs we're interested in
43*11498SJerry.Gilliam@Sun.COM  */
44*11498SJerry.Gilliam@Sun.COM #define	TI_JEDEC	0x17
45*11498SJerry.Gilliam@Sun.COM #define	SUNW_JEDEC	0x22
46*11498SJerry.Gilliam@Sun.COM 
47*11498SJerry.Gilliam@Sun.COM /*
48*11498SJerry.Gilliam@Sun.COM  * Implementation codes for the CPUs we're interested in
49*11498SJerry.Gilliam@Sun.COM  */
50*11498SJerry.Gilliam@Sun.COM #define	IMPL_US_I	0x10
51*11498SJerry.Gilliam@Sun.COM 
52*11498SJerry.Gilliam@Sun.COM static pnode_t
visit(pnode_t node)53*11498SJerry.Gilliam@Sun.COM visit(pnode_t node)
54*11498SJerry.Gilliam@Sun.COM {
55*11498SJerry.Gilliam@Sun.COM 	int impl, manu;
56*11498SJerry.Gilliam@Sun.COM 	char name[32];
57*11498SJerry.Gilliam@Sun.COM 	static char ultrasparc[] = "SUNW,UltraSPARC";
58*11498SJerry.Gilliam@Sun.COM 	static char implementation[] = "implementation#";
59*11498SJerry.Gilliam@Sun.COM 	static char manufacturer[] = "manufacturer#";
60*11498SJerry.Gilliam@Sun.COM 
61*11498SJerry.Gilliam@Sun.COM 	/*
62*11498SJerry.Gilliam@Sun.COM 	 * if name isn't 'SUNW,UltraSPARC', continue.
63*11498SJerry.Gilliam@Sun.COM 	 */
64*11498SJerry.Gilliam@Sun.COM 	if (prom_getproplen(node, "name") != sizeof (ultrasparc))
65*11498SJerry.Gilliam@Sun.COM 		return ((pnode_t)0);
66*11498SJerry.Gilliam@Sun.COM 	(void) prom_getprop(node, "name", name);
67*11498SJerry.Gilliam@Sun.COM 	if (strncmp(name, ultrasparc, sizeof (ultrasparc)) != 0)
68*11498SJerry.Gilliam@Sun.COM 		return ((pnode_t)0);
69*11498SJerry.Gilliam@Sun.COM 
70*11498SJerry.Gilliam@Sun.COM 	if (prom_getproplen(node, manufacturer) != sizeof (int))
71*11498SJerry.Gilliam@Sun.COM 		return ((pnode_t)0);
72*11498SJerry.Gilliam@Sun.COM 	(void) prom_getprop(node, manufacturer, (caddr_t)&manu);
73*11498SJerry.Gilliam@Sun.COM 
74*11498SJerry.Gilliam@Sun.COM 	if ((manu != SUNW_JEDEC) && (manu != TI_JEDEC))
75*11498SJerry.Gilliam@Sun.COM 		return ((pnode_t)0);
76*11498SJerry.Gilliam@Sun.COM 
77*11498SJerry.Gilliam@Sun.COM 	if (prom_getproplen(node, implementation) != sizeof (int))
78*11498SJerry.Gilliam@Sun.COM 		return ((pnode_t)0);
79*11498SJerry.Gilliam@Sun.COM 	(void) prom_getprop(node, implementation, (caddr_t)&impl);
80*11498SJerry.Gilliam@Sun.COM 
81*11498SJerry.Gilliam@Sun.COM 	if (impl != IMPL_US_I)
82*11498SJerry.Gilliam@Sun.COM 		return ((pnode_t)0);
83*11498SJerry.Gilliam@Sun.COM 
84*11498SJerry.Gilliam@Sun.COM 	return (node);
85*11498SJerry.Gilliam@Sun.COM }
86*11498SJerry.Gilliam@Sun.COM 
87*11498SJerry.Gilliam@Sun.COM /*
88*11498SJerry.Gilliam@Sun.COM  * visit each node in the device tree, until we get a non-null answer
89*11498SJerry.Gilliam@Sun.COM  */
90*11498SJerry.Gilliam@Sun.COM static pnode_t
walk(pnode_t node)91*11498SJerry.Gilliam@Sun.COM walk(pnode_t node)
92*11498SJerry.Gilliam@Sun.COM {
93*11498SJerry.Gilliam@Sun.COM 	pnode_t id;
94*11498SJerry.Gilliam@Sun.COM 
95*11498SJerry.Gilliam@Sun.COM 	if (visit(node))
96*11498SJerry.Gilliam@Sun.COM 		return (node);
97*11498SJerry.Gilliam@Sun.COM 
98*11498SJerry.Gilliam@Sun.COM 	for (node = prom_childnode(node); node; node = prom_nextnode(node))
99*11498SJerry.Gilliam@Sun.COM 		if ((id = walk(node)) != (pnode_t)0)
100*11498SJerry.Gilliam@Sun.COM 			return (id);
101*11498SJerry.Gilliam@Sun.COM 
102*11498SJerry.Gilliam@Sun.COM 	return ((pnode_t)0);
103*11498SJerry.Gilliam@Sun.COM }
104*11498SJerry.Gilliam@Sun.COM 
105*11498SJerry.Gilliam@Sun.COM /*
106*11498SJerry.Gilliam@Sun.COM  * Check if the CPU is an UltraSPARC-1 or not.
107*11498SJerry.Gilliam@Sun.COM  */
108*11498SJerry.Gilliam@Sun.COM int
cpu_is_ultrasparc_1(void)109*11498SJerry.Gilliam@Sun.COM cpu_is_ultrasparc_1(void)
110*11498SJerry.Gilliam@Sun.COM {
111*11498SJerry.Gilliam@Sun.COM 	static int cpu_checked;
112*11498SJerry.Gilliam@Sun.COM 	static int cpu_default;
113*11498SJerry.Gilliam@Sun.COM 
114*11498SJerry.Gilliam@Sun.COM 	/*
115*11498SJerry.Gilliam@Sun.COM 	 * If we already checked or the machine is
116*11498SJerry.Gilliam@Sun.COM 	 * a sun4v, we already know the answer.
117*11498SJerry.Gilliam@Sun.COM 	 */
118*11498SJerry.Gilliam@Sun.COM 	if (!is_sun4v || cpu_checked == 0) {
119*11498SJerry.Gilliam@Sun.COM 		if (walk(prom_rootnode()))
120*11498SJerry.Gilliam@Sun.COM 			cpu_default = 1;
121*11498SJerry.Gilliam@Sun.COM 		cpu_checked = 1;
122*11498SJerry.Gilliam@Sun.COM 	}
123*11498SJerry.Gilliam@Sun.COM 
124*11498SJerry.Gilliam@Sun.COM 	return (cpu_default);
125*11498SJerry.Gilliam@Sun.COM }
126*11498SJerry.Gilliam@Sun.COM 
127*11498SJerry.Gilliam@Sun.COM /*
128*11498SJerry.Gilliam@Sun.COM  * Retain a page or reclaim a previously retained page of physical
129*11498SJerry.Gilliam@Sun.COM  * memory for use by the prom upgrade. If successful, leave
130*11498SJerry.Gilliam@Sun.COM  * an indication that a page was retained by creating a boolean
131*11498SJerry.Gilliam@Sun.COM  * property in the root node.
132*11498SJerry.Gilliam@Sun.COM  *
133*11498SJerry.Gilliam@Sun.COM  * XXX: SUNW,retain doesn't work as expected on server systems,
134*11498SJerry.Gilliam@Sun.COM  * so we don't try to retain any memory on those systems.
135*11498SJerry.Gilliam@Sun.COM  *
136*11498SJerry.Gilliam@Sun.COM  * XXX: do a '0 to my-self' as a workaround for 4160914
137*11498SJerry.Gilliam@Sun.COM  */
138*11498SJerry.Gilliam@Sun.COM 
139*11498SJerry.Gilliam@Sun.COM int dont_retain_memory;
140*11498SJerry.Gilliam@Sun.COM 
141*11498SJerry.Gilliam@Sun.COM void
retain_nvram_page(void)142*11498SJerry.Gilliam@Sun.COM retain_nvram_page(void)
143*11498SJerry.Gilliam@Sun.COM {
144*11498SJerry.Gilliam@Sun.COM 	unsigned long long phys = 0;
145*11498SJerry.Gilliam@Sun.COM 	int len;
146*11498SJerry.Gilliam@Sun.COM 	char name[32];
147*11498SJerry.Gilliam@Sun.COM 	static char create_prop[] =
148*11498SJerry.Gilliam@Sun.COM 	    "0 to my-self dev / 0 0 \" boot-retained-page\" property";
149*11498SJerry.Gilliam@Sun.COM 	static char ue10000[] = "SUNW,Ultra-Enterprise-10000";
150*11498SJerry.Gilliam@Sun.COM 	static char ue[] = "SUNW,Ultra-Enterprise";
151*11498SJerry.Gilliam@Sun.COM 	extern int verbosemode;
152*11498SJerry.Gilliam@Sun.COM 
153*11498SJerry.Gilliam@Sun.COM 	if (dont_retain_memory)
154*11498SJerry.Gilliam@Sun.COM 		return;
155*11498SJerry.Gilliam@Sun.COM 
156*11498SJerry.Gilliam@Sun.COM 	if (!is_sun4v) {
157*11498SJerry.Gilliam@Sun.COM 		len = prom_getproplen(prom_rootnode(), "name");
158*11498SJerry.Gilliam@Sun.COM 		if ((len != -1) && (len <= sizeof (name))) {
159*11498SJerry.Gilliam@Sun.COM 			(void) prom_getprop(prom_rootnode(), "name", name);
160*11498SJerry.Gilliam@Sun.COM 			if ((strcmp(name, ue) == 0) ||
161*11498SJerry.Gilliam@Sun.COM 			    (strcmp(name, ue10000) == 0))
162*11498SJerry.Gilliam@Sun.COM 				return;
163*11498SJerry.Gilliam@Sun.COM 		}
164*11498SJerry.Gilliam@Sun.COM 	}
165*11498SJerry.Gilliam@Sun.COM 
166*11498SJerry.Gilliam@Sun.COM 	if (prom_retain("OBPnvram", PAGESIZE, PAGESIZE, &phys) != 0) {
167*11498SJerry.Gilliam@Sun.COM 		printf("prom_retain failed\n");
168*11498SJerry.Gilliam@Sun.COM 		return;
169*11498SJerry.Gilliam@Sun.COM 	}
170*11498SJerry.Gilliam@Sun.COM 	if (verbosemode)
171*11498SJerry.Gilliam@Sun.COM 		printf("retained OBPnvram page at 0x%llx\n", phys);
172*11498SJerry.Gilliam@Sun.COM 
173*11498SJerry.Gilliam@Sun.COM 	prom_interpret(create_prop, 0, 0, 0, 0, 0);
174*11498SJerry.Gilliam@Sun.COM }
175