2005-04-16 15:20:36 -07:00
/*
* proc_devtree . c - handles / proc / device - tree
*
* Copyright 1997 Paul Mackerras
*/
# include <linux/errno.h>
# include <linux/time.h>
# include <linux/proc_fs.h>
# include <linux/stat.h>
# include <linux/string.h>
# include <asm/prom.h>
# include <asm/uaccess.h>
# ifndef HAVE_ARCH_DEVTREE_FIXUPS
2005-06-01 17:07:27 +10:00
static inline void set_node_proc_entry ( struct device_node * np ,
struct proc_dir_entry * de )
2005-04-16 15:20:36 -07:00
{
}
# endif
static struct proc_dir_entry * proc_device_tree ;
/*
* Supply data on a read from / proc / device - tree / node / property .
*/
static int property_read_proc ( char * page , char * * start , off_t off ,
int count , int * eof , void * data )
{
struct property * pp = data ;
int n ;
if ( off > = pp - > length ) {
* eof = 1 ;
return 0 ;
}
n = pp - > length - off ;
if ( n > count )
n = count ;
else
* eof = 1 ;
2007-04-03 10:58:52 +10:00
memcpy ( page , ( char * ) pp - > value + off , n ) ;
2005-04-16 15:20:36 -07:00
* start = page ;
return n ;
}
/*
* For a node with a name like " gc@10 " , we make symlinks called " gc "
* and " @10 " to it .
*/
2005-11-07 14:29:02 +11:00
/*
* Add a property to a node
*/
static struct proc_dir_entry *
2006-03-27 14:26:26 +11:00
__proc_device_tree_add_prop ( struct proc_dir_entry * de , struct property * pp ,
const char * name )
2005-11-07 14:29:02 +11:00
{
struct proc_dir_entry * ent ;
/*
* Unfortunately proc_register puts each new entry
* at the beginning of the list . So we rearrange them .
*/
2006-03-27 14:26:26 +11:00
ent = create_proc_read_entry ( name ,
strncmp ( name , " security- " , 9 )
2005-11-07 14:29:02 +11:00
? S_IRUGO : S_IRUSR , de ,
property_read_proc , pp ) ;
if ( ent = = NULL )
return NULL ;
2006-03-27 14:26:26 +11:00
if ( ! strncmp ( name , " security- " , 9 ) )
2005-11-07 14:29:02 +11:00
ent - > size = 0 ; /* don't leak number of password chars */
else
ent - > size = pp - > length ;
return ent ;
}
void proc_device_tree_add_prop ( struct proc_dir_entry * pde , struct property * prop )
{
2006-03-27 14:26:26 +11:00
__proc_device_tree_add_prop ( pde , prop , prop - > name ) ;
2005-11-07 14:29:02 +11:00
}
2006-01-12 16:07:17 -06:00
void proc_device_tree_remove_prop ( struct proc_dir_entry * pde ,
struct property * prop )
{
remove_proc_entry ( prop - > name , pde ) ;
}
void proc_device_tree_update_prop ( struct proc_dir_entry * pde ,
struct property * newprop ,
struct property * oldprop )
{
struct proc_dir_entry * ent ;
for ( ent = pde - > subdir ; ent ! = NULL ; ent = ent - > next )
if ( ent - > data = = oldprop )
break ;
if ( ent = = NULL ) {
printk ( KERN_WARNING " device-tree: property \" %s \" "
" does not exist \n " , oldprop - > name ) ;
} else {
ent - > data = newprop ;
ent - > size = newprop - > length ;
}
}
2006-03-27 14:26:26 +11:00
/*
* Various dodgy firmware might give us nodes and / or properties with
* conflicting names . That ' s generally ok , except for exporting via / proc ,
* so munge names here to ensure they ' re unique .
*/
static int duplicate_name ( struct proc_dir_entry * de , const char * name )
{
struct proc_dir_entry * ent ;
int found = 0 ;
spin_lock ( & proc_subdir_lock ) ;
for ( ent = de - > subdir ; ent ! = NULL ; ent = ent - > next ) {
if ( strcmp ( ent - > name , name ) = = 0 ) {
found = 1 ;
break ;
}
}
spin_unlock ( & proc_subdir_lock ) ;
return found ;
}
static const char * fixup_name ( struct device_node * np , struct proc_dir_entry * de ,
const char * name )
{
char * fixed_name ;
int fixup_len = strlen ( name ) + 2 + 1 ; /* name + #x + \0 */
int i = 1 , size ;
realloc :
fixed_name = kmalloc ( fixup_len , GFP_KERNEL ) ;
if ( fixed_name = = NULL ) {
printk ( KERN_ERR " device-tree: Out of memory trying to fixup "
" name \" %s \" \n " , name ) ;
return name ;
}
retry :
size = snprintf ( fixed_name , fixup_len , " %s#%d " , name , i ) ;
size + + ; /* account for NULL */
if ( size > fixup_len ) {
/* We ran out of space, free and reallocate. */
kfree ( fixed_name ) ;
fixup_len = size ;
goto realloc ;
}
if ( duplicate_name ( de , fixed_name ) ) {
/* Multiple duplicates. Retry with a different offset. */
i + + ;
goto retry ;
}
printk ( KERN_WARNING " device-tree: Duplicate name in %s, "
" renamed to \" %s \" \n " , np - > full_name , fixed_name ) ;
return fixed_name ;
}
2005-04-16 15:20:36 -07:00
/*
* Process a node , adding entries for its children and its properties .
*/
2005-06-01 17:07:27 +10:00
void proc_device_tree_add_node ( struct device_node * np ,
struct proc_dir_entry * de )
2005-04-16 15:20:36 -07:00
{
struct property * pp ;
struct proc_dir_entry * ent ;
2005-06-01 17:07:27 +10:00
struct device_node * child ;
const char * p ;
2005-04-16 15:20:36 -07:00
set_node_proc_entry ( np , de ) ;
2005-06-01 17:07:27 +10:00
for ( child = NULL ; ( child = of_get_next_child ( np , child ) ) ; ) {
2006-03-27 14:26:26 +11:00
/* Use everything after the last slash, or the full name */
2005-04-16 15:20:36 -07:00
p = strrchr ( child - > full_name , ' / ' ) ;
if ( ! p )
p = child - > full_name ;
else
+ + p ;
2006-03-27 14:26:26 +11:00
if ( duplicate_name ( de , p ) )
p = fixup_name ( np , de , p ) ;
2005-04-16 15:20:36 -07:00
ent = proc_mkdir ( p , de ) ;
if ( ent = = 0 )
break ;
proc_device_tree_add_node ( child , ent ) ;
2005-06-01 17:07:27 +10:00
}
of_node_put ( child ) ;
2006-03-27 14:26:26 +11:00
2005-06-01 17:07:27 +10:00
for ( pp = np - > properties ; pp ! = 0 ; pp = pp - > next ) {
2006-03-27 14:26:26 +11:00
p = pp - > name ;
if ( duplicate_name ( de , p ) )
p = fixup_name ( np , de , p ) ;
2005-06-01 17:07:27 +10:00
2006-03-27 14:26:26 +11:00
ent = __proc_device_tree_add_prop ( de , pp , p ) ;
2005-06-01 17:07:27 +10:00
if ( ent = = 0 )
2005-04-16 15:20:36 -07:00
break ;
}
}
/*
* Called on initialization to set up the / proc / device - tree subtree
*/
void proc_device_tree_init ( void )
{
struct device_node * root ;
if ( ! have_of )
return ;
proc_device_tree = proc_mkdir ( " device-tree " , NULL ) ;
if ( proc_device_tree = = 0 )
return ;
root = of_find_node_by_path ( " / " ) ;
if ( root = = 0 ) {
printk ( KERN_ERR " /proc/device-tree: can't find root \n " ) ;
return ;
}
proc_device_tree_add_node ( root , proc_device_tree ) ;
of_node_put ( root ) ;
}