2005-04-17 02:20:36 +04:00
/*
* kobject . c - library routines for handling generic kernel objects
*
* Copyright ( c ) 2002 - 2003 Patrick Mochel < mochel @ osdl . org >
2007-09-28 01:48:53 +04:00
* Copyright ( c ) 2006 - 2007 Greg Kroah - Hartman < greg @ kroah . com >
* Copyright ( c ) 2006 - 2007 Novell Inc .
2005-04-17 02:20:36 +04:00
*
* This file is released under the GPLv2 .
*
*
* Please see the file Documentation / kobject . txt for critical information
* about using the kobject interface .
*/
# include <linux/kobject.h>
# include <linux/string.h>
# include <linux/module.h>
# include <linux/stat.h>
2005-10-31 02:03:48 +03:00
# include <linux/slab.h>
2005-04-17 02:20:36 +04:00
/**
* populate_dir - populate directory with attributes .
* @ kobj : object we ' re working on .
*
* Most subsystems have a set of default attributes that
* are associated with an object that registers with them .
* This is a helper called during object registration that
* loops through the default attributes of the subsystem
* and creates attributes files for them in sysfs .
*
*/
static int populate_dir ( struct kobject * kobj )
{
struct kobj_type * t = get_ktype ( kobj ) ;
struct attribute * attr ;
int error = 0 ;
int i ;
if ( t & & t - > default_attrs ) {
for ( i = 0 ; ( attr = t - > default_attrs [ i ] ) ! = NULL ; i + + ) {
if ( ( error = sysfs_create_file ( kobj , attr ) ) )
break ;
}
}
return error ;
}
2007-07-31 14:15:08 +04:00
static int create_dir ( struct kobject * kobj )
2005-04-17 02:20:36 +04:00
{
int error = 0 ;
if ( kobject_name ( kobj ) ) {
2007-07-31 14:15:08 +04:00
error = sysfs_create_dir ( kobj ) ;
2005-04-17 02:20:36 +04:00
if ( ! error ) {
if ( ( error = populate_dir ( kobj ) ) )
sysfs_remove_dir ( kobj ) ;
}
}
return error ;
}
static inline struct kobject * to_kobj ( struct list_head * entry )
{
return container_of ( entry , struct kobject , entry ) ;
}
static int get_kobj_path_length ( struct kobject * kobj )
{
int length = 1 ;
struct kobject * parent = kobj ;
/* walk up the ancestors until we hit the one pointing to the
* root .
* Add 1 to strlen for leading ' / ' of each level .
*/
do {
2006-01-13 04:02:00 +03:00
if ( kobject_name ( parent ) = = NULL )
return 0 ;
2005-04-17 02:20:36 +04:00
length + = strlen ( kobject_name ( parent ) ) + 1 ;
parent = parent - > parent ;
} while ( parent ) ;
return length ;
}
static void fill_kobj_path ( struct kobject * kobj , char * path , int length )
{
struct kobject * parent ;
- - length ;
for ( parent = kobj ; parent ; parent = parent - > parent ) {
int cur = strlen ( kobject_name ( parent ) ) ;
/* back up enough to print this name with '/' */
length - = cur ;
strncpy ( path + length , kobject_name ( parent ) , cur ) ;
* ( path + - - length ) = ' / ' ;
}
pr_debug ( " %s: path = '%s' \n " , __FUNCTION__ , path ) ;
}
/**
2007-02-10 12:45:59 +03:00
* kobject_get_path - generate and return the path associated with a given kobj and kset pair .
2005-04-17 02:20:36 +04:00
*
* @ kobj : kobject in question , with which to build the path
* @ gfp_mask : the allocation type used to allocate the path
2007-02-10 12:45:59 +03:00
*
* The result must be freed by the caller with kfree ( ) .
2005-04-17 02:20:36 +04:00
*/
2005-10-21 11:18:50 +04:00
char * kobject_get_path ( struct kobject * kobj , gfp_t gfp_mask )
2005-04-17 02:20:36 +04:00
{
char * path ;
int len ;
len = get_kobj_path_length ( kobj ) ;
2006-01-13 04:02:00 +03:00
if ( len = = 0 )
return NULL ;
2006-12-07 07:38:51 +03:00
path = kzalloc ( len , gfp_mask ) ;
2005-04-17 02:20:36 +04:00
if ( ! path )
return NULL ;
fill_kobj_path ( kobj , path , len ) ;
return path ;
}
2006-10-11 09:43:58 +04:00
EXPORT_SYMBOL_GPL ( kobject_get_path ) ;
2005-04-17 02:20:36 +04:00
/**
* kobject_init - initialize object .
* @ kobj : object in question .
*/
void kobject_init ( struct kobject * kobj )
{
2007-01-18 23:23:51 +03:00
if ( ! kobj )
return ;
2005-04-17 02:20:36 +04:00
kref_init ( & kobj - > kref ) ;
INIT_LIST_HEAD ( & kobj - > entry ) ;
kobj - > kset = kset_get ( kobj - > kset ) ;
}
/**
* unlink - remove kobject from kset list .
* @ kobj : kobject .
*
* Remove the kobject from the kset list and decrement
* its parent ' s refcount .
* This is separated out , so we can use it in both
* kobject_del ( ) and kobject_add ( ) on error .
*/
static void unlink ( struct kobject * kobj )
{
if ( kobj - > kset ) {
spin_lock ( & kobj - > kset - > list_lock ) ;
list_del_init ( & kobj - > entry ) ;
spin_unlock ( & kobj - > kset - > list_lock ) ;
}
kobject_put ( kobj ) ;
}
/**
2007-07-31 14:15:08 +04:00
* kobject_add - add an object to the hierarchy .
2005-04-17 02:20:36 +04:00
* @ kobj : object .
*/
2007-07-31 14:15:08 +04:00
int kobject_add ( struct kobject * kobj )
2005-04-17 02:20:36 +04:00
{
int error = 0 ;
struct kobject * parent ;
if ( ! ( kobj = kobject_get ( kobj ) ) )
return - ENOENT ;
if ( ! kobj - > k_name )
2007-09-13 02:06:57 +04:00
kobject_set_name ( kobj , " NO_NAME " ) ;
2007-02-06 03:15:23 +03:00
if ( ! * kobj - > k_name ) {
2006-01-21 01:08:59 +03:00
pr_debug ( " kobject attempted to be registered with no name! \n " ) ;
WARN_ON ( 1 ) ;
2007-04-10 16:35:27 +04:00
kobject_put ( kobj ) ;
2006-01-21 01:08:59 +03:00
return - EINVAL ;
}
2005-04-17 02:20:36 +04:00
parent = kobject_get ( kobj - > parent ) ;
pr_debug ( " kobject %s: registering. parent: %s, set: %s \n " ,
kobject_name ( kobj ) , parent ? kobject_name ( parent ) : " <NULL> " ,
2007-09-13 02:06:57 +04:00
kobj - > kset ? kobject_name ( & kobj - > kset - > kobj ) : " <NULL> " ) ;
2005-04-17 02:20:36 +04:00
if ( kobj - > kset ) {
spin_lock ( & kobj - > kset - > list_lock ) ;
2007-09-28 01:48:53 +04:00
if ( ! parent ) {
2005-04-17 02:20:36 +04:00
parent = kobject_get ( & kobj - > kset - > kobj ) ;
2007-09-28 01:48:53 +04:00
/*
* If the kset is our parent , get a second
* reference , we drop both the kset and the
* parent ref on cleanup
*/
kobject_get ( parent ) ;
}
2005-04-17 02:20:36 +04:00
list_add_tail ( & kobj - > entry , & kobj - > kset - > list ) ;
spin_unlock ( & kobj - > kset - > list_lock ) ;
2007-03-10 14:00:10 +03:00
kobj - > parent = parent ;
2005-04-17 02:20:36 +04:00
}
2007-07-31 14:15:08 +04:00
error = create_dir ( kobj ) ;
2005-04-17 02:20:36 +04:00
if ( error ) {
/* unlink does the kobject_put() for us */
unlink ( kobj ) ;
2007-01-02 15:44:44 +03:00
kobject_put ( parent ) ;
2006-03-21 00:17:13 +03:00
/* be noisy on error issues */
if ( error = = - EEXIST )
2007-06-08 02:05:15 +04:00
printk ( KERN_ERR " kobject_add failed for %s with "
" -EEXIST, don't try to register things with "
" the same name in the same directory. \n " ,
2006-03-21 00:17:13 +03:00
kobject_name ( kobj ) ) ;
else
2007-06-08 02:05:15 +04:00
printk ( KERN_ERR " kobject_add failed for %s (%d) \n " ,
2006-03-21 00:17:13 +03:00
kobject_name ( kobj ) , error ) ;
2007-06-08 02:05:15 +04:00
dump_stack ( ) ;
2005-04-17 02:20:36 +04:00
}
return error ;
}
/**
* kobject_register - initialize and add an object .
* @ kobj : object in question .
*/
int kobject_register ( struct kobject * kobj )
{
2006-03-21 00:17:13 +03:00
int error = - EINVAL ;
2005-04-17 02:20:36 +04:00
if ( kobj ) {
kobject_init ( kobj ) ;
error = kobject_add ( kobj ) ;
2006-03-21 00:17:13 +03:00
if ( ! error )
2005-11-16 11:00:00 +03:00
kobject_uevent ( kobj , KOBJ_ADD ) ;
2006-03-21 00:17:13 +03:00
}
2005-04-17 02:20:36 +04:00
return error ;
}
2007-11-30 02:32:47 +03:00
/**
* kobject_set_name_vargs - Set the name of an kobject
* @ kobj : struct kobject to set the name of
* @ fmt : format string used to build the name
* @ vargs : vargs to format the string .
*/
static int kobject_set_name_vargs ( struct kobject * kobj , const char * fmt ,
va_list vargs )
{
va_list aq ;
char * name ;
va_copy ( aq , vargs ) ;
name = kvasprintf ( GFP_KERNEL , fmt , vargs ) ;
va_end ( aq ) ;
if ( ! name )
return - ENOMEM ;
/* Free the old name, if necessary. */
kfree ( kobj - > k_name ) ;
/* Now, set the new name */
kobj - > k_name = name ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
/**
2007-12-04 09:45:47 +03:00
* kobject_set_name - Set the name of a kobject
2007-11-30 02:32:47 +03:00
* @ kobj : struct kobject to set the name of
2007-12-04 09:45:47 +03:00
* @ fmt : format string used to build the name
2005-04-17 02:20:36 +04:00
*
2007-12-04 09:45:47 +03:00
* This sets the name of the kobject . If you have already added the
* kobject to the system , you must call kobject_rename ( ) in order to
* change the name of the kobject .
2005-04-17 02:20:36 +04:00
*/
2007-11-30 02:32:47 +03:00
int kobject_set_name ( struct kobject * kobj , const char * fmt , . . . )
2005-04-17 02:20:36 +04:00
{
va_list args ;
2007-11-30 02:32:47 +03:00
int retval ;
2005-04-17 02:20:36 +04:00
2007-09-13 02:06:57 +04:00
va_start ( args , fmt ) ;
2007-11-30 02:32:47 +03:00
retval = kobject_set_name_vargs ( kobj , fmt , args ) ;
2005-04-17 02:20:36 +04:00
va_end ( args ) ;
2007-11-30 02:32:47 +03:00
return retval ;
2005-04-17 02:20:36 +04:00
}
EXPORT_SYMBOL ( kobject_set_name ) ;
2007-12-04 08:31:08 +03:00
/**
* kobject_init_ng - initialize a kobject structure
* @ kobj : pointer to the kobject to initialize
* @ ktype : pointer to the ktype for this kobject .
*
* This function will properly initialize a kobject such that it can then
* be passed to the kobject_add ( ) call .
*
* After this function is called , the kobject MUST be cleaned up by a call
* to kobject_put ( ) , not by a call to kfree directly to ensure that all of
* the memory is cleaned up properly .
*/
void kobject_init_ng ( struct kobject * kobj , struct kobj_type * ktype )
{
char * err_str ;
if ( ! kobj ) {
err_str = " invalid kobject pointer! " ;
goto error ;
}
if ( ! ktype ) {
err_str = " must have a ktype to be initialized properly! \n " ;
goto error ;
}
if ( atomic_read ( & kobj - > kref . refcount ) ) {
/* do not error out as sometimes we can recover */
printk ( KERN_ERR " kobject: reference count is already set, "
" something is seriously wrong. \n " ) ;
dump_stack ( ) ;
}
kref_init ( & kobj - > kref ) ;
INIT_LIST_HEAD ( & kobj - > entry ) ;
kobj - > ktype = ktype ;
return ;
error :
printk ( KERN_ERR " kobject: %s \n " , err_str ) ;
dump_stack ( ) ;
}
EXPORT_SYMBOL ( kobject_init_ng ) ;
2007-12-04 08:31:08 +03:00
static int kobject_add_varg ( struct kobject * kobj , struct kobject * parent ,
const char * fmt , va_list vargs )
{
va_list aq ;
int retval ;
va_copy ( aq , vargs ) ;
retval = kobject_set_name_vargs ( kobj , fmt , aq ) ;
va_end ( aq ) ;
if ( retval ) {
printk ( KERN_ERR " kobject: can not set name properly! \n " ) ;
return retval ;
}
kobj - > parent = parent ;
return kobject_add ( kobj ) ;
}
/**
* kobject_add_ng - the main kobject add function
* @ kobj : the kobject to add
* @ parent : pointer to the parent of the kobject .
* @ fmt : format to name the kobject with .
*
* The kobject name is set and added to the kobject hierarchy in this
* function .
*
* If @ parent is set , then the parent of the @ kobj will be set to it .
* If @ parent is NULL , then the parent of the @ kobj will be set to the
* kobject associted with the kset assigned to this kobject . If no kset
* is assigned to the kobject , then the kobject will be located in the
* root of the sysfs tree .
*
* If this function returns an error , kobject_put ( ) must be called to
* properly clean up the memory associated with the object .
*
* If the function is successful , the only way to properly clean up the
* memory is with a call to kobject_del ( ) , in which case , a call to
* kobject_put ( ) is not necessary ( kobject_del ( ) does the final
* kobject_put ( ) to call the release function in the ktype ' s release
* pointer . )
*
* Under no instance should the kobject that is passed to this function
* be directly freed with a call to kfree ( ) , that can leak memory .
*
* Note , no uevent will be created with this call , the caller should set
* up all of the necessary sysfs files for the object and then call
* kobject_uevent ( ) with the UEVENT_ADD parameter to ensure that
* userspace is properly notified of this kobject ' s creation .
*/
int kobject_add_ng ( struct kobject * kobj , struct kobject * parent ,
const char * fmt , . . . )
{
va_list args ;
int retval ;
if ( ! kobj )
return - EINVAL ;
va_start ( args , fmt ) ;
retval = kobject_add_varg ( kobj , parent , fmt , args ) ;
va_end ( args ) ;
return retval ;
}
EXPORT_SYMBOL ( kobject_add_ng ) ;
2007-12-04 08:31:08 +03:00
/**
* kobject_init_and_add - initialize a kobject structure and add it to the kobject hierarchy
* @ kobj : pointer to the kobject to initialize
* @ ktype : pointer to the ktype for this kobject .
* @ parent : pointer to the parent of this kobject .
* @ fmt : the name of the kobject .
*
* This function combines the call to kobject_init_ng ( ) and
* kobject_add_ng ( ) . The same type of error handling after a call to
* kobject_add_ng ( ) and kobject lifetime rules are the same here .
*/
int kobject_init_and_add ( struct kobject * kobj , struct kobj_type * ktype ,
struct kobject * parent , const char * fmt , . . . )
{
va_list args ;
int retval ;
kobject_init_ng ( kobj , ktype ) ;
va_start ( args , fmt ) ;
retval = kobject_add_varg ( kobj , parent , fmt , args ) ;
va_end ( args ) ;
return retval ;
}
EXPORT_SYMBOL_GPL ( kobject_init_and_add ) ;
2005-04-17 02:20:36 +04:00
/**
* kobject_rename - change the name of an object
* @ kobj : object in question .
* @ new_name : object ' s new name
*/
2005-04-26 11:32:00 +04:00
int kobject_rename ( struct kobject * kobj , const char * new_name )
2005-04-17 02:20:36 +04:00
{
int error = 0 ;
2007-03-07 21:49:30 +03:00
const char * devpath = NULL ;
char * devpath_string = NULL ;
char * envp [ 2 ] ;
2005-04-17 02:20:36 +04:00
kobj = kobject_get ( kobj ) ;
if ( ! kobj )
return - EINVAL ;
2007-01-24 22:35:52 +03:00
if ( ! kobj - > parent )
return - EINVAL ;
2007-03-07 21:49:30 +03:00
2007-10-25 03:52:31 +04:00
/* see if this name is already in use */
if ( kobj - > kset ) {
struct kobject * temp_kobj ;
temp_kobj = kset_find_obj ( kobj - > kset , new_name ) ;
if ( temp_kobj ) {
2007-11-05 15:59:11 +03:00
printk ( KERN_WARNING " kobject '%s' cannot be renamed "
" to '%s' as '%s' is already in existence. \n " ,
2007-10-25 03:52:31 +04:00
kobject_name ( kobj ) , new_name , new_name ) ;
kobject_put ( temp_kobj ) ;
return - EINVAL ;
}
}
2007-03-07 21:49:30 +03:00
devpath = kobject_get_path ( kobj , GFP_KERNEL ) ;
if ( ! devpath ) {
error = - ENOMEM ;
goto out ;
}
devpath_string = kmalloc ( strlen ( devpath ) + 15 , GFP_KERNEL ) ;
if ( ! devpath_string ) {
error = - ENOMEM ;
goto out ;
}
sprintf ( devpath_string , " DEVPATH_OLD=%s " , devpath ) ;
envp [ 0 ] = devpath_string ;
envp [ 1 ] = NULL ;
2007-07-31 14:15:08 +04:00
error = sysfs_rename_dir ( kobj , new_name ) ;
2007-03-07 21:49:30 +03:00
/* This function is mostly/only used for network interface.
* Some hotplug package track interfaces by their name and
* therefore want to know when the name is changed by the user . */
if ( ! error )
kobject_uevent_env ( kobj , KOBJ_MOVE , envp ) ;
out :
kfree ( devpath_string ) ;
kfree ( devpath ) ;
2007-01-24 22:35:52 +03:00
kobject_put ( kobj ) ;
return error ;
}
2006-11-20 19:07:51 +03:00
/**
* kobject_move - move object to another parent
* @ kobj : object in question .
2007-01-08 22:16:44 +03:00
* @ new_parent : object ' s new parent ( can be NULL )
2006-11-20 19:07:51 +03:00
*/
int kobject_move ( struct kobject * kobj , struct kobject * new_parent )
{
int error ;
struct kobject * old_parent ;
const char * devpath = NULL ;
char * devpath_string = NULL ;
char * envp [ 2 ] ;
kobj = kobject_get ( kobj ) ;
if ( ! kobj )
return - EINVAL ;
new_parent = kobject_get ( new_parent ) ;
if ( ! new_parent ) {
2007-01-08 22:16:44 +03:00
if ( kobj - > kset )
new_parent = kobject_get ( & kobj - > kset - > kobj ) ;
2006-11-20 19:07:51 +03:00
}
/* old object path */
devpath = kobject_get_path ( kobj , GFP_KERNEL ) ;
if ( ! devpath ) {
error = - ENOMEM ;
goto out ;
}
devpath_string = kmalloc ( strlen ( devpath ) + 15 , GFP_KERNEL ) ;
if ( ! devpath_string ) {
error = - ENOMEM ;
goto out ;
}
sprintf ( devpath_string , " DEVPATH_OLD=%s " , devpath ) ;
envp [ 0 ] = devpath_string ;
envp [ 1 ] = NULL ;
error = sysfs_move_dir ( kobj , new_parent ) ;
if ( error )
goto out ;
old_parent = kobj - > parent ;
kobj - > parent = new_parent ;
2007-03-03 16:11:21 +03:00
new_parent = NULL ;
2006-11-20 19:07:51 +03:00
kobject_put ( old_parent ) ;
kobject_uevent_env ( kobj , KOBJ_MOVE , envp ) ;
out :
2007-03-03 16:11:21 +03:00
kobject_put ( new_parent ) ;
2006-11-20 19:07:51 +03:00
kobject_put ( kobj ) ;
kfree ( devpath_string ) ;
kfree ( devpath ) ;
return error ;
}
2005-04-17 02:20:36 +04:00
/**
* kobject_del - unlink kobject from hierarchy .
* @ kobj : object .
*/
void kobject_del ( struct kobject * kobj )
{
2007-01-18 23:23:51 +03:00
if ( ! kobj )
return ;
2005-04-17 02:20:36 +04:00
sysfs_remove_dir ( kobj ) ;
unlink ( kobj ) ;
}
/**
* kobject_unregister - remove object from hierarchy and decrement refcount .
* @ kobj : object going away .
*/
void kobject_unregister ( struct kobject * kobj )
{
2007-01-18 23:23:51 +03:00
if ( ! kobj )
return ;
2005-04-17 02:20:36 +04:00
pr_debug ( " kobject %s: unregistering \n " , kobject_name ( kobj ) ) ;
2005-11-16 11:00:00 +03:00
kobject_uevent ( kobj , KOBJ_REMOVE ) ;
2005-04-17 02:20:36 +04:00
kobject_del ( kobj ) ;
kobject_put ( kobj ) ;
}
/**
* kobject_get - increment refcount for object .
* @ kobj : object .
*/
struct kobject * kobject_get ( struct kobject * kobj )
{
if ( kobj )
kref_get ( & kobj - > kref ) ;
return kobj ;
}
2007-12-04 08:31:08 +03:00
/*
* kobject_cleanup - free kobject resources .
* @ kobj : object to cleanup
2005-04-17 02:20:36 +04:00
*/
2007-12-04 08:31:08 +03:00
static void kobject_cleanup ( struct kobject * kobj )
2005-04-17 02:20:36 +04:00
{
struct kobj_type * t = get_ktype ( kobj ) ;
struct kset * s = kobj - > kset ;
struct kobject * parent = kobj - > parent ;
2007-09-13 02:06:57 +04:00
const char * name = kobj - > k_name ;
2005-04-17 02:20:36 +04:00
pr_debug ( " kobject %s: cleaning up \n " , kobject_name ( kobj ) ) ;
2007-09-13 02:06:57 +04:00
if ( t & & t - > release ) {
2005-04-17 02:20:36 +04:00
t - > release ( kobj ) ;
2007-09-13 02:06:57 +04:00
/* If we have a release function, we can guess that this was
* not a statically allocated kobject , so we should be safe to
* free the name */
kfree ( name ) ;
}
2005-04-17 02:20:36 +04:00
if ( s )
kset_put ( s ) ;
2007-01-02 15:44:44 +03:00
kobject_put ( parent ) ;
2005-04-17 02:20:36 +04:00
}
static void kobject_release ( struct kref * kref )
{
kobject_cleanup ( container_of ( kref , struct kobject , kref ) ) ;
}
/**
* kobject_put - decrement refcount for object .
* @ kobj : object .
*
* Decrement the refcount , and if 0 , call kobject_cleanup ( ) .
*/
void kobject_put ( struct kobject * kobj )
{
if ( kobj )
kref_put ( & kobj - > kref , kobject_release ) ;
}
2007-11-06 00:16:15 +03:00
static void dynamic_kobj_release ( struct kobject * kobj )
2006-03-14 01:14:25 +03:00
{
2007-11-06 00:16:15 +03:00
pr_debug ( " %s: freeing %s \n " , __FUNCTION__ , kobject_name ( kobj ) ) ;
2006-03-14 01:14:25 +03:00
kfree ( kobj ) ;
}
2007-11-06 00:16:15 +03:00
static struct kobj_type dynamic_kobj_ktype = {
. release = dynamic_kobj_release ,
2006-03-14 01:14:25 +03:00
} ;
2007-11-06 09:24:43 +03:00
/**
2007-11-06 00:16:15 +03:00
* kobject_create - create a struct kobject dynamically
*
* This function creates a kobject structure dynamically and sets it up
* to be a " dynamic " kobject with a default release function set up .
*
* If the kobject was not able to be created , NULL will be returned .
2007-11-06 09:24:43 +03:00
* The kobject structure returned from here must be cleaned up with a
* call to kobject_put ( ) and not kfree ( ) , as kobject_init_ng ( ) has
* already been called on this structure .
2007-11-06 00:16:15 +03:00
*/
2007-11-06 09:24:43 +03:00
struct kobject * kobject_create ( void )
2007-11-06 00:16:15 +03:00
{
struct kobject * kobj ;
kobj = kzalloc ( sizeof ( * kobj ) , GFP_KERNEL ) ;
if ( ! kobj )
return NULL ;
kobject_init_ng ( kobj , & dynamic_kobj_ktype ) ;
return kobj ;
}
/**
* kobject_create_and_add - create a struct kobject dynamically and register it with sysfs
*
* @ name : the name for the kset
* @ parent : the parent kobject of this kobject , if any .
*
* This function creates a kset structure dynamically and registers it
* with sysfs . When you are finished with this structure , call
* kobject_unregister ( ) and the structure will be dynamically freed when
* it is no longer being used .
*
* If the kobject was not able to be created , NULL will be returned .
*/
struct kobject * kobject_create_and_add ( const char * name , struct kobject * parent )
{
struct kobject * kobj ;
int retval ;
kobj = kobject_create ( ) ;
if ( ! kobj )
return NULL ;
retval = kobject_add_ng ( kobj , parent , " %s " , name ) ;
if ( retval ) {
printk ( KERN_WARNING " %s: kobject_add error: %d \n " ,
__FUNCTION__ , retval ) ;
kobject_put ( kobj ) ;
kobj = NULL ;
}
return kobj ;
}
EXPORT_SYMBOL_GPL ( kobject_create_and_add ) ;
2005-04-17 02:20:36 +04:00
/**
* kset_init - initialize a kset for use
* @ k : kset
*/
void kset_init ( struct kset * k )
{
kobject_init ( & k - > kobj ) ;
INIT_LIST_HEAD ( & k - > list ) ;
spin_lock_init ( & k - > list_lock ) ;
}
2007-11-02 15:47:53 +03:00
/* default kobject attribute operations */
static ssize_t kobj_attr_show ( struct kobject * kobj , struct attribute * attr ,
char * buf )
{
struct kobj_attribute * kattr ;
ssize_t ret = - EIO ;
kattr = container_of ( attr , struct kobj_attribute , attr ) ;
if ( kattr - > show )
ret = kattr - > show ( kobj , kattr , buf ) ;
return ret ;
}
static ssize_t kobj_attr_store ( struct kobject * kobj , struct attribute * attr ,
const char * buf , size_t count )
{
struct kobj_attribute * kattr ;
ssize_t ret = - EIO ;
kattr = container_of ( attr , struct kobj_attribute , attr ) ;
if ( kattr - > store )
ret = kattr - > store ( kobj , kattr , buf , count ) ;
return ret ;
}
struct sysfs_ops kobj_sysfs_ops = {
. show = kobj_attr_show ,
. store = kobj_attr_store ,
} ;
2005-04-17 02:20:36 +04:00
/**
* kset_add - add a kset object to the hierarchy .
* @ k : kset .
*/
int kset_add ( struct kset * k )
{
return kobject_add ( & k - > kobj ) ;
}
/**
* kset_register - initialize and add a kset .
* @ k : kset .
*/
int kset_register ( struct kset * k )
{
2007-05-26 13:21:36 +04:00
int err ;
2007-01-18 23:23:51 +03:00
if ( ! k )
return - EINVAL ;
2007-05-26 13:21:36 +04:00
2005-04-17 02:20:36 +04:00
kset_init ( k ) ;
2007-05-26 13:21:36 +04:00
err = kset_add ( k ) ;
if ( err )
return err ;
kobject_uevent ( & k - > kobj , KOBJ_ADD ) ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
/**
* kset_unregister - remove a kset .
* @ k : kset .
*/
void kset_unregister ( struct kset * k )
{
2007-01-18 23:23:51 +03:00
if ( ! k )
return ;
2005-04-17 02:20:36 +04:00
kobject_unregister ( & k - > kobj ) ;
}
/**
* kset_find_obj - search for object in kset .
* @ kset : kset we ' re looking in .
* @ name : object ' s name .
*
* Lock kset via @ kset - > subsys , and iterate over @ kset - > list ,
* looking for a matching kobject . If matching object is found
* take a reference and return the object .
*/
struct kobject * kset_find_obj ( struct kset * kset , const char * name )
{
struct list_head * entry ;
struct kobject * ret = NULL ;
spin_lock ( & kset - > list_lock ) ;
list_for_each ( entry , & kset - > list ) {
struct kobject * k = to_kobj ( entry ) ;
if ( kobject_name ( k ) & & ! strcmp ( kobject_name ( k ) , name ) ) {
ret = kobject_get ( k ) ;
break ;
}
}
spin_unlock ( & kset - > list_lock ) ;
return ret ;
}
2007-04-14 00:15:19 +04:00
int subsystem_register ( struct kset * s )
2005-04-17 02:20:36 +04:00
{
2007-04-14 00:15:19 +04:00
return kset_register ( s ) ;
2005-04-17 02:20:36 +04:00
}
2007-04-14 00:15:19 +04:00
void subsystem_unregister ( struct kset * s )
2005-04-17 02:20:36 +04:00
{
2007-04-14 00:15:19 +04:00
kset_unregister ( s ) ;
2005-04-17 02:20:36 +04:00
}
/**
* subsystem_create_file - export sysfs attribute file .
* @ s : subsystem .
* @ a : subsystem attribute descriptor .
*/
2007-04-14 00:15:19 +04:00
int subsys_create_file ( struct kset * s , struct subsys_attribute * a )
2005-04-17 02:20:36 +04:00
{
int error = 0 ;
2007-01-18 23:23:51 +03:00
if ( ! s | | ! a )
return - EINVAL ;
2007-09-13 02:06:57 +04:00
if ( kset_get ( s ) ) {
2007-04-14 00:15:19 +04:00
error = sysfs_create_file ( & s - > kobj , & a - > attr ) ;
2007-09-13 02:06:57 +04:00
kset_put ( s ) ;
2005-04-17 02:20:36 +04:00
}
return error ;
}
2007-09-28 01:48:53 +04:00
static void kset_release ( struct kobject * kobj )
{
struct kset * kset = container_of ( kobj , struct kset , kobj ) ;
pr_debug ( " kset %s: now freed \n " , kobject_name ( kobj ) ) ;
kfree ( kset ) ;
}
static struct kobj_type kset_type = {
. release = kset_release ,
} ;
/**
* kset_create - create a struct kset dynamically
*
* @ name : the name for the kset
* @ uevent_ops : a struct kset_uevent_ops for the kset
* @ parent_kobj : the parent kobject of this kset , if any .
*
* This function creates a kset structure dynamically . This structure can
* then be registered with the system and show up in sysfs with a call to
* kset_register ( ) . When you are finished with this structure , if
* kset_register ( ) has been called , call kset_unregister ( ) and the
* structure will be dynamically freed when it is no longer being used .
*
* If the kset was not able to be created , NULL will be returned .
*/
static struct kset * kset_create ( const char * name ,
struct kset_uevent_ops * uevent_ops ,
struct kobject * parent_kobj )
{
struct kset * kset ;
kset = kzalloc ( sizeof ( * kset ) , GFP_KERNEL ) ;
if ( ! kset )
return NULL ;
kobject_set_name ( & kset - > kobj , name ) ;
kset - > uevent_ops = uevent_ops ;
kset - > kobj . parent = parent_kobj ;
/*
* The kobject of this kset will have a type of kset_type and belong to
* no kset itself . That way we can properly free it when it is
* finished being used .
*/
kset - > kobj . ktype = & kset_type ;
kset - > kobj . kset = NULL ;
return kset ;
}
/**
* kset_create_and_add - create a struct kset dynamically and add it to sysfs
*
* @ name : the name for the kset
* @ uevent_ops : a struct kset_uevent_ops for the kset
* @ parent_kobj : the parent kobject of this kset , if any .
*
* This function creates a kset structure dynamically and registers it
* with sysfs . When you are finished with this structure , call
* kset_unregister ( ) and the structure will be dynamically freed when it
* is no longer being used .
*
* If the kset was not able to be created , NULL will be returned .
*/
struct kset * kset_create_and_add ( const char * name ,
struct kset_uevent_ops * uevent_ops ,
struct kobject * parent_kobj )
{
struct kset * kset ;
int error ;
kset = kset_create ( name , uevent_ops , parent_kobj ) ;
if ( ! kset )
return NULL ;
error = kset_register ( kset ) ;
if ( error ) {
kfree ( kset ) ;
return NULL ;
}
return kset ;
}
EXPORT_SYMBOL_GPL ( kset_create_and_add ) ;
2005-04-17 02:20:36 +04:00
EXPORT_SYMBOL ( kobject_init ) ;
EXPORT_SYMBOL ( kobject_register ) ;
EXPORT_SYMBOL ( kobject_unregister ) ;
EXPORT_SYMBOL ( kobject_get ) ;
EXPORT_SYMBOL ( kobject_put ) ;
EXPORT_SYMBOL ( kobject_add ) ;
EXPORT_SYMBOL ( kobject_del ) ;
EXPORT_SYMBOL ( kset_register ) ;
EXPORT_SYMBOL ( kset_unregister ) ;
EXPORT_SYMBOL ( subsystem_register ) ;
EXPORT_SYMBOL ( subsystem_unregister ) ;
EXPORT_SYMBOL ( subsys_create_file ) ;