2007-05-03 02:38:57 +10:00
/*
* Copyright ( C ) 2006 Benjamin Herrenschmidt , IBM Corp .
* < benh @ kernel . crashing . org >
* and Arnd Bergmann , IBM Corp .
* Merged from powerpc / kernel / of_platform . c and
* sparc { , 64 } / kernel / of_device . c by Stephen Rothwell
*
* 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/errno.h>
2007-10-17 21:17:42 -07:00
# include <linux/module.h>
2011-06-21 10:59:34 -06:00
# include <linux/amba/bus.h>
2007-05-03 02:38:57 +10:00
# include <linux/device.h>
2010-06-08 07:48:13 -06:00
# include <linux/dma-mapping.h>
2010-06-08 07:48:20 -06:00
# include <linux/slab.h>
2010-06-08 07:48:26 -06:00
# include <linux/of_address.h>
2007-05-03 02:38:57 +10:00
# include <linux/of_device.h>
2010-06-08 07:48:26 -06:00
# include <linux/of_irq.h>
2007-05-03 02:38:57 +10:00
# include <linux/of_platform.h>
2010-06-08 07:48:21 -06:00
# include <linux/platform_device.h>
2011-06-21 10:59:34 -06:00
const struct of_device_id of_default_bus_match_table [ ] = {
{ . compatible = " simple-bus " , } ,
# ifdef CONFIG_ARM_AMBA
{ . compatible = " arm,amba-bus " , } ,
# endif /* CONFIG_ARM_AMBA */
{ } /* Empty terminated list */
} ;
2010-07-23 19:19:35 +02:00
static int of_dev_node_match ( struct device * dev , void * data )
{
return dev - > of_node = = data ;
}
/**
* of_find_device_by_node - Find the platform_device associated with a node
* @ np : Pointer to device tree node
*
* Returns platform_device pointer , or NULL if not found
*/
struct platform_device * of_find_device_by_node ( struct device_node * np )
{
struct device * dev ;
dev = bus_find_device ( & platform_bus_type , NULL , np , of_dev_node_match ) ;
return dev ? to_platform_device ( dev ) : NULL ;
}
EXPORT_SYMBOL ( of_find_device_by_node ) ;
2010-07-14 23:51:43 -06:00
# if defined(CONFIG_PPC_DCR)
# include <asm/dcr.h>
# endif
2010-06-08 07:48:13 -06:00
# if !defined(CONFIG_SPARC)
/*
* The following routines scan a subtree and registers a device for
* each applicable node .
*
* Note : sparc doesn ' t use these routines because it has a different
* mechanism for creating devices from device tree nodes .
*/
2010-06-08 07:48:14 -06:00
/**
* of_device_make_bus_id - Use the device node data to assign a unique name
* @ dev : pointer to device structure that is linked to a device tree node
*
* This routine will first try using either the dcr - reg or the reg property
* value to derive a unique name . As a last resort it will use the node
* name followed by a unique number .
*/
2010-07-23 15:04:01 -06:00
void of_device_make_bus_id ( struct device * dev )
2010-06-08 07:48:14 -06:00
{
static atomic_t bus_no_reg_magic ;
struct device_node * node = dev - > of_node ;
const u32 * reg ;
u64 addr ;
int magic ;
# ifdef CONFIG_PPC_DCR
/*
* If it ' s a DCR based device , use ' d ' for native DCRs
* and ' D ' for MMIO DCRs .
*/
reg = of_get_property ( node , " dcr-reg " , NULL ) ;
if ( reg ) {
# ifdef CONFIG_PPC_DCR_NATIVE
dev_set_name ( dev , " d%x.%s " , * reg , node - > name ) ;
# else /* CONFIG_PPC_DCR_NATIVE */
u64 addr = of_translate_dcr_address ( node , * reg , NULL ) ;
if ( addr ! = OF_BAD_ADDR ) {
dev_set_name ( dev , " D%llx.%s " ,
( unsigned long long ) addr , node - > name ) ;
return ;
}
# endif /* !CONFIG_PPC_DCR_NATIVE */
}
# endif /* CONFIG_PPC_DCR */
/*
* For MMIO , get the physical address
*/
reg = of_get_property ( node , " reg " , NULL ) ;
if ( reg ) {
addr = of_translate_address ( node , reg ) ;
if ( addr ! = OF_BAD_ADDR ) {
dev_set_name ( dev , " %llx.%s " ,
( unsigned long long ) addr , node - > name ) ;
return ;
}
}
/*
* No BusID , use the node name and add a globally incremented
* counter ( and pray . . . )
*/
magic = atomic_add_return ( 1 , & bus_no_reg_magic ) ;
dev_set_name ( dev , " %s.%d " , node - > name , magic - 1 ) ;
}
/**
* of_device_alloc - Allocate and initialize an of_device
* @ np : device node to assign to device
* @ bus_id : Name to assign to the device . May be null to use default name .
* @ parent : Parent device .
*/
2010-07-22 13:59:23 -06:00
struct platform_device * of_device_alloc ( struct device_node * np ,
2010-06-08 07:48:14 -06:00
const char * bus_id ,
struct device * parent )
{
2010-07-22 13:59:23 -06:00
struct platform_device * dev ;
2010-10-10 21:35:05 -06:00
int rc , i , num_reg = 0 , num_irq ;
2010-06-08 07:48:15 -06:00
struct resource * res , temp_res ;
2010-10-20 11:45:13 -06:00
dev = platform_device_alloc ( " " , - 1 ) ;
if ( ! dev )
return NULL ;
/* count the io and irq resources */
2010-06-08 07:48:15 -06:00
while ( of_address_to_resource ( np , num_reg , & temp_res ) = = 0 )
num_reg + + ;
2010-10-10 21:35:05 -06:00
num_irq = of_irq_count ( np ) ;
2010-06-08 07:48:15 -06:00
/* Populate the resource table */
if ( num_irq | | num_reg ) {
2010-10-20 11:45:13 -06:00
res = kzalloc ( sizeof ( * res ) * ( num_irq + num_reg ) , GFP_KERNEL ) ;
if ( ! res ) {
platform_device_put ( dev ) ;
return NULL ;
}
2010-06-08 07:48:15 -06:00
dev - > num_resources = num_reg + num_irq ;
dev - > resource = res ;
for ( i = 0 ; i < num_reg ; i + + , res + + ) {
rc = of_address_to_resource ( np , i , res ) ;
WARN_ON ( rc ) ;
}
2010-10-10 21:35:05 -06:00
WARN_ON ( of_irq_to_resource_table ( np , res , num_irq ) ! = num_irq ) ;
2010-06-08 07:48:15 -06:00
}
2010-06-08 07:48:14 -06:00
dev - > dev . of_node = of_node_get ( np ) ;
2011-07-08 00:17:27 -05:00
# if defined(CONFIG_MICROBLAZE)
2010-06-08 07:48:14 -06:00
dev - > dev . dma_mask = & dev - > archdata . dma_mask ;
2010-06-08 07:48:26 -06:00
# endif
2010-06-08 07:48:14 -06:00
dev - > dev . parent = parent ;
if ( bus_id )
dev_set_name ( & dev - > dev , " %s " , bus_id ) ;
else
of_device_make_bus_id ( & dev - > dev ) ;
return dev ;
}
EXPORT_SYMBOL ( of_device_alloc ) ;
2010-06-08 07:48:13 -06:00
/**
2011-06-21 10:59:35 -06:00
* of_platform_device_create_pdata - Alloc , initialize and register an of_device
2010-06-08 07:48:13 -06:00
* @ np : pointer to node to create device for
* @ bus_id : name to assign device
2011-06-21 10:59:35 -06:00
* @ platform_data : pointer to populate platform_data pointer with
2010-06-08 07:48:13 -06:00
* @ parent : Linux device model parent device .
2011-01-03 15:51:11 -07:00
*
* Returns pointer to created platform device , or NULL if a device was not
* registered . Unavailable devices will not get registered .
2010-06-08 07:48:13 -06:00
*/
2011-06-21 10:59:35 -06:00
struct platform_device * of_platform_device_create_pdata (
struct device_node * np ,
const char * bus_id ,
void * platform_data ,
struct device * parent )
2010-06-08 07:48:13 -06:00
{
2010-07-22 13:59:23 -06:00
struct platform_device * dev ;
2010-06-08 07:48:13 -06:00
2011-01-03 15:51:11 -07:00
if ( ! of_device_is_available ( np ) )
return NULL ;
2010-06-08 07:48:13 -06:00
dev = of_device_alloc ( np , bus_id , parent ) ;
if ( ! dev )
return NULL ;
2011-07-08 00:17:27 -05:00
# if defined(CONFIG_MICROBLAZE)
2010-06-08 07:48:13 -06:00
dev - > archdata . dma_mask = 0xffffffffUL ;
2010-06-08 07:48:26 -06:00
# endif
2010-06-08 07:48:13 -06:00
dev - > dev . coherent_dma_mask = DMA_BIT_MASK ( 32 ) ;
2010-06-08 07:48:21 -06:00
dev - > dev . bus = & platform_bus_type ;
2011-06-21 10:59:35 -06:00
dev - > dev . platform_data = platform_data ;
2010-06-08 07:48:13 -06:00
/* We do not fill the DMA ops for platform devices by default.
* This is currently the responsibility of the platform code
* to do such , possibly using a device notifier
*/
2010-10-20 11:45:13 -06:00
if ( of_device_add ( dev ) ! = 0 ) {
platform_device_put ( dev ) ;
2010-06-08 07:48:13 -06:00
return NULL ;
}
return dev ;
}
2011-06-21 10:59:35 -06:00
/**
* of_platform_device_create - Alloc , initialize and register an of_device
* @ np : pointer to node to create device for
* @ bus_id : name to assign device
* @ parent : Linux device model parent device .
*
* Returns pointer to created platform device , or NULL if a device was not
* registered . Unavailable devices will not get registered .
*/
struct platform_device * of_platform_device_create ( struct device_node * np ,
const char * bus_id ,
struct device * parent )
{
return of_platform_device_create_pdata ( np , bus_id , NULL , parent ) ;
}
2010-06-08 07:48:13 -06:00
EXPORT_SYMBOL ( of_platform_device_create ) ;
2011-06-21 10:59:34 -06:00
# ifdef CONFIG_ARM_AMBA
static struct amba_device * of_amba_device_create ( struct device_node * node ,
const char * bus_id ,
void * platform_data ,
struct device * parent )
{
struct amba_device * dev ;
const void * prop ;
int i , ret ;
pr_debug ( " Creating amba device %s \n " , node - > full_name ) ;
if ( ! of_device_is_available ( node ) )
return NULL ;
dev = kzalloc ( sizeof ( * dev ) , GFP_KERNEL ) ;
if ( ! dev )
return NULL ;
/* setup generic device info */
dev - > dev . coherent_dma_mask = ~ 0 ;
dev - > dev . of_node = of_node_get ( node ) ;
dev - > dev . parent = parent ;
dev - > dev . platform_data = platform_data ;
if ( bus_id )
dev_set_name ( & dev - > dev , " %s " , bus_id ) ;
else
of_device_make_bus_id ( & dev - > dev ) ;
/* setup amba-specific device info */
dev - > dma_mask = ~ 0 ;
/* Allow the HW Peripheral ID to be overridden */
prop = of_get_property ( node , " arm,primecell-periphid " , NULL ) ;
if ( prop )
dev - > periphid = of_read_ulong ( prop , 1 ) ;
/* Decode the IRQs and address ranges */
for ( i = 0 ; i < AMBA_NR_IRQS ; i + + )
dev - > irq [ i ] = irq_of_parse_and_map ( node , i ) ;
ret = of_address_to_resource ( node , 0 , & dev - > res ) ;
if ( ret )
goto err_free ;
ret = amba_device_register ( dev , & iomem_resource ) ;
if ( ret )
goto err_free ;
return dev ;
err_free :
kfree ( dev ) ;
return NULL ;
}
# else /* CONFIG_ARM_AMBA */
static struct amba_device * of_amba_device_create ( struct device_node * node ,
const char * bus_id ,
void * platform_data ,
struct device * parent )
{
return NULL ;
}
# endif /* CONFIG_ARM_AMBA */
2011-06-21 10:59:35 -06:00
/**
* of_devname_lookup ( ) - Given a device node , lookup the preferred Linux name
*/
static const struct of_dev_auxdata * of_dev_lookup ( const struct of_dev_auxdata * lookup ,
struct device_node * np )
{
struct resource res ;
if ( lookup ) {
for ( ; lookup - > name ! = NULL ; lookup + + ) {
if ( ! of_device_is_compatible ( np , lookup - > compatible ) )
continue ;
if ( of_address_to_resource ( np , 0 , & res ) )
continue ;
if ( res . start ! = lookup - > phys_addr )
continue ;
pr_debug ( " %s: devname=%s \n " , np - > full_name , lookup - > name ) ;
return lookup ;
}
}
return NULL ;
}
2010-06-08 07:48:13 -06:00
/**
2011-03-18 10:21:28 -06:00
* of_platform_bus_create ( ) - Create a device for a node and its children .
2010-06-08 07:48:13 -06:00
* @ bus : device node of the bus to instantiate
2011-03-18 10:21:29 -06:00
* @ matches : match table for bus nodes
2011-03-18 10:21:28 -06:00
* disallow recursive creation of child buses
* @ parent : parent for new device , or NULL for top level .
*
* Creates a platform_device for the provided device_node , and optionally
* recursively create devices for all the child nodes .
2010-06-08 07:48:13 -06:00
*/
2011-03-18 10:21:28 -06:00
static int of_platform_bus_create ( struct device_node * bus ,
2010-06-08 07:48:13 -06:00
const struct of_device_id * matches ,
2011-06-21 10:59:35 -06:00
const struct of_dev_auxdata * lookup ,
2011-06-21 10:59:34 -06:00
struct device * parent , bool strict )
2010-06-08 07:48:13 -06:00
{
2011-06-21 10:59:35 -06:00
const struct of_dev_auxdata * auxdata ;
2010-06-08 07:48:13 -06:00
struct device_node * child ;
2010-07-22 13:59:23 -06:00
struct platform_device * dev ;
2011-06-21 10:59:35 -06:00
const char * bus_id = NULL ;
void * platform_data = NULL ;
2010-06-08 07:48:13 -06:00
int rc = 0 ;
2011-06-21 10:59:34 -06:00
/* Make sure it has a compatible property */
if ( strict & & ( ! of_get_property ( bus , " compatible " , NULL ) ) ) {
pr_debug ( " %s() - skipping %s, no compatible prop \n " ,
__func__ , bus - > full_name ) ;
return 0 ;
}
2011-06-21 10:59:35 -06:00
auxdata = of_dev_lookup ( lookup , bus ) ;
if ( auxdata ) {
bus_id = auxdata - > name ;
platform_data = auxdata - > platform_data ;
}
2011-06-21 10:59:34 -06:00
if ( of_device_is_compatible ( bus , " arm,primecell " ) ) {
2011-06-21 10:59:35 -06:00
of_amba_device_create ( bus , bus_id , platform_data , parent ) ;
2011-06-21 10:59:34 -06:00
return 0 ;
}
2011-06-21 10:59:35 -06:00
dev = of_platform_device_create_pdata ( bus , bus_id , platform_data , parent ) ;
2011-03-18 10:21:28 -06:00
if ( ! dev | | ! of_match_node ( matches , bus ) )
return 0 ;
2010-06-08 07:48:13 -06:00
for_each_child_of_node ( bus , child ) {
pr_debug ( " create child: %s \n " , child - > full_name ) ;
2011-06-21 10:59:35 -06:00
rc = of_platform_bus_create ( child , matches , lookup , & dev - > dev , strict ) ;
2010-06-08 07:48:13 -06:00
if ( rc ) {
of_node_put ( child ) ;
break ;
}
}
return rc ;
}
/**
2011-03-18 10:21:28 -06:00
* of_platform_bus_probe ( ) - Probe the device - tree for platform buses
2010-06-08 07:48:13 -06:00
* @ root : parent of the first level to probe or NULL for the root of the tree
2011-03-18 10:21:29 -06:00
* @ matches : match table for bus nodes
2010-06-08 07:48:13 -06:00
* @ parent : parent to hook devices from , NULL for toplevel
*
* Note that children of the provided root are not instantiated as devices
* unless the specified root itself matches the bus list and is not NULL .
*/
int of_platform_bus_probe ( struct device_node * root ,
const struct of_device_id * matches ,
struct device * parent )
{
struct device_node * child ;
int rc = 0 ;
2011-03-18 10:21:29 -06:00
root = root ? of_node_get ( root ) : of_find_node_by_path ( " / " ) ;
if ( ! root )
2010-06-08 07:48:25 -06:00
return - EINVAL ;
2010-06-08 07:48:13 -06:00
pr_debug ( " of_platform_bus_probe() \n " ) ;
pr_debug ( " starting at: %s \n " , root - > full_name ) ;
2011-03-18 10:21:29 -06:00
/* Do a self check of bus type, if there's a match, create children */
2010-06-08 07:48:13 -06:00
if ( of_match_node ( matches , root ) ) {
2011-06-21 10:59:35 -06:00
rc = of_platform_bus_create ( root , matches , NULL , parent , false ) ;
2011-03-18 10:21:28 -06:00
} else for_each_child_of_node ( root , child ) {
2010-06-08 07:48:13 -06:00
if ( ! of_match_node ( matches , child ) )
continue ;
2011-06-21 10:59:35 -06:00
rc = of_platform_bus_create ( child , matches , NULL , parent , false ) ;
2011-03-18 10:21:28 -06:00
if ( rc )
2010-06-08 07:48:13 -06:00
break ;
}
2011-03-18 10:21:28 -06:00
2010-06-08 07:48:13 -06:00
of_node_put ( root ) ;
return rc ;
}
EXPORT_SYMBOL ( of_platform_bus_probe ) ;
2011-06-21 10:59:34 -06:00
/**
* of_platform_populate ( ) - Populate platform_devices from device tree data
* @ root : parent of the first level to probe or NULL for the root of the tree
* @ matches : match table , NULL to use the default
* @ parent : parent to hook devices from , NULL for toplevel
*
* Similar to of_platform_bus_probe ( ) , this function walks the device tree
* and creates devices from nodes . It differs in that it follows the modern
* convention of requiring all device nodes to have a ' compatible ' property ,
* and it is suitable for creating devices which are children of the root
* node ( of_platform_bus_probe will only create children of the root which
* are selected by the @ matches argument ) .
*
* New board support should be using this function instead of
* of_platform_bus_probe ( ) .
*
* Returns 0 on success , < 0 on failure .
*/
int of_platform_populate ( struct device_node * root ,
const struct of_device_id * matches ,
2011-06-21 10:59:35 -06:00
const struct of_dev_auxdata * lookup ,
2011-06-21 10:59:34 -06:00
struct device * parent )
{
struct device_node * child ;
int rc = 0 ;
root = root ? of_node_get ( root ) : of_find_node_by_path ( " / " ) ;
if ( ! root )
return - EINVAL ;
for_each_child_of_node ( root , child ) {
2011-06-21 10:59:35 -06:00
rc = of_platform_bus_create ( child , matches , lookup , parent , true ) ;
2011-06-21 10:59:34 -06:00
if ( rc )
break ;
}
of_node_put ( root ) ;
return rc ;
}
2010-06-08 07:48:13 -06:00
# endif /* !CONFIG_SPARC */