2006-06-21 15:35:28 -07:00
/*
* Procedures for creating , accessing and interpreting the device tree .
*
* Paul Mackerras August 1996.
* Copyright ( C ) 1996 - 2005 Paul Mackerras .
*
* Adapted for 64 bit PowerPC by Dave Engebretsen and Peter Bergner .
* { engebret | bergner } @ us . ibm . com
*
* Adapted for sparc64 by David S . Miller davem @ davemloft . net
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*/
# include <linux/kernel.h>
# include <linux/types.h>
# include <linux/string.h>
# include <linux/mm.h>
2006-06-22 16:18:54 -07:00
# include <linux/module.h>
2008-02-13 19:21:51 -08:00
# include <linux/lmb.h>
2008-08-07 15:33:36 -07:00
# include <linux/of_device.h>
2006-06-21 15:35:28 -07:00
# include <asm/prom.h>
# include <asm/oplib.h>
2006-06-29 15:07:37 -07:00
# include <asm/irq.h>
# include <asm/asi.h>
# include <asm/upa.h>
2007-05-25 15:49:59 -07:00
# include <asm/smp.h>
2006-06-21 15:35:28 -07:00
2008-12-04 20:12:20 -08:00
# include "prom.h"
2006-06-25 23:18:36 -07:00
2008-12-05 00:40:43 -08:00
void * __init prom_early_alloc ( unsigned long size )
2006-06-21 15:35:28 -07:00
{
2008-02-13 19:21:51 -08:00
unsigned long paddr = lmb_alloc ( size , SMP_CACHE_BYTES ) ;
2006-06-21 15:35:28 -07:00
void * ret ;
2008-02-13 19:21:51 -08:00
if ( ! paddr ) {
prom_printf ( " prom_early_alloc(%lu) failed \n " ) ;
prom_halt ( ) ;
}
2006-06-21 15:35:28 -07:00
2008-02-13 19:21:51 -08:00
ret = __va ( paddr ) ;
memset ( ret , 0 , size ) ;
2006-06-21 15:35:28 -07:00
prom_early_allocated + = size ;
return ret ;
}
/* The following routines deal with the black magic of fully naming a
* node .
*
* Certain well known named nodes are just the simple name string .
*
* Actual devices have an address specifier appended to the base name
* string , like this " foo@addr " . The " addr " can be in any number of
* formats , and the platform plus the type of the node determine the
* format and how it is constructed .
*
* For children of the ROOT node , the naming convention is fixed and
* determined by whether this is a sun4u or sun4v system .
*
* For children of other nodes , it is bus type specific . So
* we walk up the tree until we discover a " device_type " property
* we recognize and we go from there .
*
* As an example , the boot device on my workstation has a full path :
*
* / pci @ 1 e , 600000 / ide @ d / disk @ 0 , 0 : c
*/
static void __init sun4v_path_component ( struct device_node * dp , char * tmp_buf )
{
struct linux_prom64_registers * regs ;
struct property * rprop ;
u32 high_bits , low_bits , type ;
rprop = of_find_property ( dp , " reg " , NULL ) ;
if ( ! rprop )
return ;
regs = rprop - > value ;
if ( ! is_root_node ( dp - > parent ) ) {
sprintf ( tmp_buf , " %s@%x,%x " ,
dp - > name ,
( unsigned int ) ( regs - > phys_addr > > 32UL ) ,
( unsigned int ) ( regs - > phys_addr & 0xffffffffUL ) ) ;
return ;
}
type = regs - > phys_addr > > 60UL ;
high_bits = ( regs - > phys_addr > > 32UL ) & 0x0fffffffUL ;
low_bits = ( regs - > phys_addr & 0xffffffffUL ) ;
if ( type = = 0 | | type = = 8 ) {
const char * prefix = ( type = = 0 ) ? " m " : " i " ;
if ( low_bits )
sprintf ( tmp_buf , " %s@%s%x,%x " ,
dp - > name , prefix ,
high_bits , low_bits ) ;
else
sprintf ( tmp_buf , " %s@%s%x " ,
dp - > name ,
prefix ,
high_bits ) ;
} else if ( type = = 12 ) {
sprintf ( tmp_buf , " %s@%x " ,
dp - > name , high_bits ) ;
}
}
static void __init sun4u_path_component ( struct device_node * dp , char * tmp_buf )
{
struct linux_prom64_registers * regs ;
struct property * prop ;
prop = of_find_property ( dp , " reg " , NULL ) ;
if ( ! prop )
return ;
regs = prop - > value ;
if ( ! is_root_node ( dp - > parent ) ) {
sprintf ( tmp_buf , " %s@%x,%x " ,
dp - > name ,
( unsigned int ) ( regs - > phys_addr > > 32UL ) ,
( unsigned int ) ( regs - > phys_addr & 0xffffffffUL ) ) ;
return ;
}
prop = of_find_property ( dp , " upa-portid " , NULL ) ;
if ( ! prop )
prop = of_find_property ( dp , " portid " , NULL ) ;
if ( prop ) {
unsigned long mask = 0xffffffffUL ;
if ( tlb_type > = cheetah )
mask = 0x7fffff ;
sprintf ( tmp_buf , " %s@%x,%x " ,
dp - > name ,
* ( u32 * ) prop - > value ,
( unsigned int ) ( regs - > phys_addr & mask ) ) ;
}
}
/* "name@slot,offset" */
static void __init sbus_path_component ( struct device_node * dp , char * tmp_buf )
{
struct linux_prom_registers * regs ;
struct property * prop ;
prop = of_find_property ( dp , " reg " , NULL ) ;
if ( ! prop )
return ;
regs = prop - > value ;
sprintf ( tmp_buf , " %s@%x,%x " ,
dp - > name ,
regs - > which_io ,
regs - > phys_addr ) ;
}
/* "name@devnum[,func]" */
static void __init pci_path_component ( struct device_node * dp , char * tmp_buf )
{
struct linux_prom_pci_registers * regs ;
struct property * prop ;
unsigned int devfn ;
prop = of_find_property ( dp , " reg " , NULL ) ;
if ( ! prop )
return ;
regs = prop - > value ;
devfn = ( regs - > phys_hi > > 8 ) & 0xff ;
if ( devfn & 0x07 ) {
sprintf ( tmp_buf , " %s@%x,%x " ,
dp - > name ,
devfn > > 3 ,
devfn & 0x07 ) ;
} else {
sprintf ( tmp_buf , " %s@%x " ,
dp - > name ,
devfn > > 3 ) ;
}
}
/* "name@UPA_PORTID,offset" */
static void __init upa_path_component ( struct device_node * dp , char * tmp_buf )
{
struct linux_prom64_registers * regs ;
struct property * prop ;
prop = of_find_property ( dp , " reg " , NULL ) ;
if ( ! prop )
return ;
regs = prop - > value ;
prop = of_find_property ( dp , " upa-portid " , NULL ) ;
if ( ! prop )
return ;
sprintf ( tmp_buf , " %s@%x,%x " ,
dp - > name ,
* ( u32 * ) prop - > value ,
( unsigned int ) ( regs - > phys_addr & 0xffffffffUL ) ) ;
}
/* "name@reg" */
static void __init vdev_path_component ( struct device_node * dp , char * tmp_buf )
{
struct property * prop ;
u32 * regs ;
prop = of_find_property ( dp , " reg " , NULL ) ;
if ( ! prop )
return ;
regs = prop - > value ;
sprintf ( tmp_buf , " %s@%x " , dp - > name , * regs ) ;
}
/* "name@addrhi,addrlo" */
static void __init ebus_path_component ( struct device_node * dp , char * tmp_buf )
{
struct linux_prom64_registers * regs ;
struct property * prop ;
prop = of_find_property ( dp , " reg " , NULL ) ;
if ( ! prop )
return ;
regs = prop - > value ;
sprintf ( tmp_buf , " %s@%x,%x " ,
dp - > name ,
( unsigned int ) ( regs - > phys_addr > > 32UL ) ,
( unsigned int ) ( regs - > phys_addr & 0xffffffffUL ) ) ;
}
/* "name@bus,addr" */
static void __init i2c_path_component ( struct device_node * dp , char * tmp_buf )
{
struct property * prop ;
u32 * regs ;
prop = of_find_property ( dp , " reg " , NULL ) ;
if ( ! prop )
return ;
regs = prop - > value ;
/* This actually isn't right... should look at the #address-cells
* property of the i2c bus node etc . etc .
*/
sprintf ( tmp_buf , " %s@%x,%x " ,
dp - > name , regs [ 0 ] , regs [ 1 ] ) ;
}
/* "name@reg0[,reg1]" */
static void __init usb_path_component ( struct device_node * dp , char * tmp_buf )
{
struct property * prop ;
u32 * regs ;
prop = of_find_property ( dp , " reg " , NULL ) ;
if ( ! prop )
return ;
regs = prop - > value ;
if ( prop - > length = = sizeof ( u32 ) | | regs [ 1 ] = = 1 ) {
sprintf ( tmp_buf , " %s@%x " ,
dp - > name , regs [ 0 ] ) ;
} else {
sprintf ( tmp_buf , " %s@%x,%x " ,
dp - > name , regs [ 0 ] , regs [ 1 ] ) ;
}
}
/* "name@reg0reg1[,reg2reg3]" */
static void __init ieee1394_path_component ( struct device_node * dp , char * tmp_buf )
{
struct property * prop ;
u32 * regs ;
prop = of_find_property ( dp , " reg " , NULL ) ;
if ( ! prop )
return ;
regs = prop - > value ;
if ( regs [ 2 ] | | regs [ 3 ] ) {
sprintf ( tmp_buf , " %s@%08x%08x,%04x%08x " ,
dp - > name , regs [ 0 ] , regs [ 1 ] , regs [ 2 ] , regs [ 3 ] ) ;
} else {
sprintf ( tmp_buf , " %s@%08x%08x " ,
dp - > name , regs [ 0 ] , regs [ 1 ] ) ;
}
}
static void __init __build_path_component ( struct device_node * dp , char * tmp_buf )
{
struct device_node * parent = dp - > parent ;
if ( parent ! = NULL ) {
if ( ! strcmp ( parent - > type , " pci " ) | |
2008-09-11 23:52:35 -07:00
! strcmp ( parent - > type , " pciex " ) ) {
pci_path_component ( dp , tmp_buf ) ;
return ;
}
if ( ! strcmp ( parent - > type , " sbus " ) ) {
sbus_path_component ( dp , tmp_buf ) ;
return ;
}
if ( ! strcmp ( parent - > type , " upa " ) ) {
upa_path_component ( dp , tmp_buf ) ;
return ;
}
if ( ! strcmp ( parent - > type , " ebus " ) ) {
ebus_path_component ( dp , tmp_buf ) ;
return ;
}
2006-06-21 15:35:28 -07:00
if ( ! strcmp ( parent - > name , " usb " ) | |
2008-09-11 23:52:35 -07:00
! strcmp ( parent - > name , " hub " ) ) {
usb_path_component ( dp , tmp_buf ) ;
return ;
}
if ( ! strcmp ( parent - > type , " i2c " ) ) {
i2c_path_component ( dp , tmp_buf ) ;
return ;
}
if ( ! strcmp ( parent - > type , " firewire " ) ) {
ieee1394_path_component ( dp , tmp_buf ) ;
return ;
}
if ( ! strcmp ( parent - > type , " virtual-devices " ) ) {
vdev_path_component ( dp , tmp_buf ) ;
return ;
}
2006-06-21 15:35:28 -07:00
/* "isa" is handled with platform naming */
}
/* Use platform naming convention. */
2008-09-11 23:52:35 -07:00
if ( tlb_type = = hypervisor ) {
sun4v_path_component ( dp , tmp_buf ) ;
return ;
} else {
sun4u_path_component ( dp , tmp_buf ) ;
}
2006-06-21 15:35:28 -07:00
}
2008-12-05 01:21:41 -08:00
char * __init build_path_component ( struct device_node * dp )
2006-06-21 15:35:28 -07:00
{
char tmp_buf [ 64 ] , * n ;
tmp_buf [ 0 ] = ' \0 ' ;
__build_path_component ( dp , tmp_buf ) ;
if ( tmp_buf [ 0 ] = = ' \0 ' )
strcpy ( tmp_buf , dp - > name ) ;
n = prom_early_alloc ( strlen ( tmp_buf ) + 1 ) ;
strcpy ( n , tmp_buf ) ;
return n ;
}
2007-05-25 15:49:59 -07:00
static const char * get_mid_prop ( void )
{
return ( tlb_type = = spitfire ? " upa-portid " : " portid " ) ;
}
2009-05-26 21:46:14 -07:00
static void * of_iterate_over_cpus ( void * ( * func ) ( struct device_node * , int , int ) , int arg )
2007-05-25 15:49:59 -07:00
{
struct device_node * dp ;
2008-12-05 18:16:48 -08:00
const char * mid_prop ;
mid_prop = get_mid_prop ( ) ;
2007-05-25 15:49:59 -07:00
for_each_node_by_type ( dp , " cpu " ) {
int cpuid = of_getintprop_default ( dp , mid_prop , - 1 ) ;
const char * this_mid_prop = mid_prop ;
2009-05-26 21:46:14 -07:00
void * ret ;
2007-05-25 15:49:59 -07:00
if ( cpuid < 0 ) {
this_mid_prop = " cpuid " ;
cpuid = of_getintprop_default ( dp , this_mid_prop , - 1 ) ;
}
if ( cpuid < 0 ) {
prom_printf ( " OF: Serious problem, cpu lacks "
" %s property " , this_mid_prop ) ;
prom_halt ( ) ;
}
# ifdef CONFIG_SMP
2007-09-16 14:45:06 -07:00
if ( cpuid > = NR_CPUS ) {
printk ( KERN_WARNING " Ignoring CPU %d which is "
" >= NR_CPUS (%d) \n " ,
cpuid , NR_CPUS ) ;
2007-05-25 15:49:59 -07:00
continue ;
2007-09-16 14:45:06 -07:00
}
2007-05-25 15:49:59 -07:00
# endif
2009-05-26 21:46:14 -07:00
ret = func ( dp , cpuid , arg ) ;
if ( ret )
return ret ;
}
return NULL ;
}
2007-05-25 15:49:59 -07:00
2009-05-26 21:46:14 -07:00
static void * check_cpu_node ( struct device_node * dp , int cpuid , int id )
{
if ( id = = cpuid )
return dp ;
return NULL ;
}
struct device_node * of_find_node_by_cpuid ( int cpuid )
{
return of_iterate_over_cpus ( check_cpu_node , cpuid ) ;
}
static void * record_one_cpu ( struct device_node * dp , int cpuid , int arg )
{
ncpus_probed + + ;
2007-06-04 21:48:33 -07:00
# ifdef CONFIG_SMP
2009-05-26 21:46:14 -07:00
set_cpu_present ( cpuid , true ) ;
set_cpu_possible ( cpuid , true ) ;
2007-06-04 21:48:33 -07:00
# endif
2009-05-26 21:46:14 -07:00
return NULL ;
}
2007-05-25 15:49:59 -07:00
2009-05-26 21:46:14 -07:00
void __init of_populate_present_mask ( void )
{
if ( tlb_type = = hypervisor )
return ;
ncpus_probed = 0 ;
of_iterate_over_cpus ( record_one_cpu , 0 ) ;
}
2007-05-25 15:49:59 -07:00
2009-05-26 21:46:14 -07:00
static void * fill_in_one_cpu ( struct device_node * dp , int cpuid , int arg )
{
struct device_node * portid_parent = NULL ;
int portid = - 1 ;
if ( of_find_property ( dp , " cpuid " , NULL ) ) {
int limit = 2 ;
portid_parent = dp ;
while ( limit - - ) {
portid_parent = portid_parent - > parent ;
if ( ! portid_parent )
break ;
portid = of_getintprop_default ( portid_parent ,
" portid " , - 1 ) ;
if ( portid > = 0 )
break ;
}
}
# ifndef CONFIG_SMP
/* On uniprocessor we only want the values for the
* real physical cpu the kernel booted onto , however
* cpu_data ( ) only has one entry at index 0.
*/
if ( cpuid ! = real_hard_smp_processor_id ( ) )
return NULL ;
cpuid = 0 ;
# endif
cpu_data ( cpuid ) . clock_tick =
of_getintprop_default ( dp , " clock-frequency " , 0 ) ;
if ( portid_parent ) {
cpu_data ( cpuid ) . dcache_size =
of_getintprop_default ( dp , " l1-dcache-size " ,
16 * 1024 ) ;
cpu_data ( cpuid ) . dcache_line_size =
of_getintprop_default ( dp , " l1-dcache-line-size " ,
32 ) ;
cpu_data ( cpuid ) . icache_size =
of_getintprop_default ( dp , " l1-icache-size " ,
8 * 1024 ) ;
cpu_data ( cpuid ) . icache_line_size =
of_getintprop_default ( dp , " l1-icache-line-size " ,
32 ) ;
cpu_data ( cpuid ) . ecache_size =
of_getintprop_default ( dp , " l2-cache-size " , 0 ) ;
cpu_data ( cpuid ) . ecache_line_size =
of_getintprop_default ( dp , " l2-cache-line-size " , 0 ) ;
if ( ! cpu_data ( cpuid ) . ecache_size | |
! cpu_data ( cpuid ) . ecache_line_size ) {
2007-05-25 15:49:59 -07:00
cpu_data ( cpuid ) . ecache_size =
2009-05-26 21:46:14 -07:00
of_getintprop_default ( portid_parent ,
" l2-cache-size " ,
2007-05-25 15:49:59 -07:00
( 4 * 1024 * 1024 ) ) ;
cpu_data ( cpuid ) . ecache_line_size =
2009-05-26 21:46:14 -07:00
of_getintprop_default ( portid_parent ,
" l2-cache-line-size " , 64 ) ;
2007-05-25 15:49:59 -07:00
}
2009-05-26 21:46:14 -07:00
cpu_data ( cpuid ) . core_id = portid + 1 ;
cpu_data ( cpuid ) . proc_id = portid ;
2007-05-25 15:49:59 -07:00
# ifdef CONFIG_SMP
2009-05-26 21:46:14 -07:00
sparc64_multi_core = 1 ;
2007-05-25 15:49:59 -07:00
# endif
2009-05-26 21:46:14 -07:00
} else {
cpu_data ( cpuid ) . dcache_size =
of_getintprop_default ( dp , " dcache-size " , 16 * 1024 ) ;
cpu_data ( cpuid ) . dcache_line_size =
of_getintprop_default ( dp , " dcache-line-size " , 32 ) ;
cpu_data ( cpuid ) . icache_size =
of_getintprop_default ( dp , " icache-size " , 16 * 1024 ) ;
cpu_data ( cpuid ) . icache_line_size =
of_getintprop_default ( dp , " icache-line-size " , 32 ) ;
cpu_data ( cpuid ) . ecache_size =
of_getintprop_default ( dp , " ecache-size " ,
( 4 * 1024 * 1024 ) ) ;
cpu_data ( cpuid ) . ecache_line_size =
of_getintprop_default ( dp , " ecache-line-size " , 64 ) ;
cpu_data ( cpuid ) . core_id = 0 ;
cpu_data ( cpuid ) . proc_id = - 1 ;
2007-05-25 15:49:59 -07:00
}
2009-05-26 21:46:14 -07:00
return NULL ;
}
void __init of_fill_in_cpu_data ( void )
{
if ( tlb_type = = hypervisor )
return ;
of_iterate_over_cpus ( fill_in_one_cpu , 0 ) ;
2007-05-25 15:49:59 -07:00
smp_fill_in_sib_core_maps ( ) ;
}
2008-12-05 18:16:48 -08:00
void __init of_console_init ( void )
2007-07-20 16:59:26 -07:00
{
char * msg = " OF stdout device is: %s \n " ;
struct device_node * dp ;
const char * type ;
phandle node ;
of_console_path = prom_early_alloc ( 256 ) ;
if ( prom_ihandle2path ( prom_stdout , of_console_path , 256 ) < 0 ) {
prom_printf ( " Cannot obtain path of stdout. \n " ) ;
prom_halt ( ) ;
}
of_console_options = strrchr ( of_console_path , ' : ' ) ;
if ( of_console_options ) {
of_console_options + + ;
if ( * of_console_options = = ' \0 ' )
of_console_options = NULL ;
}
node = prom_inst2pkg ( prom_stdout ) ;
if ( ! node ) {
prom_printf ( " Cannot resolve stdout node from "
" instance %08x. \n " , prom_stdout ) ;
prom_halt ( ) ;
}
dp = of_find_node_by_phandle ( node ) ;
type = of_get_property ( dp , " device_type " , NULL ) ;
if ( ! type ) {
prom_printf ( " Console stdout lacks device_type property. \n " ) ;
prom_halt ( ) ;
}
if ( strcmp ( type , " display " ) & & strcmp ( type , " serial " ) ) {
prom_printf ( " Console device_type is neither display "
" nor serial. \n " ) ;
prom_halt ( ) ;
}
of_console_device = dp ;
printk ( msg , of_console_path ) ;
}