2005-04-17 02:20:36 +04:00
/*
* fs / sysfs / group . c - Operations for adding / removing multiple files at once .
*
* Copyright ( c ) 2003 Patrick Mochel
* Copyright ( c ) 2003 Open Source Development Lab
*
* This file is released undert the GPL v2 .
*
*/
# include <linux/kobject.h>
# include <linux/module.h>
# include <linux/dcache.h>
2005-06-23 11:09:12 +04:00
# include <linux/namei.h>
2005-04-17 02:20:36 +04:00
# include <linux/err.h>
# include "sysfs.h"
2007-10-31 17:38:04 +03:00
static void remove_files ( struct sysfs_dirent * dir_sd , struct kobject * kobj ,
2007-06-13 23:27:22 +04:00
const struct attribute_group * grp )
2005-04-17 02:20:36 +04:00
{
struct attribute * const * attr ;
2007-10-31 17:38:04 +03:00
int i ;
2005-04-17 02:20:36 +04:00
2007-10-31 17:38:04 +03:00
for ( i = 0 , attr = grp - > attrs ; * attr ; i + + , attr + + )
if ( ! grp - > is_visible | |
grp - > is_visible ( kobj , * attr , i ) )
sysfs_hash_and_remove ( dir_sd , ( * attr ) - > name ) ;
2005-04-17 02:20:36 +04:00
}
2007-10-31 17:38:04 +03:00
static int create_files ( struct sysfs_dirent * dir_sd , struct kobject * kobj ,
2007-06-13 23:27:22 +04:00
const struct attribute_group * grp )
2005-04-17 02:20:36 +04:00
{
struct attribute * const * attr ;
2007-10-31 17:38:04 +03:00
int error = 0 , i ;
2005-04-17 02:20:36 +04:00
2007-10-31 17:38:04 +03:00
for ( i = 0 , attr = grp - > attrs ; * attr & & ! error ; i + + , attr + + )
if ( ! grp - > is_visible | |
grp - > is_visible ( kobj , * attr , i ) )
error | =
sysfs_add_file ( dir_sd , * attr , SYSFS_KOBJ_ATTR ) ;
2005-04-17 02:20:36 +04:00
if ( error )
2007-10-31 17:38:04 +03:00
remove_files ( dir_sd , kobj , grp ) ;
2005-04-17 02:20:36 +04:00
return error ;
}
int sysfs_create_group ( struct kobject * kobj ,
const struct attribute_group * grp )
{
2007-06-13 23:27:22 +04:00
struct sysfs_dirent * sd ;
2005-04-17 02:20:36 +04:00
int error ;
2007-06-13 23:27:22 +04:00
BUG_ON ( ! kobj | | ! kobj - > sd ) ;
2005-04-17 02:20:36 +04:00
if ( grp - > name ) {
2007-06-13 23:27:22 +04:00
error = sysfs_create_subdir ( kobj , grp - > name , & sd ) ;
2005-04-17 02:20:36 +04:00
if ( error )
return error ;
} else
2007-06-13 23:27:22 +04:00
sd = kobj - > sd ;
sysfs_get ( sd ) ;
2007-10-31 17:38:04 +03:00
error = create_files ( sd , kobj , grp ) ;
2007-06-13 23:27:22 +04:00
if ( error ) {
2005-04-17 02:20:36 +04:00
if ( grp - > name )
2007-06-13 23:27:22 +04:00
sysfs_remove_subdir ( sd ) ;
2005-04-17 02:20:36 +04:00
}
2007-06-13 23:27:22 +04:00
sysfs_put ( sd ) ;
2005-04-17 02:20:36 +04:00
return error ;
}
void sysfs_remove_group ( struct kobject * kobj ,
const struct attribute_group * grp )
{
2007-06-13 23:27:22 +04:00
struct sysfs_dirent * dir_sd = kobj - > sd ;
struct sysfs_dirent * sd ;
2005-04-17 02:20:36 +04:00
2007-04-26 11:12:05 +04:00
if ( grp - > name ) {
2007-06-13 23:27:22 +04:00
sd = sysfs_get_dirent ( dir_sd , grp - > name ) ;
2008-02-07 19:58:54 +03:00
if ( ! sd ) {
printk ( KERN_WARNING " sysfs group %p not found for "
" kobject '%s' \n " , grp , kobject_name ( kobj ) ) ;
WARN_ON ( ! sd ) ;
return ;
}
2007-06-13 23:27:22 +04:00
} else
sd = sysfs_get ( dir_sd ) ;
2005-04-17 02:20:36 +04:00
2007-10-31 17:38:04 +03:00
remove_files ( sd , kobj , grp ) ;
2005-04-17 02:20:36 +04:00
if ( grp - > name )
2007-06-13 23:27:22 +04:00
sysfs_remove_subdir ( sd ) ;
sysfs_put ( sd ) ;
2005-04-17 02:20:36 +04:00
}
EXPORT_SYMBOL_GPL ( sysfs_create_group ) ;
EXPORT_SYMBOL_GPL ( sysfs_remove_group ) ;