2007-05-01 16:40:36 +10:00
# include <linux/string.h>
# include <linux/kernel.h>
# include <linux/of.h>
2007-05-01 16:49:51 +10:00
# include <linux/of_device.h>
2007-05-01 16:40:36 +10:00
# include <linux/init.h>
# include <linux/module.h>
# include <linux/mod_devicetable.h>
# include <linux/slab.h>
# include <asm/errno.h>
/**
2010-04-13 16:13:22 -07:00
* of_match_device - Tell if a struct device matches an of_device_id list
2007-05-01 16:40:36 +10:00
* @ ids : array of of device match structures to search in
* @ dev : the of device structure to match against
*
2010-08-06 09:25:50 -06:00
* Used by a driver to check whether an platform_device present in the
2007-05-01 16:40:36 +10:00
* system is in its list of supported devices .
*/
const struct of_device_id * of_match_device ( const struct of_device_id * matches ,
2010-04-13 16:13:22 -07:00
const struct device * dev )
2007-05-01 16:40:36 +10:00
{
2010-06-08 07:48:17 -06:00
if ( ( ! matches ) | | ( ! dev - > of_node ) )
2007-05-01 16:40:36 +10:00
return NULL ;
2010-04-13 16:13:22 -07:00
return of_match_node ( matches , dev - > of_node ) ;
2007-05-01 16:40:36 +10:00
}
EXPORT_SYMBOL ( of_match_device ) ;
2010-07-22 13:59:23 -06:00
struct platform_device * of_dev_get ( struct platform_device * dev )
2007-05-01 16:40:36 +10:00
{
struct device * tmp ;
if ( ! dev )
return NULL ;
tmp = get_device ( & dev - > dev ) ;
if ( tmp )
2010-07-22 13:59:23 -06:00
return to_platform_device ( tmp ) ;
2007-05-01 16:40:36 +10:00
else
return NULL ;
}
EXPORT_SYMBOL ( of_dev_get ) ;
2010-07-22 13:59:23 -06:00
void of_dev_put ( struct platform_device * dev )
2007-05-01 16:40:36 +10:00
{
if ( dev )
put_device ( & dev - > dev ) ;
}
EXPORT_SYMBOL ( of_dev_put ) ;
2008-04-24 23:16:00 +10:00
static ssize_t devspec_show ( struct device * dev ,
2007-05-01 16:40:36 +10:00
struct device_attribute * attr , char * buf )
{
2010-07-22 13:59:23 -06:00
struct platform_device * ofdev ;
2007-05-01 16:40:36 +10:00
2010-07-22 13:59:23 -06:00
ofdev = to_platform_device ( dev ) ;
2010-04-13 16:12:29 -07:00
return sprintf ( buf , " %s \n " , ofdev - > dev . of_node - > full_name ) ;
2007-05-01 16:40:36 +10:00
}
2008-08-06 00:30:18 +10:00
static ssize_t name_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
2010-07-22 13:59:23 -06:00
struct platform_device * ofdev ;
2008-08-06 00:30:18 +10:00
2010-07-22 13:59:23 -06:00
ofdev = to_platform_device ( dev ) ;
2010-04-13 16:12:29 -07:00
return sprintf ( buf , " %s \n " , ofdev - > dev . of_node - > name ) ;
2008-08-06 00:30:18 +10:00
}
2008-04-24 23:16:00 +10:00
static ssize_t modalias_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
2010-06-08 07:48:13 -06:00
ssize_t len = of_device_get_modalias ( dev , buf , PAGE_SIZE - 2 ) ;
2008-04-24 23:16:00 +10:00
buf [ len ] = ' \n ' ;
buf [ len + 1 ] = 0 ;
return len + 1 ;
}
struct device_attribute of_platform_device_attrs [ ] = {
__ATTR_RO ( devspec ) ,
2008-08-06 00:30:18 +10:00
__ATTR_RO ( name ) ,
2008-04-24 23:16:00 +10:00
__ATTR_RO ( modalias ) ,
__ATTR_NULL
} ;
2007-05-01 16:40:36 +10:00
/**
* of_release_dev - free an of device structure when all users of it are finished .
* @ dev : device that ' s been disconnected
*
* Will be called only by the device core when all users of this of device are
* done .
*/
void of_release_dev ( struct device * dev )
{
2010-07-22 13:59:23 -06:00
struct platform_device * ofdev ;
2007-05-01 16:40:36 +10:00
2010-07-22 13:59:23 -06:00
ofdev = to_platform_device ( dev ) ;
2010-04-13 16:12:29 -07:00
of_node_put ( ofdev - > dev . of_node ) ;
2007-05-01 16:40:36 +10:00
kfree ( ofdev ) ;
}
EXPORT_SYMBOL ( of_release_dev ) ;
2010-07-22 13:59:23 -06:00
int of_device_register ( struct platform_device * ofdev )
2007-05-01 16:40:36 +10:00
{
2010-04-13 16:12:29 -07:00
BUG_ON ( ofdev - > dev . of_node = = NULL ) ;
2008-10-26 21:51:25 +00:00
device_initialize ( & ofdev - > dev ) ;
2010-06-08 07:48:21 -06:00
/* name and id have to be set so that the platform bus doesn't get
* confused on matching */
ofdev - > name = dev_name ( & ofdev - > dev ) ;
ofdev - > id = - 1 ;
2008-10-26 21:51:25 +00:00
/* device_add will assume that this device is on the same node as
* the parent . If there is no parent defined , set the node
* explicitly */
if ( ! ofdev - > dev . parent )
2010-04-13 16:12:29 -07:00
set_dev_node ( & ofdev - > dev , of_node_to_nid ( ofdev - > dev . of_node ) ) ;
2008-10-26 21:51:25 +00:00
return device_add ( & ofdev - > dev ) ;
2007-05-01 16:40:36 +10:00
}
EXPORT_SYMBOL ( of_device_register ) ;
2010-07-22 13:59:23 -06:00
void of_device_unregister ( struct platform_device * ofdev )
2007-05-01 16:40:36 +10:00
{
device_unregister ( & ofdev - > dev ) ;
}
EXPORT_SYMBOL ( of_device_unregister ) ;
2008-05-16 11:57:45 +10:00
2010-06-08 07:48:13 -06:00
ssize_t of_device_get_modalias ( struct device * dev , char * str , ssize_t len )
2008-05-16 11:57:45 +10:00
{
const char * compat ;
int cplen , i ;
ssize_t tsize , csize , repend ;
/* Name & Type */
2010-06-08 07:48:13 -06:00
csize = snprintf ( str , len , " of:N%sT%s " , dev - > of_node - > name ,
dev - > of_node - > type ) ;
2008-05-16 11:57:45 +10:00
/* Get compatible property if any */
2010-06-08 07:48:13 -06:00
compat = of_get_property ( dev - > of_node , " compatible " , & cplen ) ;
2008-05-16 11:57:45 +10:00
if ( ! compat )
return csize ;
/* Find true end (we tolerate multiple \0 at the end */
for ( i = ( cplen - 1 ) ; i > = 0 & & ! compat [ i ] ; i - - )
cplen - - ;
if ( ! cplen )
return csize ;
cplen + + ;
/* Check space (need cplen+1 chars including final \0) */
tsize = csize + cplen ;
repend = tsize ;
if ( csize > = len ) /* @ the limit, all is already filled */
return tsize ;
if ( tsize > = len ) { /* limit compat list */
cplen = len - csize - 1 ;
repend = len ;
}
/* Copy and do char replacement */
memcpy ( & str [ csize + 1 ] , compat , cplen ) ;
for ( i = csize ; i < repend ; i + + ) {
char c = str [ i ] ;
if ( c = = ' \0 ' )
str [ i ] = ' C ' ;
else if ( c = = ' ' )
str [ i ] = ' _ ' ;
}
return tsize ;
}
2010-06-08 07:48:12 -06:00
/**
* of_device_uevent - Display OF related uevent information
*/
int of_device_uevent ( struct device * dev , struct kobj_uevent_env * env )
{
const char * compat ;
int seen = 0 , cplen , sl ;
if ( ( ! dev ) | | ( ! dev - > of_node ) )
return - ENODEV ;
if ( add_uevent_var ( env , " OF_NAME=%s " , dev - > of_node - > name ) )
return - ENOMEM ;
if ( add_uevent_var ( env , " OF_TYPE=%s " , dev - > of_node - > type ) )
return - ENOMEM ;
/* Since the compatible field can contain pretty much anything
* it ' s not really legal to split it out with commas . We split it
* up using a number of environment variables instead . */
compat = of_get_property ( dev - > of_node , " compatible " , & cplen ) ;
while ( compat & & * compat & & cplen > 0 ) {
if ( add_uevent_var ( env , " OF_COMPATIBLE_%d=%s " , seen , compat ) )
return - ENOMEM ;
sl = strlen ( compat ) + 1 ;
compat + = sl ;
cplen - = sl ;
seen + + ;
}
if ( add_uevent_var ( env , " OF_COMPATIBLE_N=%d " , seen ) )
return - ENOMEM ;
/* modalias is trickier, we add it in 2 steps */
if ( add_uevent_var ( env , " MODALIAS= " ) )
return - ENOMEM ;
2010-06-08 07:48:13 -06:00
sl = of_device_get_modalias ( dev , & env - > buf [ env - > buflen - 1 ] ,
2010-06-08 07:48:12 -06:00
sizeof ( env - > buf ) - env - > buflen ) ;
if ( sl > = ( sizeof ( env - > buf ) - env - > buflen ) )
return - ENOMEM ;
env - > buflen + = sl ;
return 0 ;
}