2008-12-05 07:28:22 +03:00
/* prom_common.c: OF device tree support common code.
*
* 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 sparc 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/module.h>
# include <linux/errno.h>
# include <linux/mutex.h>
# include <linux/slab.h>
# include <linux/of.h>
# include <asm/prom.h>
# include <asm/oplib.h>
# include "prom.h"
2008-12-06 02:20:26 +03:00
struct device_node * of_console_device ;
EXPORT_SYMBOL ( of_console_device ) ;
char * of_console_path ;
EXPORT_SYMBOL ( of_console_path ) ;
char * of_console_options ;
EXPORT_SYMBOL ( of_console_options ) ;
2008-12-05 07:28:22 +03:00
struct device_node * of_find_node_by_phandle ( phandle handle )
{
struct device_node * np ;
for ( np = allnodes ; np ; np = np - > allnext )
if ( np - > node = = handle )
break ;
return np ;
}
EXPORT_SYMBOL ( of_find_node_by_phandle ) ;
int of_getintprop_default ( struct device_node * np , const char * name , int def )
{
struct property * prop ;
int len ;
prop = of_find_property ( np , name , & len ) ;
if ( ! prop | | len ! = 4 )
return def ;
return * ( int * ) prop - > value ;
}
EXPORT_SYMBOL ( of_getintprop_default ) ;
DEFINE_MUTEX ( of_set_property_mutex ) ;
EXPORT_SYMBOL ( of_set_property_mutex ) ;
int of_set_property ( struct device_node * dp , const char * name , void * val , int len )
{
struct property * * prevp ;
void * new_val ;
int err ;
new_val = kmalloc ( len , GFP_KERNEL ) ;
if ( ! new_val )
return - ENOMEM ;
memcpy ( new_val , val , len ) ;
err = - ENODEV ;
write_lock ( & devtree_lock ) ;
prevp = & dp - > properties ;
while ( * prevp ) {
struct property * prop = * prevp ;
if ( ! strcasecmp ( prop - > name , name ) ) {
void * old_val = prop - > value ;
int ret ;
mutex_lock ( & of_set_property_mutex ) ;
ret = prom_setprop ( dp - > node , name , val , len ) ;
mutex_unlock ( & of_set_property_mutex ) ;
err = - EINVAL ;
if ( ret > = 0 ) {
prop - > value = new_val ;
prop - > length = len ;
if ( OF_IS_DYNAMIC ( prop ) )
kfree ( old_val ) ;
OF_MARK_DYNAMIC ( prop ) ;
err = 0 ;
}
break ;
}
prevp = & ( * prevp ) - > next ;
}
write_unlock ( & devtree_lock ) ;
/* XXX Upate procfs if necessary... */
return err ;
}
EXPORT_SYMBOL ( of_set_property ) ;
int of_find_in_proplist ( const char * list , const char * match , int len )
{
while ( len > 0 ) {
int l ;
if ( ! strcmp ( list , match ) )
return 1 ;
l = strlen ( list ) + 1 ;
list + = l ;
len - = l ;
}
return 0 ;
}
EXPORT_SYMBOL ( of_find_in_proplist ) ;
2008-12-05 11:50:22 +03:00
unsigned int prom_unique_id ;
2008-12-05 12:00:46 +03:00
static struct property * __init build_one_prop ( phandle node , char * prev ,
char * special_name ,
void * special_val ,
int special_len )
{
static struct property * tmp = NULL ;
struct property * p ;
const char * name ;
if ( tmp ) {
p = tmp ;
memset ( p , 0 , sizeof ( * p ) + 32 ) ;
tmp = NULL ;
} else {
p = prom_early_alloc ( sizeof ( struct property ) + 32 ) ;
p - > unique_id = prom_unique_id + + ;
}
p - > name = ( char * ) ( p + 1 ) ;
if ( special_name ) {
strcpy ( p - > name , special_name ) ;
p - > length = special_len ;
p - > value = prom_early_alloc ( special_len ) ;
memcpy ( p - > value , special_val , special_len ) ;
} else {
# ifdef CONFIG_SPARC32
if ( prev = = NULL ) {
name = prom_firstprop ( node , NULL ) ;
} else {
name = prom_nextprop ( node , prev , NULL ) ;
}
# else
if ( prev = = NULL ) {
prom_firstprop ( node , p - > name ) ;
} else {
prom_nextprop ( node , prev , p - > name ) ;
}
name = p - > name ;
# endif
if ( strlen ( name ) = = 0 ) {
tmp = p ;
return NULL ;
}
# ifdef CONFIG_SPARC32
strcpy ( p - > name , name ) ;
# endif
p - > length = prom_getproplen ( node , p - > name ) ;
if ( p - > length < = 0 ) {
p - > length = 0 ;
} else {
int len ;
p - > value = prom_early_alloc ( p - > length + 1 ) ;
len = prom_getproperty ( node , p - > name , p - > value ,
p - > length ) ;
if ( len < = 0 )
p - > length = 0 ;
( ( unsigned char * ) p - > value ) [ p - > length ] = ' \0 ' ;
}
}
return p ;
}
2008-12-05 12:10:18 +03:00
static struct property * __init build_prop_list ( phandle node )
2008-12-05 12:00:46 +03:00
{
struct property * head , * tail ;
head = tail = build_one_prop ( node , NULL ,
" .node " , & node , sizeof ( node ) ) ;
tail - > next = build_one_prop ( node , NULL , NULL , NULL , 0 ) ;
tail = tail - > next ;
while ( tail ) {
tail - > next = build_one_prop ( node , tail - > name ,
NULL , NULL , 0 ) ;
tail = tail - > next ;
}
return head ;
}
2008-12-05 12:10:18 +03:00
static char * __init get_one_property ( phandle node , const char * name )
{
char * buf = " <NULL> " ;
int len ;
len = prom_getproplen ( node , name ) ;
if ( len > 0 ) {
buf = prom_early_alloc ( len ) ;
len = prom_getproperty ( node , name , buf , len ) ;
}
return buf ;
}
2008-12-06 05:16:48 +03:00
static struct device_node * __init prom_create_node ( phandle node ,
struct device_node * parent )
2008-12-05 12:10:18 +03:00
{
struct device_node * dp ;
if ( ! node )
return NULL ;
dp = prom_early_alloc ( sizeof ( * dp ) ) ;
dp - > unique_id = prom_unique_id + + ;
dp - > parent = parent ;
kref_init ( & dp - > kref ) ;
dp - > name = get_one_property ( node , " name " ) ;
dp - > type = get_one_property ( node , " device_type " ) ;
dp - > node = node ;
dp - > properties = build_prop_list ( node ) ;
2008-12-07 11:46:33 +03:00
irq_trans_init ( dp ) ;
2008-12-05 12:10:18 +03:00
return dp ;
}
2008-12-05 12:21:41 +03:00
static char * __init build_full_name ( struct device_node * dp )
{
int len , ourlen , plen ;
char * n ;
plen = strlen ( dp - > parent - > full_name ) ;
ourlen = strlen ( dp - > path_component_name ) ;
len = ourlen + plen + 2 ;
n = prom_early_alloc ( len ) ;
strcpy ( n , dp - > parent - > full_name ) ;
if ( ! is_root_node ( dp - > parent ) ) {
strcpy ( n + plen , " / " ) ;
plen + + ;
}
strcpy ( n + plen , dp - > path_component_name ) ;
return n ;
}
2008-12-06 05:16:48 +03:00
static struct device_node * __init prom_build_tree ( struct device_node * parent ,
phandle node ,
struct device_node * * * nextp )
2008-12-05 12:21:41 +03:00
{
struct device_node * ret = NULL , * prev_sibling = NULL ;
struct device_node * dp ;
while ( 1 ) {
dp = prom_create_node ( node , parent ) ;
if ( ! dp )
break ;
if ( prev_sibling )
prev_sibling - > sibling = dp ;
if ( ! ret )
ret = dp ;
prev_sibling = dp ;
* ( * nextp ) = dp ;
* nextp = & dp - > allnext ;
dp - > path_component_name = build_path_component ( dp ) ;
dp - > full_name = build_full_name ( dp ) ;
dp - > child = prom_build_tree ( dp , prom_getchild ( node ) , nextp ) ;
node = prom_getsibling ( node ) ;
}
return ret ;
}
2008-12-06 05:16:48 +03:00
unsigned int prom_early_allocated __initdata ;
void __init prom_build_devicetree ( void )
{
struct device_node * * nextp ;
allnodes = prom_create_node ( prom_root_node , NULL ) ;
allnodes - > path_component_name = " " ;
allnodes - > full_name = " / " ;
nextp = & allnodes - > allnext ;
allnodes - > child = prom_build_tree ( allnodes ,
prom_getchild ( allnodes - > node ) ,
& nextp ) ;
of_console_init ( ) ;
printk ( " PROM: Built device tree with %u bytes of memory. \n " ,
prom_early_allocated ) ;
of_fill_in_cpu_data ( ) ;
}