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