2005-04-16 15:20:36 -07:00
/*
* kernel / ksysfs . c - sysfs attributes in / sys / kernel , which
* are not related to any other subsystem
*
* Copyright ( C ) 2004 Kay Sievers < kay . sievers @ vrfy . org >
*
* This file is release under the GPLv2
*
*/
# include <linux/kobject.h>
# include <linux/string.h>
# include <linux/sysfs.h>
2011-05-23 14:51:41 -04:00
# include <linux/export.h>
2005-04-16 15:20:36 -07:00
# include <linux/init.h>
2006-06-23 02:05:07 -07:00
# include <linux/kexec.h>
2008-10-15 22:01:46 -07:00
# include <linux/profile.h>
2011-07-28 14:22:29 -04:00
# include <linux/stat.h>
2007-10-15 17:00:14 +02:00
# include <linux/sched.h>
2011-02-28 15:57:17 +01:00
# include <linux/capability.h>
2014-04-07 15:39:20 -07:00
# include <linux/compiler.h>
2005-04-16 15:20:36 -07:00
2015-11-24 15:44:06 -08:00
# include <linux/rcupdate.h> /* rcu_expedited and rcu_normal */
2014-02-11 16:10:12 -05:00
2005-04-16 15:20:36 -07:00
# define KERNEL_ATTR_RO(_name) \
2007-11-02 13:47:53 +01:00
static struct kobj_attribute _name # # _attr = __ATTR_RO ( _name )
2005-04-16 15:20:36 -07:00
# define KERNEL_ATTR_RW(_name) \
2007-11-02 13:47:53 +01:00
static struct kobj_attribute _name # # _attr = \
2005-04-16 15:20:36 -07:00
__ATTR ( _name , 0644 , _name # # _show , _name # # _store )
2005-11-11 04:58:04 +01:00
/* current uevent sequence number */
2007-11-02 13:47:53 +01:00
static ssize_t uevent_seqnum_show ( struct kobject * kobj ,
struct kobj_attribute * attr , char * buf )
2005-04-16 15:20:36 -07:00
{
2007-11-02 13:47:53 +01:00
return sprintf ( buf , " %llu \n " , ( unsigned long long ) uevent_seqnum ) ;
2005-04-16 15:20:36 -07:00
}
2005-11-11 04:58:04 +01:00
KERNEL_ATTR_RO ( uevent_seqnum ) ;
2014-04-10 14:09:31 -07:00
# ifdef CONFIG_UEVENT_HELPER
2010-01-17 19:14:26 -02:00
/* uevent helper program, used during early boot */
2007-11-02 13:47:53 +01:00
static ssize_t uevent_helper_show ( struct kobject * kobj ,
struct kobj_attribute * attr , char * buf )
2005-11-11 04:58:04 +01:00
{
2007-11-02 13:47:53 +01:00
return sprintf ( buf , " %s \n " , uevent_helper ) ;
2005-11-11 04:58:04 +01:00
}
2007-11-02 13:47:53 +01:00
static ssize_t uevent_helper_store ( struct kobject * kobj ,
struct kobj_attribute * attr ,
const char * buf , size_t count )
2005-11-11 04:58:04 +01:00
{
2005-11-16 09:00:00 +01:00
if ( count + 1 > UEVENT_HELPER_PATH_LEN )
2005-11-11 04:58:04 +01:00
return - ENOENT ;
2007-11-02 13:47:53 +01:00
memcpy ( uevent_helper , buf , count ) ;
2005-11-16 09:00:00 +01:00
uevent_helper [ count ] = ' \0 ' ;
if ( count & & uevent_helper [ count - 1 ] = = ' \n ' )
uevent_helper [ count - 1 ] = ' \0 ' ;
2005-11-11 04:58:04 +01:00
return count ;
}
KERNEL_ATTR_RW ( uevent_helper ) ;
2014-04-10 14:09:31 -07:00
# endif
2005-04-16 15:20:36 -07:00
2008-10-15 22:01:46 -07:00
# ifdef CONFIG_PROFILING
static ssize_t profiling_show ( struct kobject * kobj ,
struct kobj_attribute * attr , char * buf )
{
return sprintf ( buf , " %d \n " , prof_on ) ;
}
static ssize_t profiling_store ( struct kobject * kobj ,
struct kobj_attribute * attr ,
const char * buf , size_t count )
{
int ret ;
if ( prof_on )
return - EEXIST ;
/*
* This eventually calls into get_option ( ) which
* has a ton of callers and is not const . It is
* easiest to cast it away here .
*/
profile_setup ( ( char * ) buf ) ;
ret = profile_init ( ) ;
if ( ret )
return ret ;
ret = create_proc_profile ( ) ;
if ( ret )
return ret ;
return count ;
}
KERNEL_ATTR_RW ( profiling ) ;
# endif
2015-09-09 15:38:55 -07:00
# ifdef CONFIG_KEXEC_CORE
2007-11-02 13:47:53 +01:00
static ssize_t kexec_loaded_show ( struct kobject * kobj ,
struct kobj_attribute * attr , char * buf )
2006-06-23 02:05:07 -07:00
{
2007-11-02 13:47:53 +01:00
return sprintf ( buf , " %d \n " , ! ! kexec_image ) ;
2006-06-23 02:05:07 -07:00
}
KERNEL_ATTR_RO ( kexec_loaded ) ;
2007-11-02 13:47:53 +01:00
static ssize_t kexec_crash_loaded_show ( struct kobject * kobj ,
struct kobj_attribute * attr , char * buf )
2006-06-23 02:05:07 -07:00
{
2016-08-02 14:06:16 -07:00
return sprintf ( buf , " %d \n " , kexec_crash_loaded ( ) ) ;
2006-06-23 02:05:07 -07:00
}
KERNEL_ATTR_RO ( kexec_crash_loaded ) ;
2007-10-16 23:27:27 -07:00
2009-12-15 16:47:46 -08:00
static ssize_t kexec_crash_size_show ( struct kobject * kobj ,
struct kobj_attribute * attr , char * buf )
{
return sprintf ( buf , " %zu \n " , crash_get_memory_size ( ) ) ;
}
static ssize_t kexec_crash_size_store ( struct kobject * kobj ,
struct kobj_attribute * attr ,
const char * buf , size_t count )
{
unsigned long cnt ;
int ret ;
2013-09-12 15:14:07 -07:00
if ( kstrtoul ( buf , 0 , & cnt ) )
2009-12-15 16:47:46 -08:00
return - EINVAL ;
ret = crash_shrink_memory ( cnt ) ;
return ret < 0 ? ret : count ;
}
KERNEL_ATTR_RW ( kexec_crash_size ) ;
2007-11-02 13:47:53 +01:00
static ssize_t vmcoreinfo_show ( struct kobject * kobj ,
struct kobj_attribute * attr , char * buf )
2007-10-16 23:27:27 -07:00
{
2016-08-02 14:06:00 -07:00
phys_addr_t vmcore_base = paddr_vmcoreinfo_note ( ) ;
return sprintf ( buf , " %pa %x \n " , & vmcore_base ,
2014-01-23 15:56:00 -08:00
( unsigned int ) sizeof ( vmcoreinfo_note ) ) ;
2007-10-16 23:27:27 -07:00
}
KERNEL_ATTR_RO ( vmcoreinfo ) ;
2015-09-09 15:38:55 -07:00
# endif /* CONFIG_KEXEC_CORE */
2006-06-23 02:05:07 -07:00
2011-02-28 15:57:17 +01:00
/* whether file capabilities are enabled */
static ssize_t fscaps_show ( struct kobject * kobj ,
struct kobj_attribute * attr , char * buf )
{
return sprintf ( buf , " %d \n " , file_caps_enabled ) ;
}
KERNEL_ATTR_RO ( fscaps ) ;
2015-12-07 13:09:52 -08:00
# ifndef CONFIG_TINY_RCU
2012-10-05 09:59:15 +03:00
int rcu_expedited ;
static ssize_t rcu_expedited_show ( struct kobject * kobj ,
struct kobj_attribute * attr , char * buf )
{
2015-11-24 15:44:06 -08:00
return sprintf ( buf , " %d \n " , READ_ONCE ( rcu_expedited ) ) ;
2012-10-05 09:59:15 +03:00
}
static ssize_t rcu_expedited_store ( struct kobject * kobj ,
struct kobj_attribute * attr ,
const char * buf , size_t count )
{
if ( kstrtoint ( buf , 0 , & rcu_expedited ) )
return - EINVAL ;
return count ;
}
KERNEL_ATTR_RW ( rcu_expedited ) ;
2015-11-24 15:44:06 -08:00
int rcu_normal ;
static ssize_t rcu_normal_show ( struct kobject * kobj ,
struct kobj_attribute * attr , char * buf )
{
return sprintf ( buf , " %d \n " , READ_ONCE ( rcu_normal ) ) ;
}
static ssize_t rcu_normal_store ( struct kobject * kobj ,
struct kobj_attribute * attr ,
const char * buf , size_t count )
{
if ( kstrtoint ( buf , 0 , & rcu_normal ) )
return - EINVAL ;
return count ;
}
KERNEL_ATTR_RW ( rcu_normal ) ;
2015-12-07 13:09:52 -08:00
# endif /* #ifndef CONFIG_TINY_RCU */
2015-11-24 15:44:06 -08:00
2007-07-19 01:48:39 -07:00
/*
* Make / sys / kernel / notes give the raw contents of our kernel . notes section .
*/
2014-04-07 15:39:20 -07:00
extern const void __start_notes __weak ;
extern const void __stop_notes __weak ;
2007-07-19 01:48:39 -07:00
# define notes_size (&__stop_notes - &__start_notes)
2010-05-12 18:28:57 -07:00
static ssize_t notes_read ( struct file * filp , struct kobject * kobj ,
struct bin_attribute * bin_attr ,
2007-07-19 01:48:39 -07:00
char * buf , loff_t off , size_t count )
{
memcpy ( buf , & __start_notes + off , count ) ;
return count ;
}
static struct bin_attribute notes_attr = {
. attr = {
. name = " notes " ,
. mode = S_IRUGO ,
} ,
. read = & notes_read ,
} ;
2007-11-06 10:36:58 -08:00
struct kobject * kernel_kobj ;
EXPORT_SYMBOL_GPL ( kernel_kobj ) ;
2005-04-16 15:20:36 -07:00
static struct attribute * kernel_attrs [ ] = {
2011-02-28 15:57:17 +01:00
& fscaps_attr . attr ,
2005-11-11 04:58:04 +01:00
& uevent_seqnum_attr . attr ,
2014-04-10 14:09:31 -07:00
# ifdef CONFIG_UEVENT_HELPER
2005-11-11 04:58:04 +01:00
& uevent_helper_attr . attr ,
2014-04-10 14:09:31 -07:00
# endif
2008-10-15 22:01:46 -07:00
# ifdef CONFIG_PROFILING
& profiling_attr . attr ,
# endif
2015-09-09 15:38:55 -07:00
# ifdef CONFIG_KEXEC_CORE
2006-06-23 02:05:07 -07:00
& kexec_loaded_attr . attr ,
& kexec_crash_loaded_attr . attr ,
2009-12-15 16:47:46 -08:00
& kexec_crash_size_attr . attr ,
2007-10-16 23:27:27 -07:00
& vmcoreinfo_attr . attr ,
2005-04-16 15:20:36 -07:00
# endif
2015-12-07 13:09:52 -08:00
# ifndef CONFIG_TINY_RCU
2012-10-05 09:59:15 +03:00
& rcu_expedited_attr . attr ,
2015-11-24 15:44:06 -08:00
& rcu_normal_attr . attr ,
2015-12-07 13:09:52 -08:00
# endif
2005-04-16 15:20:36 -07:00
NULL
} ;
static struct attribute_group kernel_attr_group = {
. attrs = kernel_attrs ,
} ;
static int __init ksysfs_init ( void )
{
2007-10-29 20:13:17 +01:00
int error ;
2005-04-16 15:20:36 -07:00
2007-11-06 10:36:58 -08:00
kernel_kobj = kobject_create_and_add ( " kernel " , NULL ) ;
if ( ! kernel_kobj ) {
2007-10-29 20:13:17 +01:00
error = - ENOMEM ;
goto exit ;
}
2007-11-06 10:36:58 -08:00
error = sysfs_create_group ( kernel_kobj , & kernel_attr_group ) ;
2007-10-29 20:13:17 +01:00
if ( error )
goto kset_exit ;
if ( notes_size > 0 ) {
2007-07-19 01:48:39 -07:00
notes_attr . size = notes_size ;
2007-11-06 10:36:58 -08:00
error = sysfs_create_bin_file ( kernel_kobj , & notes_attr ) ;
2007-10-29 20:13:17 +01:00
if ( error )
goto group_exit ;
2007-07-19 01:48:39 -07:00
}
2007-10-29 20:13:17 +01:00
return 0 ;
group_exit :
2007-11-06 10:36:58 -08:00
sysfs_remove_group ( kernel_kobj , & kernel_attr_group ) ;
2007-10-29 20:13:17 +01:00
kset_exit :
2007-12-20 08:13:05 -08:00
kobject_put ( kernel_kobj ) ;
2007-10-29 20:13:17 +01:00
exit :
2005-04-16 15:20:36 -07:00
return error ;
}
core_initcall ( ksysfs_init ) ;