2018-11-09 17:21:36 +03:00
// SPDX-License-Identifier: GPL-2.0
/*
* Software nodes for the firmware node framework .
*
* Copyright ( C ) 2018 , Intel Corporation
* Author : Heikki Krogerus < heikki . krogerus @ linux . intel . com >
*/
# include <linux/device.h>
# include <linux/kernel.h>
# include <linux/property.h>
# include <linux/slab.h>
2019-05-31 17:15:34 +03:00
struct swnode {
2018-11-09 17:21:36 +03:00
int id ;
struct kobject kobj ;
struct fwnode_handle fwnode ;
2019-05-31 17:15:34 +03:00
const struct software_node * node ;
2018-11-09 17:21:36 +03:00
/* hierarchy */
struct ida child_ids ;
struct list_head entry ;
struct list_head children ;
2019-05-31 17:15:34 +03:00
struct swnode * parent ;
2018-11-09 17:21:36 +03:00
2019-05-31 17:15:34 +03:00
unsigned int allocated : 1 ;
2018-11-09 17:21:36 +03:00
} ;
static DEFINE_IDA ( swnode_root_ids ) ;
static struct kset * swnode_kset ;
2019-05-31 17:15:34 +03:00
# define kobj_to_swnode(_kobj_) container_of(_kobj_, struct swnode, kobj)
2018-11-09 17:21:36 +03:00
static const struct fwnode_operations software_node_ops ;
bool is_software_node ( const struct fwnode_handle * fwnode )
{
return ! IS_ERR_OR_NULL ( fwnode ) & & fwnode - > ops = = & software_node_ops ;
}
2019-05-31 17:15:34 +03:00
EXPORT_SYMBOL_GPL ( is_software_node ) ;
2018-11-09 17:21:36 +03:00
2019-05-31 17:15:34 +03:00
# define to_swnode(__fwnode) \
2018-11-09 17:21:36 +03:00
( { \
2019-05-31 17:15:34 +03:00
typeof ( __fwnode ) __to_swnode_fwnode = __fwnode ; \
2018-11-09 17:21:36 +03:00
\
2019-05-31 17:15:34 +03:00
is_software_node ( __to_swnode_fwnode ) ? \
container_of ( __to_swnode_fwnode , \
struct swnode , fwnode ) : NULL ; \
2018-11-09 17:21:36 +03:00
} )
2019-05-31 17:15:34 +03:00
static struct swnode *
software_node_to_swnode ( const struct software_node * node )
{
2019-08-30 10:51:55 +03:00
struct swnode * swnode = NULL ;
2019-05-31 17:15:34 +03:00
struct kobject * k ;
if ( ! node )
return NULL ;
spin_lock ( & swnode_kset - > list_lock ) ;
list_for_each_entry ( k , & swnode_kset - > list , entry ) {
swnode = kobj_to_swnode ( k ) ;
if ( swnode - > node = = node )
break ;
swnode = NULL ;
}
spin_unlock ( & swnode_kset - > list_lock ) ;
return swnode ;
}
const struct software_node * to_software_node ( struct fwnode_handle * fwnode )
{
struct swnode * swnode = to_swnode ( fwnode ) ;
return swnode ? swnode - > node : NULL ;
}
EXPORT_SYMBOL_GPL ( to_software_node ) ;
struct fwnode_handle * software_node_fwnode ( const struct software_node * node )
{
struct swnode * swnode = software_node_to_swnode ( node ) ;
return swnode ? & swnode - > fwnode : NULL ;
}
EXPORT_SYMBOL_GPL ( software_node_fwnode ) ;
2018-11-09 17:21:36 +03:00
/* -------------------------------------------------------------------------- */
/* property_entry processing */
static const struct property_entry *
property_entry_get ( const struct property_entry * prop , const char * name )
{
if ( ! prop )
return NULL ;
for ( ; prop - > name ; prop + + )
if ( ! strcmp ( name , prop - > name ) )
return prop ;
return NULL ;
}
2018-11-09 17:21:37 +03:00
static void
property_set_pointer ( struct property_entry * prop , const void * pointer )
{
switch ( prop - > type ) {
case DEV_PROP_U8 :
if ( prop - > is_array )
prop - > pointer . u8_data = pointer ;
else
prop - > value . u8_data = * ( ( u8 * ) pointer ) ;
break ;
case DEV_PROP_U16 :
if ( prop - > is_array )
prop - > pointer . u16_data = pointer ;
else
prop - > value . u16_data = * ( ( u16 * ) pointer ) ;
break ;
case DEV_PROP_U32 :
if ( prop - > is_array )
prop - > pointer . u32_data = pointer ;
else
prop - > value . u32_data = * ( ( u32 * ) pointer ) ;
break ;
case DEV_PROP_U64 :
if ( prop - > is_array )
prop - > pointer . u64_data = pointer ;
else
prop - > value . u64_data = * ( ( u64 * ) pointer ) ;
break ;
case DEV_PROP_STRING :
if ( prop - > is_array )
prop - > pointer . str = pointer ;
else
prop - > value . str = pointer ;
break ;
default :
break ;
}
}
2018-11-09 17:21:36 +03:00
static const void * property_get_pointer ( const struct property_entry * prop )
{
switch ( prop - > type ) {
case DEV_PROP_U8 :
if ( prop - > is_array )
return prop - > pointer . u8_data ;
return & prop - > value . u8_data ;
case DEV_PROP_U16 :
if ( prop - > is_array )
return prop - > pointer . u16_data ;
return & prop - > value . u16_data ;
case DEV_PROP_U32 :
if ( prop - > is_array )
return prop - > pointer . u32_data ;
return & prop - > value . u32_data ;
case DEV_PROP_U64 :
if ( prop - > is_array )
return prop - > pointer . u64_data ;
return & prop - > value . u64_data ;
case DEV_PROP_STRING :
if ( prop - > is_array )
return prop - > pointer . str ;
return & prop - > value . str ;
default :
return NULL ;
}
}
static const void * property_entry_find ( const struct property_entry * props ,
const char * propname , size_t length )
{
const struct property_entry * prop ;
const void * pointer ;
prop = property_entry_get ( props , propname ) ;
if ( ! prop )
return ERR_PTR ( - EINVAL ) ;
pointer = property_get_pointer ( prop ) ;
if ( ! pointer )
return ERR_PTR ( - ENODATA ) ;
if ( length > prop - > length )
return ERR_PTR ( - EOVERFLOW ) ;
return pointer ;
}
static int property_entry_read_u8_array ( const struct property_entry * props ,
const char * propname ,
u8 * values , size_t nval )
{
const void * pointer ;
size_t length = nval * sizeof ( * values ) ;
pointer = property_entry_find ( props , propname , length ) ;
if ( IS_ERR ( pointer ) )
return PTR_ERR ( pointer ) ;
memcpy ( values , pointer , length ) ;
return 0 ;
}
static int property_entry_read_u16_array ( const struct property_entry * props ,
const char * propname ,
u16 * values , size_t nval )
{
const void * pointer ;
size_t length = nval * sizeof ( * values ) ;
pointer = property_entry_find ( props , propname , length ) ;
if ( IS_ERR ( pointer ) )
return PTR_ERR ( pointer ) ;
memcpy ( values , pointer , length ) ;
return 0 ;
}
static int property_entry_read_u32_array ( const struct property_entry * props ,
const char * propname ,
u32 * values , size_t nval )
{
const void * pointer ;
size_t length = nval * sizeof ( * values ) ;
pointer = property_entry_find ( props , propname , length ) ;
if ( IS_ERR ( pointer ) )
return PTR_ERR ( pointer ) ;
memcpy ( values , pointer , length ) ;
return 0 ;
}
static int property_entry_read_u64_array ( const struct property_entry * props ,
const char * propname ,
u64 * values , size_t nval )
{
const void * pointer ;
size_t length = nval * sizeof ( * values ) ;
pointer = property_entry_find ( props , propname , length ) ;
if ( IS_ERR ( pointer ) )
return PTR_ERR ( pointer ) ;
memcpy ( values , pointer , length ) ;
return 0 ;
}
static int
property_entry_count_elems_of_size ( const struct property_entry * props ,
const char * propname , size_t length )
{
const struct property_entry * prop ;
prop = property_entry_get ( props , propname ) ;
if ( ! prop )
return - EINVAL ;
return prop - > length / length ;
}
static int property_entry_read_int_array ( const struct property_entry * props ,
const char * name ,
unsigned int elem_size , void * val ,
size_t nval )
{
if ( ! val )
return property_entry_count_elems_of_size ( props , name ,
elem_size ) ;
switch ( elem_size ) {
case sizeof ( u8 ) :
return property_entry_read_u8_array ( props , name , val , nval ) ;
case sizeof ( u16 ) :
return property_entry_read_u16_array ( props , name , val , nval ) ;
case sizeof ( u32 ) :
return property_entry_read_u32_array ( props , name , val , nval ) ;
case sizeof ( u64 ) :
return property_entry_read_u64_array ( props , name , val , nval ) ;
}
return - ENXIO ;
}
static int property_entry_read_string_array ( const struct property_entry * props ,
const char * propname ,
const char * * strings , size_t nval )
{
const struct property_entry * prop ;
const void * pointer ;
size_t array_len , length ;
/* Find out the array length. */
prop = property_entry_get ( props , propname ) ;
if ( ! prop )
return - EINVAL ;
if ( prop - > is_array )
/* Find the length of an array. */
array_len = property_entry_count_elems_of_size ( props , propname ,
sizeof ( const char * ) ) ;
else
/* The array length for a non-array string property is 1. */
array_len = 1 ;
/* Return how many there are if strings is NULL. */
if ( ! strings )
return array_len ;
array_len = min ( nval , array_len ) ;
length = array_len * sizeof ( * strings ) ;
pointer = property_entry_find ( props , propname , length ) ;
if ( IS_ERR ( pointer ) )
return PTR_ERR ( pointer ) ;
memcpy ( strings , pointer , length ) ;
return array_len ;
}
2018-11-09 17:21:37 +03:00
static void property_entry_free_data ( const struct property_entry * p )
{
const void * pointer = property_get_pointer ( p ) ;
size_t i , nval ;
if ( p - > is_array ) {
if ( p - > type = = DEV_PROP_STRING & & p - > pointer . str ) {
nval = p - > length / sizeof ( const char * ) ;
for ( i = 0 ; i < nval ; i + + )
kfree ( p - > pointer . str [ i ] ) ;
}
kfree ( pointer ) ;
} else if ( p - > type = = DEV_PROP_STRING ) {
kfree ( p - > value . str ) ;
}
kfree ( p - > name ) ;
}
static int property_copy_string_array ( struct property_entry * dst ,
const struct property_entry * src )
{
const char * * d ;
size_t nval = src - > length / sizeof ( * d ) ;
int i ;
d = kcalloc ( nval , sizeof ( * d ) , GFP_KERNEL ) ;
if ( ! d )
return - ENOMEM ;
for ( i = 0 ; i < nval ; i + + ) {
d [ i ] = kstrdup ( src - > pointer . str [ i ] , GFP_KERNEL ) ;
if ( ! d [ i ] & & src - > pointer . str [ i ] ) {
while ( - - i > = 0 )
kfree ( d [ i ] ) ;
kfree ( d ) ;
return - ENOMEM ;
}
}
dst - > pointer . str = d ;
return 0 ;
}
static int property_entry_copy_data ( struct property_entry * dst ,
const struct property_entry * src )
{
const void * pointer = property_get_pointer ( src ) ;
const void * new ;
int error ;
if ( src - > is_array ) {
if ( ! src - > length )
return - ENODATA ;
if ( src - > type = = DEV_PROP_STRING ) {
error = property_copy_string_array ( dst , src ) ;
if ( error )
return error ;
new = dst - > pointer . str ;
} else {
new = kmemdup ( pointer , src - > length , GFP_KERNEL ) ;
if ( ! new )
return - ENOMEM ;
}
} else if ( src - > type = = DEV_PROP_STRING ) {
new = kstrdup ( src - > value . str , GFP_KERNEL ) ;
if ( ! new & & src - > value . str )
return - ENOMEM ;
} else {
new = pointer ;
}
dst - > length = src - > length ;
dst - > is_array = src - > is_array ;
dst - > type = src - > type ;
property_set_pointer ( dst , new ) ;
dst - > name = kstrdup ( src - > name , GFP_KERNEL ) ;
if ( ! dst - > name )
goto out_free_data ;
return 0 ;
out_free_data :
property_entry_free_data ( dst ) ;
return - ENOMEM ;
}
/**
* property_entries_dup - duplicate array of properties
* @ properties : array of properties to copy
*
* This function creates a deep copy of the given NULL - terminated array
* of property entries .
*/
struct property_entry *
property_entries_dup ( const struct property_entry * properties )
{
struct property_entry * p ;
int i , n = 0 ;
int ret ;
2019-05-31 17:15:32 +03:00
if ( ! properties )
return NULL ;
2018-11-09 17:21:37 +03:00
while ( properties [ n ] . name )
n + + ;
p = kcalloc ( n + 1 , sizeof ( * p ) , GFP_KERNEL ) ;
if ( ! p )
return ERR_PTR ( - ENOMEM ) ;
for ( i = 0 ; i < n ; i + + ) {
ret = property_entry_copy_data ( & p [ i ] , & properties [ i ] ) ;
if ( ret ) {
while ( - - i > = 0 )
property_entry_free_data ( & p [ i ] ) ;
kfree ( p ) ;
return ERR_PTR ( ret ) ;
}
}
return p ;
}
EXPORT_SYMBOL_GPL ( property_entries_dup ) ;
/**
* property_entries_free - free previously allocated array of properties
* @ properties : array of properties to destroy
*
* This function frees given NULL - terminated array of property entries ,
* along with their data .
*/
void property_entries_free ( const struct property_entry * properties )
{
const struct property_entry * p ;
if ( ! properties )
return ;
for ( p = properties ; p - > name ; p + + )
property_entry_free_data ( p ) ;
kfree ( properties ) ;
}
EXPORT_SYMBOL_GPL ( property_entries_free ) ;
2018-11-09 17:21:36 +03:00
/* -------------------------------------------------------------------------- */
/* fwnode operations */
static struct fwnode_handle * software_node_get ( struct fwnode_handle * fwnode )
{
2019-05-31 17:15:34 +03:00
struct swnode * swnode = to_swnode ( fwnode ) ;
2018-11-09 17:21:36 +03:00
kobject_get ( & swnode - > kobj ) ;
return & swnode - > fwnode ;
}
static void software_node_put ( struct fwnode_handle * fwnode )
{
2019-05-31 17:15:34 +03:00
struct swnode * swnode = to_swnode ( fwnode ) ;
2018-11-09 17:21:36 +03:00
kobject_put ( & swnode - > kobj ) ;
}
static bool software_node_property_present ( const struct fwnode_handle * fwnode ,
const char * propname )
{
2019-05-31 17:15:34 +03:00
struct swnode * swnode = to_swnode ( fwnode ) ;
return ! ! property_entry_get ( swnode - > node - > properties , propname ) ;
2018-11-09 17:21:36 +03:00
}
static int software_node_read_int_array ( const struct fwnode_handle * fwnode ,
const char * propname ,
unsigned int elem_size , void * val ,
size_t nval )
{
2019-05-31 17:15:34 +03:00
struct swnode * swnode = to_swnode ( fwnode ) ;
2018-11-09 17:21:36 +03:00
2019-05-31 17:15:34 +03:00
return property_entry_read_int_array ( swnode - > node - > properties , propname ,
2018-11-09 17:21:36 +03:00
elem_size , val , nval ) ;
}
static int software_node_read_string_array ( const struct fwnode_handle * fwnode ,
const char * propname ,
const char * * val , size_t nval )
{
2019-05-31 17:15:34 +03:00
struct swnode * swnode = to_swnode ( fwnode ) ;
2018-11-09 17:21:36 +03:00
2019-05-31 17:15:34 +03:00
return property_entry_read_string_array ( swnode - > node - > properties ,
propname , val , nval ) ;
2018-11-09 17:21:36 +03:00
}
2019-03-19 18:20:42 +03:00
static struct fwnode_handle *
2018-11-09 17:21:36 +03:00
software_node_get_parent ( const struct fwnode_handle * fwnode )
{
2019-05-31 17:15:34 +03:00
struct swnode * swnode = to_swnode ( fwnode ) ;
2018-11-09 17:21:36 +03:00
2019-05-31 17:15:34 +03:00
return swnode ? ( swnode - > parent ? & swnode - > parent - > fwnode : NULL ) : NULL ;
2018-11-09 17:21:36 +03:00
}
2019-03-19 18:20:42 +03:00
static struct fwnode_handle *
2018-11-09 17:21:36 +03:00
software_node_get_next_child ( const struct fwnode_handle * fwnode ,
struct fwnode_handle * child )
{
2019-05-31 17:15:34 +03:00
struct swnode * p = to_swnode ( fwnode ) ;
struct swnode * c = to_swnode ( child ) ;
2018-11-09 17:21:36 +03:00
2018-12-22 15:49:39 +03:00
if ( ! p | | list_empty ( & p - > children ) | |
2018-11-09 17:21:36 +03:00
( c & & list_is_last ( & c - > entry , & p - > children ) ) )
return NULL ;
if ( c )
c = list_next_entry ( c , entry ) ;
else
2019-05-31 17:15:34 +03:00
c = list_first_entry ( & p - > children , struct swnode , entry ) ;
2018-11-09 17:21:36 +03:00
return & c - > fwnode ;
}
2019-02-13 14:55:49 +03:00
static struct fwnode_handle *
software_node_get_named_child_node ( const struct fwnode_handle * fwnode ,
const char * childname )
{
2019-05-31 17:15:34 +03:00
struct swnode * swnode = to_swnode ( fwnode ) ;
struct swnode * child ;
2019-02-13 14:55:49 +03:00
if ( ! swnode | | list_empty ( & swnode - > children ) )
return NULL ;
list_for_each_entry ( child , & swnode - > children , entry ) {
2019-05-31 17:15:35 +03:00
if ( ! strcmp ( childname , kobject_name ( & child - > kobj ) ) ) {
2019-02-13 14:55:49 +03:00
kobject_get ( & child - > kobj ) ;
return & child - > fwnode ;
}
}
return NULL ;
}
2018-11-09 17:21:36 +03:00
2019-05-31 17:15:36 +03:00
static int
software_node_get_reference_args ( const struct fwnode_handle * fwnode ,
const char * propname , const char * nargs_prop ,
unsigned int nargs , unsigned int index ,
struct fwnode_reference_args * args )
{
struct swnode * swnode = to_swnode ( fwnode ) ;
const struct software_node_reference * ref ;
const struct property_entry * prop ;
struct fwnode_handle * refnode ;
int i ;
if ( ! swnode | | ! swnode - > node - > references )
return - ENOENT ;
for ( ref = swnode - > node - > references ; ref - > name ; ref + + )
if ( ! strcmp ( ref - > name , propname ) )
break ;
if ( ! ref - > name | | index > ( ref - > nrefs - 1 ) )
return - ENOENT ;
refnode = software_node_fwnode ( ref - > refs [ index ] . node ) ;
if ( ! refnode )
return - ENOENT ;
if ( nargs_prop ) {
prop = property_entry_get ( swnode - > node - > properties , nargs_prop ) ;
if ( ! prop )
return - EINVAL ;
nargs = prop - > value . u32_data ;
}
if ( nargs > NR_FWNODE_REFERENCE_ARGS )
return - EINVAL ;
args - > fwnode = software_node_get ( refnode ) ;
args - > nargs = nargs ;
for ( i = 0 ; i < nargs ; i + + )
args - > args [ i ] = ref - > refs [ index ] . args [ i ] ;
return 0 ;
}
2018-11-09 17:21:36 +03:00
static const struct fwnode_operations software_node_ops = {
. get = software_node_get ,
. put = software_node_put ,
. property_present = software_node_property_present ,
. property_read_int_array = software_node_read_int_array ,
. property_read_string_array = software_node_read_string_array ,
. get_parent = software_node_get_parent ,
. get_next_child_node = software_node_get_next_child ,
2019-02-13 14:55:49 +03:00
. get_named_child_node = software_node_get_named_child_node ,
2019-05-31 17:15:36 +03:00
. get_reference_args = software_node_get_reference_args
2018-11-09 17:21:36 +03:00
} ;
/* -------------------------------------------------------------------------- */
2019-08-19 13:07:22 +03:00
/**
* software_node_find_by_name - Find software node by name
* @ parent : Parent of the software node
* @ name : Name of the software node
*
* The function will find a node that is child of @ parent and that is named
* @ name . If no node is found , the function returns NULL .
*
* NOTE : you will need to drop the reference with fwnode_handle_put ( ) after use .
*/
const struct software_node *
software_node_find_by_name ( const struct software_node * parent , const char * name )
{
2019-08-30 10:51:56 +03:00
struct swnode * swnode = NULL ;
2019-08-19 13:07:22 +03:00
struct kobject * k ;
if ( ! name )
return NULL ;
spin_lock ( & swnode_kset - > list_lock ) ;
list_for_each_entry ( k , & swnode_kset - > list , entry ) {
swnode = kobj_to_swnode ( k ) ;
if ( parent = = swnode - > node - > parent & & swnode - > node - > name & &
! strcmp ( name , swnode - > node - > name ) ) {
kobject_get ( & swnode - > kobj ) ;
break ;
}
swnode = NULL ;
}
spin_unlock ( & swnode_kset - > list_lock ) ;
return swnode ? swnode - > node : NULL ;
}
EXPORT_SYMBOL_GPL ( software_node_find_by_name ) ;
2018-11-09 17:21:36 +03:00
static int
2019-05-31 17:15:34 +03:00
software_node_register_properties ( struct software_node * node ,
2018-11-09 17:21:36 +03:00
const struct property_entry * properties )
{
struct property_entry * props ;
props = property_entries_dup ( properties ) ;
if ( IS_ERR ( props ) )
return PTR_ERR ( props ) ;
2019-05-31 17:15:34 +03:00
node - > properties = props ;
2018-11-09 17:21:36 +03:00
return 0 ;
}
static void software_node_release ( struct kobject * kobj )
{
2019-05-31 17:15:34 +03:00
struct swnode * swnode = kobj_to_swnode ( kobj ) ;
2018-11-09 17:21:36 +03:00
2019-05-31 17:15:34 +03:00
if ( swnode - > allocated ) {
property_entries_free ( swnode - > node - > properties ) ;
kfree ( swnode - > node ) ;
}
2018-11-09 17:21:36 +03:00
ida_destroy ( & swnode - > child_ids ) ;
kfree ( swnode ) ;
}
static struct kobj_type software_node_type = {
. release = software_node_release ,
. sysfs_ops = & kobj_sysfs_ops ,
} ;
2019-05-31 17:15:34 +03:00
static struct fwnode_handle *
swnode_register ( const struct software_node * node , struct swnode * parent ,
unsigned int allocated )
2018-11-09 17:21:36 +03:00
{
2019-05-31 17:15:34 +03:00
struct swnode * swnode ;
2018-11-09 17:21:36 +03:00
int ret ;
swnode = kzalloc ( sizeof ( * swnode ) , GFP_KERNEL ) ;
2019-05-31 17:15:34 +03:00
if ( ! swnode ) {
ret = - ENOMEM ;
goto out_err ;
}
2018-11-09 17:21:36 +03:00
2019-05-31 17:15:34 +03:00
ret = ida_simple_get ( parent ? & parent - > child_ids : & swnode_root_ids ,
0 , 0 , GFP_KERNEL ) ;
2018-11-09 17:21:36 +03:00
if ( ret < 0 ) {
kfree ( swnode ) ;
2019-05-31 17:15:34 +03:00
goto out_err ;
2018-11-09 17:21:36 +03:00
}
swnode - > id = ret ;
2019-05-31 17:15:34 +03:00
swnode - > node = node ;
swnode - > parent = parent ;
swnode - > allocated = allocated ;
2018-11-09 17:21:36 +03:00
swnode - > kobj . kset = swnode_kset ;
swnode - > fwnode . ops = & software_node_ops ;
ida_init ( & swnode - > child_ids ) ;
INIT_LIST_HEAD ( & swnode - > entry ) ;
INIT_LIST_HEAD ( & swnode - > children ) ;
2019-05-31 17:15:34 +03:00
if ( node - > name )
ret = kobject_init_and_add ( & swnode - > kobj , & software_node_type ,
parent ? & parent - > kobj : NULL ,
" %s " , node - > name ) ;
else
ret = kobject_init_and_add ( & swnode - > kobj , & software_node_type ,
parent ? & parent - > kobj : NULL ,
" node%d " , swnode - > id ) ;
2018-11-09 17:21:36 +03:00
if ( ret ) {
kobject_put ( & swnode - > kobj ) ;
return ERR_PTR ( ret ) ;
}
2019-05-31 17:15:34 +03:00
if ( parent )
list_add_tail ( & swnode - > entry , & parent - > children ) ;
kobject_uevent ( & swnode - > kobj , KOBJ_ADD ) ;
return & swnode - > fwnode ;
out_err :
if ( allocated )
property_entries_free ( node - > properties ) ;
return ERR_PTR ( ret ) ;
}
/**
* software_node_register_nodes - Register an array of software nodes
* @ nodes : Zero terminated array of software nodes to be registered
*
* Register multiple software nodes at once .
*/
int software_node_register_nodes ( const struct software_node * nodes )
{
int ret ;
int i ;
for ( i = 0 ; nodes [ i ] . name ; i + + ) {
ret = software_node_register ( & nodes [ i ] ) ;
if ( ret ) {
software_node_unregister_nodes ( nodes ) ;
return ret ;
}
}
return 0 ;
}
EXPORT_SYMBOL_GPL ( software_node_register_nodes ) ;
/**
* software_node_unregister_nodes - Unregister an array of software nodes
* @ nodes : Zero terminated array of software nodes to be unregistered
*
* Unregister multiple software nodes at once .
*/
void software_node_unregister_nodes ( const struct software_node * nodes )
{
struct swnode * swnode ;
int i ;
for ( i = 0 ; nodes [ i ] . name ; i + + ) {
swnode = software_node_to_swnode ( & nodes [ i ] ) ;
if ( swnode )
fwnode_remove_software_node ( & swnode - > fwnode ) ;
}
}
EXPORT_SYMBOL_GPL ( software_node_unregister_nodes ) ;
/**
* software_node_register - Register static software node
* @ node : The software node to be registered
*/
int software_node_register ( const struct software_node * node )
{
struct swnode * parent = software_node_to_swnode ( node - > parent ) ;
if ( software_node_to_swnode ( node ) )
return - EEXIST ;
return PTR_ERR_OR_ZERO ( swnode_register ( node , parent , 0 ) ) ;
}
EXPORT_SYMBOL_GPL ( software_node_register ) ;
struct fwnode_handle *
fwnode_create_software_node ( const struct property_entry * properties ,
const struct fwnode_handle * parent )
{
struct software_node * node ;
struct swnode * p = NULL ;
int ret ;
if ( parent ) {
if ( IS_ERR ( parent ) )
return ERR_CAST ( parent ) ;
if ( ! is_software_node ( parent ) )
return ERR_PTR ( - EINVAL ) ;
p = to_swnode ( parent ) ;
}
node = kzalloc ( sizeof ( * node ) , GFP_KERNEL ) ;
if ( ! node )
return ERR_PTR ( - ENOMEM ) ;
ret = software_node_register_properties ( node , properties ) ;
2018-11-09 17:21:36 +03:00
if ( ret ) {
2019-05-31 17:15:34 +03:00
kfree ( node ) ;
2018-11-09 17:21:36 +03:00
return ERR_PTR ( ret ) ;
}
2019-05-31 17:15:34 +03:00
node - > parent = p ? p - > node : NULL ;
2019-05-31 17:15:33 +03:00
2019-05-31 17:15:34 +03:00
return swnode_register ( node , p , 1 ) ;
2018-11-09 17:21:36 +03:00
}
EXPORT_SYMBOL_GPL ( fwnode_create_software_node ) ;
void fwnode_remove_software_node ( struct fwnode_handle * fwnode )
{
2019-05-31 17:15:34 +03:00
struct swnode * swnode = to_swnode ( fwnode ) ;
2018-11-09 17:21:36 +03:00
if ( ! swnode )
return ;
2019-05-31 17:15:33 +03:00
if ( swnode - > parent ) {
ida_simple_remove ( & swnode - > parent - > child_ids , swnode - > id ) ;
list_del ( & swnode - > entry ) ;
} else {
ida_simple_remove ( & swnode_root_ids , swnode - > id ) ;
}
2018-11-09 17:21:36 +03:00
kobject_put ( & swnode - > kobj ) ;
}
EXPORT_SYMBOL_GPL ( fwnode_remove_software_node ) ;
int software_node_notify ( struct device * dev , unsigned long action )
{
struct fwnode_handle * fwnode = dev_fwnode ( dev ) ;
2019-05-31 17:15:34 +03:00
struct swnode * swnode ;
2018-11-09 17:21:36 +03:00
int ret ;
if ( ! fwnode )
return 0 ;
if ( ! is_software_node ( fwnode ) )
fwnode = fwnode - > secondary ;
if ( ! is_software_node ( fwnode ) )
return 0 ;
2019-05-31 17:15:34 +03:00
swnode = to_swnode ( fwnode ) ;
2018-11-09 17:21:36 +03:00
switch ( action ) {
case KOBJ_ADD :
ret = sysfs_create_link ( & dev - > kobj , & swnode - > kobj ,
" software_node " ) ;
if ( ret )
break ;
ret = sysfs_create_link ( & swnode - > kobj , & dev - > kobj ,
dev_name ( dev ) ) ;
if ( ret ) {
sysfs_remove_link ( & dev - > kobj , " software_node " ) ;
break ;
}
kobject_get ( & swnode - > kobj ) ;
break ;
case KOBJ_REMOVE :
sysfs_remove_link ( & swnode - > kobj , dev_name ( dev ) ) ;
sysfs_remove_link ( & dev - > kobj , " software_node " ) ;
kobject_put ( & swnode - > kobj ) ;
break ;
default :
break ;
}
return 0 ;
}
static int __init software_node_init ( void )
{
swnode_kset = kset_create_and_add ( " software_nodes " , NULL , kernel_kobj ) ;
if ( ! swnode_kset )
return - ENOMEM ;
return 0 ;
}
postcore_initcall ( software_node_init ) ;
static void __exit software_node_exit ( void )
{
ida_destroy ( & swnode_root_ids ) ;
kset_unregister ( swnode_kset ) ;
}
__exitcall ( software_node_exit ) ;