2005-04-16 15:20:36 -07:00
/*
2007-09-20 17:31:38 +09:00
* fs / sysfs / symlink . c - sysfs symlink implementation
*
* Copyright ( c ) 2001 - 3 Patrick Mochel
* Copyright ( c ) 2007 SUSE Linux Products GmbH
* Copyright ( c ) 2007 Tejun Heo < teheo @ suse . de >
*
* This file is released under the GPLv2 .
*
* Please see Documentation / filesystems / sysfs . txt for more information .
2005-04-16 15:20:36 -07:00
*/
# include <linux/fs.h>
2002-04-09 12:14:34 -07:00
# include <linux/mount.h>
2005-04-16 15:20:36 -07:00
# include <linux/module.h>
# include <linux/kobject.h>
# include <linux/namei.h>
2007-07-26 14:53:53 +00:00
# include <linux/mutex.h>
2009-09-09 14:25:37 -04:00
# include <linux/security.h>
2005-04-16 15:20:36 -07:00
# include "sysfs.h"
2008-06-10 11:09:08 +02:00
static int sysfs_do_create_link ( struct kobject * kobj , struct kobject * target ,
const char * name , int warn )
2005-04-16 15:20:36 -07:00
{
2007-06-14 03:45:15 +09:00
struct sysfs_dirent * parent_sd = NULL ;
struct sysfs_dirent * target_sd = NULL ;
2007-06-14 04:27:23 +09:00
struct sysfs_dirent * sd = NULL ;
2007-06-14 04:27:24 +09:00
struct sysfs_addrm_cxt acxt ;
2007-06-14 04:27:23 +09:00
int error ;
2005-04-16 15:20:36 -07:00
2002-04-09 12:14:34 -07:00
BUG_ON ( ! name ) ;
2007-08-20 21:36:30 +09:00
if ( ! kobj )
parent_sd = & sysfs_root ;
else
2007-06-14 04:27:22 +09:00
parent_sd = kobj - > sd ;
2002-04-09 12:14:34 -07:00
2007-06-14 04:27:23 +09:00
error = - EFAULT ;
2007-06-14 04:27:22 +09:00
if ( ! parent_sd )
2007-06-14 04:27:23 +09:00
goto out_put ;
2007-06-14 03:45:15 +09:00
2007-06-14 04:27:22 +09:00
/* target->sd can go away beneath us but is protected with
2007-06-14 04:27:23 +09:00
* sysfs_assoc_lock . Fetch target_sd from it .
2007-06-14 03:45:15 +09:00
*/
2007-06-14 04:27:23 +09:00
spin_lock ( & sysfs_assoc_lock ) ;
2007-06-14 04:27:22 +09:00
if ( target - > sd )
target_sd = sysfs_get ( target - > sd ) ;
2007-06-14 04:27:23 +09:00
spin_unlock ( & sysfs_assoc_lock ) ;
2007-06-14 03:45:15 +09:00
2007-06-14 04:27:23 +09:00
error = - ENOENT ;
2007-06-14 03:45:15 +09:00
if ( ! target_sd )
2007-06-14 04:27:23 +09:00
goto out_put ;
error = - ENOMEM ;
sd = sysfs_new_dirent ( name , S_IFLNK | S_IRWXUGO , SYSFS_KOBJ_LINK ) ;
if ( ! sd )
goto out_put ;
2007-07-18 16:14:45 +09:00
2007-09-20 16:05:11 +09:00
sd - > s_symlink . target_sd = target_sd ;
2007-07-18 16:14:45 +09:00
target_sd = NULL ; /* reference is now owned by the symlink */
2005-04-16 15:20:36 -07:00
2007-06-14 04:27:24 +09:00
sysfs_addrm_start ( & acxt , parent_sd ) ;
2008-06-10 11:09:08 +02:00
if ( warn )
error = sysfs_add_one ( & acxt , sd ) ;
else
error = __sysfs_add_one ( & acxt , sd ) ;
2007-08-02 21:38:03 +09:00
sysfs_addrm_finish ( & acxt ) ;
2007-06-14 03:45:15 +09:00
2007-08-02 21:38:03 +09:00
if ( error )
2007-07-18 16:38:11 +09:00
goto out_put ;
return 0 ;
2007-06-14 04:27:24 +09:00
2007-06-14 04:27:23 +09:00
out_put :
sysfs_put ( target_sd ) ;
sysfs_put ( sd ) ;
2005-04-16 15:20:36 -07:00
return error ;
}
2008-06-10 11:09:08 +02:00
/**
* sysfs_create_link - create symlink between two objects .
* @ kobj : object whose directory we ' re creating the link in .
* @ target : object we ' re pointing to .
* @ name : name of the symlink .
*/
int sysfs_create_link ( struct kobject * kobj , struct kobject * target ,
const char * name )
{
return sysfs_do_create_link ( kobj , target , name , 1 ) ;
}
/**
* sysfs_create_link_nowarn - create symlink between two objects .
* @ kobj : object whose directory we ' re creating the link in .
* @ target : object we ' re pointing to .
* @ name : name of the symlink .
*
* This function does the same as sysf_create_link ( ) , but it
* doesn ' t warn if the link already exists .
*/
int sysfs_create_link_nowarn ( struct kobject * kobj , struct kobject * target ,
const char * name )
{
return sysfs_do_create_link ( kobj , target , name , 0 ) ;
}
2005-04-16 15:20:36 -07:00
/**
* sysfs_remove_link - remove symlink in object ' s directory .
* @ kobj : object we ' re acting for .
* @ name : name of the symlink to remove .
*/
2005-04-26 02:31:08 -05:00
void sysfs_remove_link ( struct kobject * kobj , const char * name )
2005-04-16 15:20:36 -07:00
{
2008-01-29 14:35:18 -08:00
struct sysfs_dirent * parent_sd = NULL ;
if ( ! kobj )
parent_sd = & sysfs_root ;
else
parent_sd = kobj - > sd ;
sysfs_hash_and_remove ( parent_sd , name ) ;
2005-04-16 15:20:36 -07:00
}
2007-11-01 20:20:52 +01:00
static int sysfs_get_target_path ( struct sysfs_dirent * parent_sd ,
struct sysfs_dirent * target_sd , char * path )
2005-04-16 15:20:36 -07:00
{
2007-11-01 20:20:52 +01:00
struct sysfs_dirent * base , * sd ;
char * s = path ;
int len = 0 ;
/* go up to the root, stop at the base */
base = parent_sd ;
while ( base - > s_parent ) {
sd = target_sd - > s_parent ;
while ( sd - > s_parent & & base ! = sd )
sd = sd - > s_parent ;
if ( base = = sd )
break ;
strcpy ( s , " ../ " ) ;
s + = 3 ;
base = base - > s_parent ;
}
/* determine end of target string for reverse fillup */
sd = target_sd ;
while ( sd - > s_parent & & sd ! = base ) {
len + = strlen ( sd - > s_name ) + 1 ;
sd = sd - > s_parent ;
}
2005-04-16 15:20:36 -07:00
2007-11-01 20:20:52 +01:00
/* check limits */
if ( len < 2 )
return - EINVAL ;
len - - ;
if ( ( s - path ) + len > PATH_MAX )
2005-04-16 15:20:36 -07:00
return - ENAMETOOLONG ;
2007-11-01 20:20:52 +01:00
/* reverse fillup of target string from target to base */
sd = target_sd ;
while ( sd - > s_parent & & sd ! = base ) {
int slen = strlen ( sd - > s_name ) ;
2005-04-16 15:20:36 -07:00
2007-11-01 20:20:52 +01:00
len - = slen ;
strncpy ( s + len , sd - > s_name , slen ) ;
if ( len )
s [ - - len ] = ' / ' ;
2005-04-16 15:20:36 -07:00
2007-11-01 20:20:52 +01:00
sd = sd - > s_parent ;
}
2005-04-16 15:20:36 -07:00
return 0 ;
}
static int sysfs_getlink ( struct dentry * dentry , char * path )
{
2007-06-14 03:45:15 +09:00
struct sysfs_dirent * sd = dentry - > d_fsdata ;
struct sysfs_dirent * parent_sd = sd - > s_parent ;
2007-09-20 16:05:11 +09:00
struct sysfs_dirent * target_sd = sd - > s_symlink . target_sd ;
2007-06-14 03:45:15 +09:00
int error ;
2005-04-16 15:20:36 -07:00
2007-06-14 04:27:23 +09:00
mutex_lock ( & sysfs_mutex ) ;
2007-06-14 03:45:15 +09:00
error = sysfs_get_target_path ( parent_sd , target_sd , path ) ;
2007-06-14 04:27:23 +09:00
mutex_unlock ( & sysfs_mutex ) ;
2005-04-16 15:20:36 -07:00
2007-06-14 03:45:15 +09:00
return error ;
2005-04-16 15:20:36 -07:00
}
2005-08-19 18:02:56 -07:00
static void * sysfs_follow_link ( struct dentry * dentry , struct nameidata * nd )
2005-04-16 15:20:36 -07:00
{
int error = - ENOMEM ;
unsigned long page = get_zeroed_page ( GFP_KERNEL ) ;
2009-04-29 07:29:59 -10:00
if ( page ) {
2005-04-16 15:20:36 -07:00
error = sysfs_getlink ( dentry , ( char * ) page ) ;
2009-04-29 07:29:59 -10:00
if ( error < 0 )
free_page ( ( unsigned long ) page ) ;
}
2005-04-16 15:20:36 -07:00
nd_set_link ( nd , error ? ERR_PTR ( error ) : ( char * ) page ) ;
2005-08-19 18:02:56 -07:00
return NULL ;
2005-04-16 15:20:36 -07:00
}
2005-08-19 18:02:56 -07:00
static void sysfs_put_link ( struct dentry * dentry , struct nameidata * nd , void * cookie )
2005-04-16 15:20:36 -07:00
{
char * page = nd_get_link ( nd ) ;
if ( ! IS_ERR ( page ) )
free_page ( ( unsigned long ) page ) ;
}
2007-02-12 00:55:40 -08:00
const struct inode_operations sysfs_symlink_inode_operations = {
2009-09-09 14:25:37 -04:00
. setxattr = sysfs_setxattr ,
2005-04-16 15:20:36 -07:00
. readlink = generic_readlink ,
. follow_link = sysfs_follow_link ,
. put_link = sysfs_put_link ,
} ;
EXPORT_SYMBOL_GPL ( sysfs_create_link ) ;
EXPORT_SYMBOL_GPL ( sysfs_remove_link ) ;