2017-11-07 17:30:05 +01:00
// SPDX-License-Identifier: GPL-2.0
2005-04-16 15:20:36 -07:00
/*
* kobject . c - library routines for handling generic kernel objects
*
* Copyright ( c ) 2002 - 2003 Patrick Mochel < mochel @ osdl . org >
2007-09-27 14:48:53 -07:00
* Copyright ( c ) 2006 - 2007 Greg Kroah - Hartman < greg @ kroah . com >
* Copyright ( c ) 2006 - 2007 Novell Inc .
2005-04-16 15:20:36 -07:00
*
2020-04-14 18:48:37 +02:00
* Please see the file Documentation / core - api / kobject . rst for critical information
2005-04-16 15:20:36 -07:00
* about using the kobject interface .
*/
# include <linux/kobject.h>
# include <linux/string.h>
2011-11-16 21:29:17 -05:00
# include <linux/export.h>
2005-04-16 15:20:36 -07:00
# include <linux/stat.h>
2005-10-30 15:03:48 -08:00
# include <linux/slab.h>
2013-12-05 17:37:51 -07:00
# include <linux/random.h>
2005-04-16 15:20:36 -07:00
2013-09-11 22:29:05 -04:00
/**
2019-05-02 12:31:40 +10:00
* kobject_namespace ( ) - Return @ kobj ' s namespace tag .
2013-09-11 22:29:05 -04:00
* @ kobj : kobject in question
*
* Returns namespace tag of @ kobj if its parent has namespace ops enabled
* and thus @ kobj should have a namespace tag associated with it . Returns
* % NULL otherwise .
*/
const void * kobject_namespace ( struct kobject * kobj )
{
const struct kobj_ns_type_operations * ns_ops = kobj_ns_ops ( kobj ) ;
if ( ! ns_ops | | ns_ops - > type = = KOBJ_NS_TYPE_NONE )
return NULL ;
2013-11-07 20:47:28 +09:00
return kobj - > ktype - > namespace ( kobj ) ;
2013-09-11 22:29:05 -04:00
}
2018-07-20 21:56:48 +00:00
/**
2019-05-02 12:31:40 +10:00
* kobject_get_ownership ( ) - Get sysfs ownership data for @ kobj .
2018-07-20 21:56:48 +00:00
* @ kobj : kobject in question
* @ uid : kernel user ID for sysfs objects
* @ gid : kernel group ID for sysfs objects
*
* Returns initial uid / gid pair that should be used when creating sysfs
* representation of given kobject . Normally used to adjust ownership of
* objects in a container .
*/
void kobject_get_ownership ( struct kobject * kobj , kuid_t * uid , kgid_t * gid )
{
* uid = GLOBAL_ROOT_UID ;
* gid = GLOBAL_ROOT_GID ;
if ( kobj - > ktype - > get_ownership )
kobj - > ktype - > get_ownership ( kobj , uid , gid ) ;
}
2008-01-24 21:59:04 -08:00
static int create_dir ( struct kobject * kobj )
2005-04-16 15:20:36 -07:00
{
2019-04-01 22:51:18 -04:00
const struct kobj_type * ktype = get_ktype ( kobj ) ;
2013-11-23 18:01:46 -05:00
const struct kobj_ns_type_operations * ops ;
2013-09-11 22:29:05 -04:00
int error ;
error = sysfs_create_dir_ns ( kobj , kobject_namespace ( kobj ) ) ;
2013-11-23 18:01:46 -05:00
if ( error )
return error ;
2019-04-01 22:51:18 -04:00
if ( ktype ) {
error = sysfs_create_groups ( kobj , ktype - > default_groups ) ;
if ( error ) {
sysfs_remove_dir ( kobj ) ;
return error ;
}
}
2013-09-18 17:15:36 -04:00
/*
* @ kobj - > sd may be deleted by an ancestor going away . Hold an
* extra reference so that it stays until @ kobj is gone .
*/
sysfs_get ( kobj - > sd ) ;
2013-11-23 18:01:46 -05:00
/*
* If @ kobj has ns_ops , its children need to be filtered based on
* their namespace tags . Enable namespace support on @ kobj - > sd .
*/
ops = kobj_child_ns_ops ( kobj ) ;
if ( ops ) {
BUG_ON ( ops - > type < = KOBJ_NS_TYPE_NONE ) ;
BUG_ON ( ops - > type > = KOBJ_NS_TYPES ) ;
BUG_ON ( ! kobj_ns_type_registered ( ops - > type ) ) ;
2014-02-07 13:32:07 -05:00
sysfs_enable_ns ( kobj - > sd ) ;
2013-11-23 18:01:46 -05:00
}
return 0 ;
2005-04-16 15:20:36 -07:00
}
static int get_kobj_path_length ( struct kobject * kobj )
{
int length = 1 ;
2008-01-24 21:59:04 -08:00
struct kobject * parent = kobj ;
2005-04-16 15:20:36 -07:00
2008-01-24 21:59:04 -08:00
/* walk up the ancestors until we hit the one pointing to the
2005-04-16 15:20:36 -07:00
* root .
* Add 1 to strlen for leading ' / ' of each level .
*/
do {
2006-01-12 20:02:00 -05:00
if ( kobject_name ( parent ) = = NULL )
return 0 ;
2005-04-16 15:20:36 -07: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 )
{
2008-01-24 21:59:04 -08:00
struct kobject * parent ;
2005-04-16 15:20:36 -07:00
- - length ;
for ( parent = kobj ; parent ; parent = parent - > parent ) {
int cur = strlen ( kobject_name ( parent ) ) ;
/* back up enough to print this name with '/' */
length - = cur ;
2018-07-01 13:57:16 -07:00
memcpy ( path + length , kobject_name ( parent ) , cur ) ;
2005-04-16 15:20:36 -07:00
* ( path + - - length ) = ' / ' ;
}
2007-11-28 23:49:41 -08:00
pr_debug ( " kobject: '%s' (%p): %s: path = '%s' \n " , kobject_name ( kobj ) ,
2008-04-30 00:55:08 -07:00
kobj , __func__ , path ) ;
2005-04-16 15:20:36 -07:00
}
/**
2019-05-02 12:31:39 +10:00
* kobject_get_path ( ) - Allocate memory and fill in the path for @ kobj .
2005-04-16 15:20:36 -07:00
* @ kobj : kobject in question , with which to build the path
* @ gfp_mask : the allocation type used to allocate the path
2007-02-10 01:45:59 -08:00
*
2019-05-02 12:31:39 +10:00
* Return : The newly allocated memory , caller must free with kfree ( ) .
2005-04-16 15:20:36 -07:00
*/
2005-10-21 03:18:50 -04:00
char * kobject_get_path ( struct kobject * kobj , gfp_t gfp_mask )
2005-04-16 15:20:36 -07:00
{
char * path ;
int len ;
len = get_kobj_path_length ( kobj ) ;
2006-01-12 20:02:00 -05:00
if ( len = = 0 )
return NULL ;
2006-12-06 20:38:51 -08:00
path = kzalloc ( len , gfp_mask ) ;
2005-04-16 15:20:36 -07:00
if ( ! path )
return NULL ;
fill_kobj_path ( kobj , path , len ) ;
return path ;
}
2006-10-11 01:43:58 -04:00
EXPORT_SYMBOL_GPL ( kobject_get_path ) ;
2005-04-16 15:20:36 -07:00
2007-12-19 01:40:42 +01:00
/* add the kobject to its kset's list */
static void kobj_kset_join ( struct kobject * kobj )
2005-04-16 15:20:36 -07:00
{
2007-12-19 01:40:42 +01:00
if ( ! kobj - > kset )
2007-01-18 12:23:51 -08:00
return ;
2007-12-19 01:40:42 +01:00
kset_get ( kobj - > kset ) ;
spin_lock ( & kobj - > kset - > list_lock ) ;
list_add_tail ( & kobj - > entry , & kobj - > kset - > list ) ;
spin_unlock ( & kobj - > kset - > list_lock ) ;
2005-04-16 15:20:36 -07:00
}
2007-12-19 01:40:42 +01:00
/* remove the kobject from its kset's list */
static void kobj_kset_leave ( struct kobject * kobj )
{
if ( ! kobj - > kset )
return ;
2005-04-16 15:20:36 -07:00
2007-12-19 01:40:42 +01:00
spin_lock ( & kobj - > kset - > list_lock ) ;
list_del_init ( & kobj - > entry ) ;
spin_unlock ( & kobj - > kset - > list_lock ) ;
kset_put ( kobj - > kset ) ;
}
2005-04-16 15:20:36 -07:00
2008-01-24 21:59:04 -08:00
static void kobject_init_internal ( struct kobject * kobj )
2005-04-16 15:20:36 -07:00
{
2007-12-19 01:40:42 +01:00
if ( ! kobj )
return ;
kref_init ( & kobj - > kref ) ;
INIT_LIST_HEAD ( & kobj - > entry ) ;
2008-02-26 09:36:38 -08:00
kobj - > state_in_sysfs = 0 ;
kobj - > state_add_uevent_sent = 0 ;
kobj - > state_remove_uevent_sent = 0 ;
kobj - > state_initialized = 1 ;
2005-04-16 15:20:36 -07:00
}
2007-12-19 01:40:42 +01:00
2007-12-17 23:05:35 -07:00
static int kobject_add_internal ( struct kobject * kobj )
2005-04-16 15:20:36 -07:00
{
int error = 0 ;
2008-01-24 21:59:04 -08:00
struct kobject * parent ;
2005-04-16 15:20:36 -07:00
2007-12-19 01:40:42 +01:00
if ( ! kobj )
2005-04-16 15:20:36 -07:00
return - ENOENT ;
2007-12-19 01:40:42 +01:00
2007-12-20 02:09:39 +01:00
if ( ! kobj - > name | | ! kobj - > name [ 0 ] ) {
2018-03-15 15:23:43 +02:00
WARN ( 1 ,
" kobject: (%p): attempted to be registered with empty name! \n " ,
kobj ) ;
2006-01-20 14:08:59 -08:00
return - EINVAL ;
}
2005-04-16 15:20:36 -07:00
2007-12-19 01:40:42 +01:00
parent = kobject_get ( kobj - > parent ) ;
2005-04-16 15:20:36 -07:00
2007-12-19 01:40:42 +01:00
/* join kset if set, use it as parent if we do not already have one */
2005-04-16 15:20:36 -07:00
if ( kobj - > kset ) {
2007-12-19 01:40:42 +01:00
if ( ! parent )
2005-04-16 15:20:36 -07:00
parent = kobject_get ( & kobj - > kset - > kobj ) ;
2007-12-19 01:40:42 +01:00
kobj_kset_join ( kobj ) ;
2007-03-10 14:00:10 +03:00
kobj - > parent = parent ;
2005-04-16 15:20:36 -07:00
}
2007-12-19 01:40:42 +01:00
pr_debug ( " kobject: '%s' (%p): %s: parent: '%s', set: '%s' \n " ,
2008-04-30 00:55:08 -07:00
kobject_name ( kobj ) , kobj , __func__ ,
2007-12-19 01:40:42 +01:00
parent ? kobject_name ( parent ) : " <NULL> " ,
2008-01-24 21:59:04 -08:00
kobj - > kset ? kobject_name ( & kobj - > kset - > kobj ) : " <NULL> " ) ;
2007-12-19 01:40:42 +01:00
2007-07-31 19:15:08 +09:00
error = create_dir ( kobj ) ;
2005-04-16 15:20:36 -07:00
if ( error ) {
2007-12-19 01:40:42 +01:00
kobj_kset_leave ( kobj ) ;
kobject_put ( parent ) ;
kobj - > parent = NULL ;
2006-03-20 13:17:13 -08:00
/* be noisy on error issues */
if ( error = = - EEXIST )
2018-04-11 17:22:43 +02:00
pr_err ( " %s failed for %s with -EEXIST, don't try to register things with the same name in the same directory. \n " ,
__func__ , kobject_name ( kobj ) ) ;
2006-03-20 13:17:13 -08:00
else
2018-04-11 17:22:43 +02:00
pr_err ( " %s failed for %s (error: %d parent: %s) \n " ,
__func__ , kobject_name ( kobj ) , error ,
parent ? kobject_name ( parent ) : " 'none' " ) ;
2007-12-19 01:40:42 +01:00
} else
kobj - > state_in_sysfs = 1 ;
2005-04-16 15:20:36 -07:00
return error ;
}
2007-11-29 18:32:47 -05:00
/**
2019-05-02 12:31:40 +10:00
* kobject_set_name_vargs ( ) - Set the name of a kobject .
2007-11-29 18:32:47 -05:00
* @ kobj : struct kobject to set the name of
* @ fmt : format string used to build the name
* @ vargs : vargs to format the string .
*/
2009-01-25 15:17:37 +01:00
int kobject_set_name_vargs ( struct kobject * kobj , const char * fmt ,
2007-11-29 18:32:47 -05:00
va_list vargs )
{
2015-11-06 16:31:23 -08:00
const char * s ;
2007-11-29 18:32:47 -05:00
2009-04-18 15:05:45 -07:00
if ( kobj - > name & & ! fmt )
return 0 ;
2015-11-06 16:31:23 -08:00
s = kvasprintf_const ( GFP_KERNEL , fmt , vargs ) ;
2015-06-25 15:02:30 -07:00
if ( ! s )
2008-04-30 02:06:29 +02:00
return - ENOMEM ;
2007-11-29 18:32:47 -05:00
2015-11-06 16:31:23 -08:00
/*
* ewww . . . some of these buggers have ' / ' in the name . . . If
* that ' s the case , we need to make sure we have an actual
* allocated copy to modify , since kvasprintf_const may have
* returned something from . rodata .
*/
if ( strchr ( s , ' / ' ) ) {
char * t ;
t = kstrdup ( s , GFP_KERNEL ) ;
kfree_const ( s ) ;
if ( ! t )
return - ENOMEM ;
strreplace ( t , ' / ' , ' ! ' ) ;
s = t ;
}
kfree_const ( kobj - > name ) ;
2015-06-25 15:02:30 -07:00
kobj - > name = s ;
2008-05-06 22:24:04 +02:00
2007-11-29 18:32:47 -05:00
return 0 ;
}
2005-04-16 15:20:36 -07:00
/**
2019-05-02 12:31:40 +10:00
* kobject_set_name ( ) - Set the name of a kobject .
2007-11-29 18:32:47 -05:00
* @ kobj : struct kobject to set the name of
2007-12-04 14:45:47 +08:00
* @ fmt : format string used to build the name
2005-04-16 15:20:36 -07:00
*
2007-12-04 14:45:47 +08: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-16 15:20:36 -07:00
*/
2007-11-29 18:32:47 -05:00
int kobject_set_name ( struct kobject * kobj , const char * fmt , . . . )
2005-04-16 15:20:36 -07:00
{
2008-04-30 02:06:29 +02:00
va_list vargs ;
2007-11-29 18:32:47 -05:00
int retval ;
2005-04-16 15:20:36 -07:00
2008-04-30 02:06:29 +02:00
va_start ( vargs , fmt ) ;
retval = kobject_set_name_vargs ( kobj , fmt , vargs ) ;
va_end ( vargs ) ;
2005-04-16 15:20:36 -07:00
2007-11-29 18:32:47 -05:00
return retval ;
2005-04-16 15:20:36 -07:00
}
EXPORT_SYMBOL ( kobject_set_name ) ;
2007-12-03 21:31:08 -08:00
/**
2019-05-02 12:31:40 +10:00
* kobject_init ( ) - Initialize a kobject structure .
2007-12-03 21:31:08 -08:00
* @ 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 .
*/
2021-12-24 23:13:45 +00:00
void kobject_init ( struct kobject * kobj , const struct kobj_type * ktype )
2007-12-03 21:31:08 -08:00
{
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 ;
}
2007-12-19 01:40:42 +01:00
if ( kobj - > state_initialized ) {
2007-12-03 21:31:08 -08:00
/* do not error out as sometimes we can recover */
2018-03-15 15:23:43 +02:00
pr_err ( " kobject (%p): tried to init an initialized object, something is seriously wrong. \n " ,
kobj ) ;
2007-12-03 21:31:08 -08:00
dump_stack ( ) ;
}
2008-02-26 09:36:38 -08:00
kobject_init_internal ( kobj ) ;
2007-12-03 21:31:08 -08:00
kobj - > ktype = ktype ;
return ;
error :
2018-03-15 15:23:43 +02:00
pr_err ( " kobject (%p): %s \n " , kobj , err_str ) ;
2007-12-03 21:31:08 -08:00
dump_stack ( ) ;
}
2007-12-17 23:05:35 -07:00
EXPORT_SYMBOL ( kobject_init ) ;
2007-12-03 21:31:08 -08:00
2015-07-17 16:23:42 -07:00
static __printf ( 3 , 0 ) int kobject_add_varg ( struct kobject * kobj ,
struct kobject * parent ,
const char * fmt , va_list vargs )
2007-12-03 21:31:08 -08:00
{
int retval ;
2008-04-30 02:06:29 +02:00
retval = kobject_set_name_vargs ( kobj , fmt , vargs ) ;
2007-12-03 21:31:08 -08:00
if ( retval ) {
2018-03-15 15:23:43 +02:00
pr_err ( " kobject: can not set name properly! \n " ) ;
2007-12-03 21:31:08 -08:00
return retval ;
}
kobj - > parent = parent ;
2007-12-17 23:05:35 -07:00
return kobject_add_internal ( kobj ) ;
2007-12-03 21:31:08 -08:00
}
/**
2019-05-02 12:31:40 +10:00
* kobject_add ( ) - The main kobject add function .
2007-12-03 21:31:08 -08:00
* @ 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
2014-01-04 14:20:04 +01:00
* kobject associated with the kset assigned to this kobject . If no kset
2007-12-03 21:31:08 -08:00
* is assigned to the kobject , then the kobject will be located in the
* root of the sysfs tree .
*
2007-12-19 01:40:42 +01:00
* Note , no " add " uevent will be created with this call , the caller should set
2007-12-03 21:31:08 -08:00
* 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 .
2019-04-28 10:48:10 +10:00
*
* Return : If this function returns an error , kobject_put ( ) must be
* called to properly clean up the memory associated with the
* object . Under no instance should the kobject that is passed
* to this function be directly freed with a call to kfree ( ) ,
* that can leak memory .
*
2019-05-02 12:22:24 +02:00
* If this function returns success , kobject_put ( ) must also be called
* in order to properly clean up the memory associated with the object .
*
* In short , once this function is called , kobject_put ( ) MUST be called
* when the use of the object is finished in order to properly free
* everything .
2007-12-03 21:31:08 -08:00
*/
2007-12-17 23:05:35 -07:00
int kobject_add ( struct kobject * kobj , struct kobject * parent ,
const char * fmt , . . . )
2007-12-03 21:31:08 -08:00
{
va_list args ;
int retval ;
if ( ! kobj )
return - EINVAL ;
2007-12-19 01:40:42 +01:00
if ( ! kobj - > state_initialized ) {
2018-03-15 15:23:43 +02:00
pr_err ( " kobject '%s' (%p): tried to add an uninitialized object, something is seriously wrong. \n " ,
2007-12-19 01:40:42 +01:00
kobject_name ( kobj ) , kobj ) ;
dump_stack ( ) ;
return - EINVAL ;
}
2007-12-03 21:31:08 -08:00
va_start ( args , fmt ) ;
retval = kobject_add_varg ( kobj , parent , fmt , args ) ;
va_end ( args ) ;
return retval ;
}
2007-12-17 23:05:35 -07:00
EXPORT_SYMBOL ( kobject_add ) ;
2007-12-03 21:31:08 -08:00
2007-12-03 21:31:08 -08:00
/**
2019-05-02 12:31:40 +10:00
* kobject_init_and_add ( ) - Initialize a kobject structure and add it to
* the kobject hierarchy .
2007-12-03 21:31:08 -08:00
* @ 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 .
*
2019-04-28 09:56:52 +10:00
* This function combines the call to kobject_init ( ) and kobject_add ( ) .
*
* If this function returns an error , kobject_put ( ) must be called to
* properly clean up the memory associated with the object . This is the
* same type of error handling after a call to kobject_add ( ) and kobject
* lifetime rules are the same here .
2007-12-03 21:31:08 -08:00
*/
2021-12-24 23:13:45 +00:00
int kobject_init_and_add ( struct kobject * kobj , const struct kobj_type * ktype ,
2007-12-03 21:31:08 -08:00
struct kobject * parent , const char * fmt , . . . )
{
va_list args ;
int retval ;
2007-12-17 23:05:35 -07:00
kobject_init ( kobj , ktype ) ;
2007-12-03 21:31:08 -08:00
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-16 15:20:36 -07:00
/**
2019-05-02 12:31:40 +10:00
* kobject_rename ( ) - Change the name of an object .
2008-01-24 21:59:04 -08:00
* @ kobj : object in question .
* @ new_name : object ' s new name
2008-05-08 14:41:00 -07:00
*
* It is the responsibility of the caller to provide mutual
* exclusion between two different calls of kobject_rename
* on the same kobject and to ensure that new_name is valid and
* won ' t conflict with other kobjects .
2005-04-16 15:20:36 -07:00
*/
2008-01-24 21:59:04 -08:00
int kobject_rename ( struct kobject * kobj , const char * new_name )
2005-04-16 15:20:36 -07:00
{
int error = 0 ;
2007-03-07 10:49:30 -08:00
const char * devpath = NULL ;
2008-07-03 18:05:28 -07:00
const char * dup_name = NULL , * name ;
2007-03-07 10:49:30 -08:00
char * devpath_string = NULL ;
char * envp [ 2 ] ;
2005-04-16 15:20:36 -07:00
kobj = kobject_get ( kobj ) ;
if ( ! kobj )
return - EINVAL ;
2019-06-03 16:08:10 +08:00
if ( ! kobj - > parent ) {
kobject_put ( kobj ) ;
2007-01-24 12:35:52 -07:00
return - EINVAL ;
2019-06-03 16:08:10 +08:00
}
2007-03-07 10:49:30 -08: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 ;
2015-11-06 16:31:23 -08:00
name = dup_name = kstrdup_const ( new_name , GFP_KERNEL ) ;
2008-07-03 18:05:28 -07:00
if ( ! name ) {
error = - ENOMEM ;
goto out ;
}
2013-09-11 22:29:05 -04:00
error = sysfs_rename_dir_ns ( kobj , new_name , kobject_namespace ( kobj ) ) ;
2008-07-03 18:05:28 -07:00
if ( error )
goto out ;
/* Install the new kobject name */
dup_name = kobj - > name ;
kobj - > name = name ;
2007-03-07 10:49:30 -08: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 . */
2008-07-03 18:05:28 -07:00
kobject_uevent_env ( kobj , KOBJ_MOVE , envp ) ;
2007-03-07 10:49:30 -08:00
out :
2015-11-06 16:31:23 -08:00
kfree_const ( dup_name ) ;
2007-03-07 10:49:30 -08:00
kfree ( devpath_string ) ;
kfree ( devpath ) ;
2007-01-24 12:35:52 -07:00
kobject_put ( kobj ) ;
return error ;
}
2008-06-10 15:30:42 -06:00
EXPORT_SYMBOL_GPL ( kobject_rename ) ;
2007-01-24 12:35:52 -07:00
2006-11-20 17:07:51 +01:00
/**
2019-05-02 12:31:40 +10:00
* kobject_move ( ) - Move object to another parent .
2008-01-24 21:59:04 -08:00
* @ kobj : object in question .
* @ new_parent : object ' s new parent ( can be NULL )
2006-11-20 17:07:51 +01: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 20:16:44 +01:00
if ( kobj - > kset )
new_parent = kobject_get ( & kobj - > kset - > kobj ) ;
2006-11-20 17:07:51 +01:00
}
2013-09-11 22:29:05 -04:00
2006-11-20 17:07:51 +01: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 ;
2013-09-11 22:29:05 -04:00
error = sysfs_move_dir_ns ( kobj , new_parent , kobject_namespace ( kobj ) ) ;
2006-11-20 17:07:51 +01:00
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 17:07:51 +01: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 17:07:51 +01:00
kobject_put ( kobj ) ;
kfree ( devpath_string ) ;
kfree ( devpath ) ;
return error ;
}
2015-02-12 07:03:37 +08:00
EXPORT_SYMBOL_GPL ( kobject_move ) ;
2006-11-20 17:07:51 +01:00
2020-06-04 19:46:46 +02:00
static void __kobject_del ( struct kobject * kobj )
2005-04-16 15:20:36 -07:00
{
2013-12-11 14:11:53 -05:00
struct kernfs_node * sd ;
2019-05-01 13:43:17 +01:00
const struct kobj_type * ktype ;
2013-09-18 17:15:36 -04:00
sd = kobj - > sd ;
2019-05-01 13:43:17 +01:00
ktype = get_ktype ( kobj ) ;
2019-04-01 22:51:18 -04:00
if ( ktype )
sysfs_remove_groups ( kobj , ktype - > default_groups ) ;
2020-05-24 17:30:41 +02:00
/* send "remove" if the caller did not do it but sent "add" */
if ( kobj - > state_add_uevent_sent & & ! kobj - > state_remove_uevent_sent ) {
pr_debug ( " kobject: '%s' (%p): auto cleanup 'remove' event \n " ,
kobject_name ( kobj ) , kobj ) ;
kobject_uevent ( kobj , KOBJ_REMOVE ) ;
}
2005-04-16 15:20:36 -07:00
sysfs_remove_dir ( kobj ) ;
2013-09-18 17:15:36 -04:00
sysfs_put ( sd ) ;
2007-12-19 01:40:42 +01:00
kobj - > state_in_sysfs = 0 ;
kobj_kset_leave ( kobj ) ;
kobj - > parent = NULL ;
2005-04-16 15:20:36 -07:00
}
2020-06-04 19:46:46 +02:00
/**
* kobject_del ( ) - Unlink kobject from hierarchy .
* @ kobj : object .
*
* This is the function that should be called to delete an object
* successfully added via kobject_add ( ) .
*/
void kobject_del ( struct kobject * kobj )
{
2020-08-03 11:27:06 +03:00
struct kobject * parent ;
if ( ! kobj )
return ;
2020-06-04 19:46:46 +02:00
2020-08-03 11:27:06 +03:00
parent = kobj - > parent ;
2020-06-04 19:46:46 +02:00
__kobject_del ( kobj ) ;
kobject_put ( parent ) ;
}
2015-08-10 15:51:43 -04:00
EXPORT_SYMBOL ( kobject_del ) ;
2005-04-16 15:20:36 -07:00
/**
2019-05-02 12:31:40 +10:00
* kobject_get ( ) - Increment refcount for object .
2008-01-24 21:59:04 -08:00
* @ kobj : object .
2005-04-16 15:20:36 -07:00
*/
2008-01-24 21:59:04 -08:00
struct kobject * kobject_get ( struct kobject * kobj )
2005-04-16 15:20:36 -07:00
{
2015-03-12 13:04:16 +09:00
if ( kobj ) {
if ( ! kobj - > state_initialized )
2018-03-15 15:23:43 +02:00
WARN ( 1 , KERN_WARNING
" kobject: '%s' (%p): is not initialized, yet kobject_get() is being called. \n " ,
kobject_name ( kobj ) , kobj ) ;
2005-04-16 15:20:36 -07:00
kref_get ( & kobj - > kref ) ;
2015-03-12 13:04:16 +09:00
}
2005-04-16 15:20:36 -07:00
return kobj ;
}
2015-08-10 15:51:43 -04:00
EXPORT_SYMBOL ( kobject_get ) ;
2005-04-16 15:20:36 -07:00
2017-03-23 01:37:01 +01:00
struct kobject * __must_check kobject_get_unless_zero ( struct kobject * kobj )
2013-04-13 15:15:30 -07:00
{
2017-03-23 01:37:01 +01:00
if ( ! kobj )
return NULL ;
2013-04-13 15:15:30 -07:00
if ( ! kref_get_unless_zero ( & kobj - > kref ) )
kobj = NULL ;
return kobj ;
}
2017-03-23 01:37:01 +01:00
EXPORT_SYMBOL ( kobject_get_unless_zero ) ;
2013-04-13 15:15:30 -07:00
2007-12-03 21:31:08 -08:00
/*
* kobject_cleanup - free kobject resources .
* @ kobj : object to cleanup
2005-04-16 15:20:36 -07:00
*/
2007-12-03 21:31:08 -08:00
static void kobject_cleanup ( struct kobject * kobj )
2005-04-16 15:20:36 -07:00
{
2020-06-04 19:46:46 +02:00
struct kobject * parent = kobj - > parent ;
2021-12-24 23:13:45 +00:00
const struct kobj_type * t = get_ktype ( kobj ) ;
2007-12-20 02:09:39 +01:00
const char * name = kobj - > name ;
2005-04-16 15:20:36 -07:00
2013-06-27 15:06:14 +01:00
pr_debug ( " kobject: '%s' (%p): %s, parent %p \n " ,
kobject_name ( kobj ) , kobj , __func__ , kobj - > parent ) ;
2007-12-19 01:40:42 +01:00
if ( t & & ! t - > release )
2020-04-14 18:48:37 +02:00
pr_debug ( " kobject: '%s' (%p): does not have a release() function, it is broken and must be fixed. See Documentation/core-api/kobject.rst. \n " ,
2007-12-19 01:40:42 +01:00
kobject_name ( kobj ) , kobj ) ;
/* remove from sysfs if the caller did not do it */
if ( kobj - > state_in_sysfs ) {
pr_debug ( " kobject: '%s' (%p): auto cleanup kobject_del \n " ,
kobject_name ( kobj ) , kobj ) ;
2020-06-04 19:46:46 +02:00
__kobject_del ( kobj ) ;
} else {
/* avoid dropping the parent reference unnecessarily */
parent = NULL ;
2007-12-19 01:40:42 +01:00
}
2007-09-12 15:06:57 -07:00
if ( t & & t - > release ) {
2007-12-19 01:40:42 +01:00
pr_debug ( " kobject: '%s' (%p): calling ktype release \n " ,
kobject_name ( kobj ) , kobj ) ;
2005-04-16 15:20:36 -07:00
t - > release ( kobj ) ;
2007-12-19 01:40:42 +01:00
}
/* free name if we allocated it */
2007-12-20 02:09:39 +01:00
if ( name ) {
2007-12-19 01:40:42 +01:00
pr_debug ( " kobject: '%s': free name \n " , name ) ;
2015-11-06 16:31:23 -08:00
kfree_const ( name ) ;
2007-09-12 15:06:57 -07:00
}
2020-06-04 19:46:46 +02:00
kobject_put ( parent ) ;
2005-04-16 15:20:36 -07:00
}
2013-06-27 15:06:14 +01:00
# ifdef CONFIG_DEBUG_KOBJECT_RELEASE
static void kobject_delayed_cleanup ( struct work_struct * work )
{
kobject_cleanup ( container_of ( to_delayed_work ( work ) ,
struct kobject , release ) ) ;
}
# endif
2005-04-16 15:20:36 -07:00
static void kobject_release ( struct kref * kref )
{
2013-06-27 15:06:14 +01:00
struct kobject * kobj = container_of ( kref , struct kobject , kref ) ;
# ifdef CONFIG_DEBUG_KOBJECT_RELEASE
2013-12-05 17:37:51 -07:00
unsigned long delay = HZ + HZ * ( get_random_int ( ) & 0x3 ) ;
pr_info ( " kobject: '%s' (%p): %s, parent %p (delayed %ld) \n " ,
kobject_name ( kobj ) , kobj , __func__ , kobj - > parent , delay ) ;
2013-06-27 15:06:14 +01:00
INIT_DELAYED_WORK ( & kobj - > release , kobject_delayed_cleanup ) ;
2013-12-05 17:37:51 -07:00
schedule_delayed_work ( & kobj - > release , delay ) ;
2013-06-27 15:06:14 +01:00
# else
kobject_cleanup ( kobj ) ;
# endif
2005-04-16 15:20:36 -07:00
}
/**
2019-05-02 12:31:40 +10:00
* kobject_put ( ) - Decrement refcount for object .
2008-01-24 21:59:04 -08:00
* @ kobj : object .
2005-04-16 15:20:36 -07:00
*
2008-01-24 21:59:04 -08:00
* Decrement the refcount , and if 0 , call kobject_cleanup ( ) .
2005-04-16 15:20:36 -07:00
*/
2008-01-24 21:59:04 -08:00
void kobject_put ( struct kobject * kobj )
2005-04-16 15:20:36 -07:00
{
2008-02-26 09:36:38 -08:00
if ( kobj ) {
2008-07-25 01:45:55 -07:00
if ( ! kobj - > state_initialized )
2018-03-15 15:23:43 +02:00
WARN ( 1 , KERN_WARNING
" kobject: '%s' (%p): is not initialized, yet kobject_put() is being called. \n " ,
kobject_name ( kobj ) , kobj ) ;
2005-04-16 15:20:36 -07:00
kref_put ( & kobj - > kref , kobject_release ) ;
2008-02-26 09:36:38 -08:00
}
2005-04-16 15:20:36 -07:00
}
2015-08-10 15:51:43 -04:00
EXPORT_SYMBOL ( kobject_put ) ;
2005-04-16 15:20:36 -07:00
2007-11-05 13:16:15 -08:00
static void dynamic_kobj_release ( struct kobject * kobj )
2006-03-13 17:14:25 -05:00
{
2008-04-30 00:55:08 -07:00
pr_debug ( " kobject: (%p): %s \n " , kobj , __func__ ) ;
2006-03-13 17:14:25 -05:00
kfree ( kobj ) ;
}
2007-11-05 13:16:15 -08:00
static struct kobj_type dynamic_kobj_ktype = {
2007-11-02 13:47:53 +01:00
. release = dynamic_kobj_release ,
. sysfs_ops = & kobj_sysfs_ops ,
2006-03-13 17:14:25 -05:00
} ;
2007-11-05 22:24:43 -08:00
/**
2019-05-02 12:31:40 +10:00
* kobject_create ( ) - Create a struct kobject dynamically .
2007-11-05 13:16:15 -08:00
*
* 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-05 22:24:43 -08:00
* The kobject structure returned from here must be cleaned up with a
2007-12-17 23:05:35 -07:00
* call to kobject_put ( ) and not kfree ( ) , as kobject_init ( ) has
2007-11-05 22:24:43 -08:00
* already been called on this structure .
2007-11-05 13:16:15 -08:00
*/
2021-08-31 17:30:44 +08:00
static struct kobject * kobject_create ( void )
2007-11-05 13:16:15 -08:00
{
struct kobject * kobj ;
kobj = kzalloc ( sizeof ( * kobj ) , GFP_KERNEL ) ;
if ( ! kobj )
return NULL ;
2007-12-17 23:05:35 -07:00
kobject_init ( kobj , & dynamic_kobj_ktype ) ;
2007-11-05 13:16:15 -08:00
return kobj ;
}
/**
2019-05-02 12:31:40 +10:00
* kobject_create_and_add ( ) - Create a struct kobject dynamically and
* register it with sysfs .
2012-05-07 10:48:25 +08:00
* @ name : the name for the kobject
2007-11-05 13:16:15 -08:00
* @ parent : the parent kobject of this kobject , if any .
*
2008-01-28 16:58:00 +08:00
* This function creates a kobject structure dynamically and registers it
2007-11-05 13:16:15 -08:00
* with sysfs . When you are finished with this structure , call
2007-12-20 08:13:05 -08:00
* kobject_put ( ) and the structure will be dynamically freed when
2007-11-05 13:16:15 -08:00
* 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 ;
2007-12-17 23:05:35 -07:00
retval = kobject_add ( kobj , parent , " %s " , name ) ;
2007-11-05 13:16:15 -08:00
if ( retval ) {
2018-03-15 15:23:43 +02:00
pr_warn ( " %s: kobject_add error: %d \n " , __func__ , retval ) ;
2007-11-05 13:16:15 -08:00
kobject_put ( kobj ) ;
kobj = NULL ;
}
return kobj ;
}
EXPORT_SYMBOL_GPL ( kobject_create_and_add ) ;
2005-04-16 15:20:36 -07:00
/**
2019-05-02 12:31:40 +10:00
* kset_init ( ) - Initialize a kset for use .
2008-01-24 21:59:04 -08:00
* @ k : kset
2005-04-16 15:20:36 -07:00
*/
2008-01-24 21:59:04 -08:00
void kset_init ( struct kset * k )
2005-04-16 15:20:36 -07:00
{
2007-12-17 23:05:35 -07:00
kobject_init_internal ( & k - > kobj ) ;
2005-04-16 15:20:36 -07:00
INIT_LIST_HEAD ( & k - > list ) ;
spin_lock_init ( & k - > list_lock ) ;
}
2007-11-02 13:47:53 +01: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 ;
}
2010-01-19 02:58:23 +01:00
const struct sysfs_ops kobj_sysfs_ops = {
2007-11-02 13:47:53 +01:00
. show = kobj_attr_show ,
. store = kobj_attr_store ,
} ;
2013-11-01 13:06:56 -04:00
EXPORT_SYMBOL_GPL ( kobj_sysfs_ops ) ;
2005-04-16 15:20:36 -07:00
/**
2019-05-02 12:31:40 +10:00
* kset_register ( ) - Initialize and add a kset .
2008-01-24 21:59:04 -08:00
* @ k : kset .
2005-04-16 15:20:36 -07:00
*/
2008-01-24 21:59:04 -08:00
int kset_register ( struct kset * k )
2005-04-16 15:20:36 -07:00
{
2007-05-26 11:21:36 +02:00
int err ;
2007-01-18 12:23:51 -08:00
if ( ! k )
return - EINVAL ;
2007-05-26 11:21:36 +02:00
2005-04-16 15:20:36 -07:00
kset_init ( k ) ;
2002-04-09 12:14:34 -07:00
err = kobject_add_internal ( & k - > kobj ) ;
2007-05-26 11:21:36 +02:00
if ( err )
return err ;
kobject_uevent ( & k - > kobj , KOBJ_ADD ) ;
return 0 ;
2005-04-16 15:20:36 -07:00
}
2015-08-10 15:51:43 -04:00
EXPORT_SYMBOL ( kset_register ) ;
2005-04-16 15:20:36 -07:00
/**
2019-05-02 12:31:40 +10:00
* kset_unregister ( ) - Remove a kset .
2008-01-24 21:59:04 -08:00
* @ k : kset .
2005-04-16 15:20:36 -07:00
*/
2008-01-24 21:59:04 -08:00
void kset_unregister ( struct kset * k )
2005-04-16 15:20:36 -07:00
{
2007-01-18 12:23:51 -08:00
if ( ! k )
return ;
2013-12-05 17:38:00 -07:00
kobject_del ( & k - > kobj ) ;
2007-12-20 08:13:05 -08:00
kobject_put ( & k - > kobj ) ;
2005-04-16 15:20:36 -07:00
}
2015-08-10 15:51:43 -04:00
EXPORT_SYMBOL ( kset_unregister ) ;
2005-04-16 15:20:36 -07:00
/**
2019-05-02 12:31:40 +10:00
* kset_find_obj ( ) - Search for object in kset .
2008-01-24 21:59:04 -08:00
* @ kset : kset we ' re looking in .
* @ name : object ' s name .
2005-04-16 15:20:36 -07:00
*
2008-01-24 21:59:04 -08:00
* 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 .
2005-04-16 15:20:36 -07:00
*/
2008-01-24 21:59:04 -08:00
struct kobject * kset_find_obj ( struct kset * kset , const char * name )
2005-04-16 15:20:36 -07:00
{
2008-03-27 01:13:34 -04:00
struct kobject * k ;
2008-01-24 21:59:04 -08:00
struct kobject * ret = NULL ;
2005-04-16 15:20:36 -07:00
spin_lock ( & kset - > list_lock ) ;
2010-09-29 14:00:54 -05:00
2008-03-27 01:13:34 -04:00
list_for_each_entry ( k , & kset - > list , entry ) {
2008-01-24 21:59:04 -08:00
if ( kobject_name ( k ) & & ! strcmp ( kobject_name ( k ) , name ) ) {
2013-04-13 15:15:30 -07:00
ret = kobject_get_unless_zero ( k ) ;
2005-04-16 15:20:36 -07:00
break ;
}
}
2010-09-29 14:00:54 -05:00
2005-04-16 15:20:36 -07:00
spin_unlock ( & kset - > list_lock ) ;
return ret ;
}
2016-01-28 09:23:12 -05:00
EXPORT_SYMBOL_GPL ( kset_find_obj ) ;
2005-04-16 15:20:36 -07:00
2007-09-27 14:48:53 -07:00
static void kset_release ( struct kobject * kobj )
{
struct kset * kset = container_of ( kobj , struct kset , kobj ) ;
2007-11-28 23:49:41 -08:00
pr_debug ( " kobject: '%s' (%p): %s \n " ,
2008-04-30 00:55:08 -07:00
kobject_name ( kobj ) , kobj , __func__ ) ;
2007-09-27 14:48:53 -07:00
kfree ( kset ) ;
}
2019-01-08 23:36:14 -08:00
static void kset_get_ownership ( struct kobject * kobj , kuid_t * uid , kgid_t * gid )
2018-07-20 21:56:49 +00:00
{
if ( kobj - > parent )
kobject_get_ownership ( kobj - > parent , uid , gid ) ;
}
2007-11-02 13:47:53 +01:00
static struct kobj_type kset_ktype = {
. sysfs_ops = & kobj_sysfs_ops ,
2018-07-20 21:56:49 +00:00
. release = kset_release ,
. get_ownership = kset_get_ownership ,
2007-09-27 14:48:53 -07:00
} ;
/**
2019-05-02 12:31:40 +10:00
* kset_create ( ) - Create a struct kset dynamically .
2007-09-27 14:48:53 -07:00
*
* @ 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 ,
2009-12-31 14:52:51 +01:00
const struct kset_uevent_ops * uevent_ops ,
2007-09-27 14:48:53 -07:00
struct kobject * parent_kobj )
{
struct kset * kset ;
2009-05-11 14:17:45 +08:00
int retval ;
2007-09-27 14:48:53 -07:00
kset = kzalloc ( sizeof ( * kset ) , GFP_KERNEL ) ;
if ( ! kset )
return NULL ;
2013-06-06 13:52:19 -07:00
retval = kobject_set_name ( & kset - > kobj , " %s " , name ) ;
2009-05-11 14:17:45 +08:00
if ( retval ) {
kfree ( kset ) ;
return NULL ;
}
2007-09-27 14:48:53 -07:00
kset - > uevent_ops = uevent_ops ;
kset - > kobj . parent = parent_kobj ;
/*
2007-11-02 13:47:53 +01:00
* The kobject of this kset will have a type of kset_ktype and belong to
2007-09-27 14:48:53 -07:00
* no kset itself . That way we can properly free it when it is
* finished being used .
*/
2007-11-02 13:47:53 +01:00
kset - > kobj . ktype = & kset_ktype ;
2007-09-27 14:48:53 -07:00
kset - > kobj . kset = NULL ;
return kset ;
}
/**
2019-05-02 12:31:40 +10:00
* kset_create_and_add ( ) - Create a struct kset dynamically and add it to sysfs .
2007-09-27 14:48:53 -07:00
*
* @ 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 ,
2009-12-31 14:52:51 +01:00
const struct kset_uevent_ops * uevent_ops ,
2007-09-27 14:48:53 -07:00
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 ) ;
2010-03-30 11:31:25 -07:00
static DEFINE_SPINLOCK ( kobj_ns_type_lock ) ;
static const struct kobj_ns_type_operations * kobj_ns_ops_tbl [ KOBJ_NS_TYPES ] ;
int kobj_ns_type_register ( const struct kobj_ns_type_operations * ops )
{
enum kobj_ns_type type = ops - > type ;
int error ;
spin_lock ( & kobj_ns_type_lock ) ;
error = - EINVAL ;
if ( type > = KOBJ_NS_TYPES )
goto out ;
error = - EINVAL ;
if ( type < = KOBJ_NS_TYPE_NONE )
goto out ;
error = - EBUSY ;
if ( kobj_ns_ops_tbl [ type ] )
goto out ;
error = 0 ;
kobj_ns_ops_tbl [ type ] = ops ;
out :
spin_unlock ( & kobj_ns_type_lock ) ;
return error ;
}
int kobj_ns_type_registered ( enum kobj_ns_type type )
{
int registered = 0 ;
spin_lock ( & kobj_ns_type_lock ) ;
if ( ( type > KOBJ_NS_TYPE_NONE ) & & ( type < KOBJ_NS_TYPES ) )
registered = kobj_ns_ops_tbl [ type ] ! = NULL ;
spin_unlock ( & kobj_ns_type_lock ) ;
return registered ;
}
const struct kobj_ns_type_operations * kobj_child_ns_ops ( struct kobject * parent )
{
const struct kobj_ns_type_operations * ops = NULL ;
2014-09-24 16:25:54 +05:30
if ( parent & & parent - > ktype & & parent - > ktype - > child_ns_type )
2010-03-30 11:31:25 -07:00
ops = parent - > ktype - > child_ns_type ( parent ) ;
return ops ;
}
const struct kobj_ns_type_operations * kobj_ns_ops ( struct kobject * kobj )
{
return kobj_child_ns_ops ( kobj - > parent ) ;
}
2013-03-25 20:07:01 -07:00
bool kobj_ns_current_may_mount ( enum kobj_ns_type type )
{
2013-09-23 14:41:17 -07:00
bool may_mount = true ;
2013-03-25 20:07:01 -07:00
spin_lock ( & kobj_ns_type_lock ) ;
if ( ( type > KOBJ_NS_TYPE_NONE ) & & ( type < KOBJ_NS_TYPES ) & &
kobj_ns_ops_tbl [ type ] )
may_mount = kobj_ns_ops_tbl [ type ] - > current_may_mount ( ) ;
spin_unlock ( & kobj_ns_type_lock ) ;
return may_mount ;
}
2010-03-30 11:31:25 -07:00
2011-06-08 21:13:01 -04:00
void * kobj_ns_grab_current ( enum kobj_ns_type type )
2010-03-30 11:31:25 -07:00
{
2011-06-08 21:13:01 -04:00
void * ns = NULL ;
2010-03-30 11:31:25 -07:00
spin_lock ( & kobj_ns_type_lock ) ;
if ( ( type > KOBJ_NS_TYPE_NONE ) & & ( type < KOBJ_NS_TYPES ) & &
kobj_ns_ops_tbl [ type ] )
2011-06-08 21:13:01 -04:00
ns = kobj_ns_ops_tbl [ type ] - > grab_current_ns ( ) ;
2010-03-30 11:31:25 -07:00
spin_unlock ( & kobj_ns_type_lock ) ;
return ns ;
}
2018-01-22 14:27:11 -08:00
EXPORT_SYMBOL_GPL ( kobj_ns_grab_current ) ;
2010-03-30 11:31:25 -07:00
const void * kobj_ns_netlink ( enum kobj_ns_type type , struct sock * sk )
{
const void * ns = NULL ;
spin_lock ( & kobj_ns_type_lock ) ;
if ( ( type > KOBJ_NS_TYPE_NONE ) & & ( type < KOBJ_NS_TYPES ) & &
kobj_ns_ops_tbl [ type ] )
ns = kobj_ns_ops_tbl [ type ] - > netlink_ns ( sk ) ;
spin_unlock ( & kobj_ns_type_lock ) ;
return ns ;
}
const void * kobj_ns_initial ( enum kobj_ns_type type )
{
const void * ns = NULL ;
spin_lock ( & kobj_ns_type_lock ) ;
if ( ( type > KOBJ_NS_TYPE_NONE ) & & ( type < KOBJ_NS_TYPES ) & &
kobj_ns_ops_tbl [ type ] )
ns = kobj_ns_ops_tbl [ type ] - > initial_ns ( ) ;
spin_unlock ( & kobj_ns_type_lock ) ;
return ns ;
}
2011-06-08 21:13:01 -04:00
void kobj_ns_drop ( enum kobj_ns_type type , void * ns )
2010-03-30 11:31:25 -07:00
{
2011-06-08 21:13:01 -04:00
spin_lock ( & kobj_ns_type_lock ) ;
if ( ( type > KOBJ_NS_TYPE_NONE ) & & ( type < KOBJ_NS_TYPES ) & &
kobj_ns_ops_tbl [ type ] & & kobj_ns_ops_tbl [ type ] - > drop_ns )
kobj_ns_ops_tbl [ type ] - > drop_ns ( ns ) ;
spin_unlock ( & kobj_ns_type_lock ) ;
2010-03-30 11:31:25 -07:00
}
2018-01-22 14:27:11 -08:00
EXPORT_SYMBOL_GPL ( kobj_ns_drop ) ;