2005-04-17 02:20:36 +04:00
/*
* drivers / base / cpu . c - basic CPU class support
*/
# include <linux/sysdev.h>
# include <linux/module.h>
# include <linux/init.h>
2006-10-18 09:47:25 +04:00
# include <linux/sched.h>
2005-04-17 02:20:36 +04:00
# include <linux/cpu.h>
# include <linux/topology.h>
# include <linux/device.h>
[PATCH] node hotplug: register cpu: remove node struct
With Goto-san's patch, we can add new pgdat/node at runtime. I'm now
considering node-hot-add with cpu + memory on ACPI.
I found acpi container, which describes node, could evaluate cpu before
memory. This means cpu-hot-add occurs before memory hot add.
In most part, cpu-hot-add doesn't depend on node hot add. But register_cpu(),
which creates symbolic link from node to cpu, requires that node should be
onlined before register_cpu(). When a node is onlined, its pgdat should be
there.
This patch-set holds off creating symbolic link from node to cpu
until node is onlined.
This removes node arguments from register_cpu().
Now, register_cpu() requires 'struct node' as its argument. But the array of
struct node is now unified in driver/base/node.c now (By Goto's node hotplug
patch). We can get struct node in generic way. So, this argument is not
necessary now.
This patch also guarantees add cpu under node only when node is onlined. It
is necessary for node-hot-add vs. cpu-hot-add patch following this.
Moreover, register_cpu calculates cpu->node_id by cpu_to_node() without regard
to its 'struct node *root' argument. This patch removes it.
Also modify callers of register_cpu()/unregister_cpu, whose args are changed
by register-cpu-remove-node-struct patch.
[Brice.Goglin@ens-lyon.org: fix it]
Signed-off-by: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Cc: Yasunori Goto <y-goto@jp.fujitsu.com>
Cc: Ashok Raj <ashok.raj@intel.com>
Cc: Dave Hansen <haveblue@us.ibm.com>
Signed-off-by: Brice Goglin <Brice.Goglin@ens-lyon.org>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-06-27 13:53:41 +04:00
# include <linux/node.h>
2005-04-17 02:20:36 +04:00
2005-10-13 20:54:41 +04:00
# include "base.h"
2005-04-17 02:20:36 +04:00
2010-01-05 14:48:03 +03:00
static struct sysdev_class_attribute * cpu_sysdev_class_attrs [ ] ;
2005-04-17 02:20:36 +04:00
struct sysdev_class cpu_sysdev_class = {
2007-12-20 04:09:39 +03:00
. name = " cpu " ,
2010-01-05 14:48:03 +03:00
. attrs = cpu_sysdev_class_attrs ,
2005-04-17 02:20:36 +04:00
} ;
EXPORT_SYMBOL ( cpu_sysdev_class ) ;
2008-05-01 15:35:16 +04:00
static DEFINE_PER_CPU ( struct sys_device * , cpu_sys_devices ) ;
2005-10-31 01:59:49 +03:00
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_HOTPLUG_CPU
2008-07-01 20:48:41 +04:00
static ssize_t show_online ( struct sys_device * dev , struct sysdev_attribute * attr ,
char * buf )
2005-04-17 02:20:36 +04:00
{
struct cpu * cpu = container_of ( dev , struct cpu , sysdev ) ;
return sprintf ( buf , " %u \n " , ! ! cpu_online ( cpu - > sysdev . id ) ) ;
}
2008-07-01 20:48:41 +04:00
static ssize_t __ref store_online ( struct sys_device * dev , struct sysdev_attribute * attr ,
const char * buf , size_t count )
2005-04-17 02:20:36 +04:00
{
struct cpu * cpu = container_of ( dev , struct cpu , sysdev ) ;
ssize_t ret ;
2009-11-26 12:59:05 +03:00
cpu_hotplug_driver_lock ( ) ;
2005-04-17 02:20:36 +04:00
switch ( buf [ 0 ] ) {
case ' 0 ' :
ret = cpu_down ( cpu - > sysdev . id ) ;
if ( ! ret )
2005-11-16 11:00:00 +03:00
kobject_uevent ( & dev - > kobj , KOBJ_OFFLINE ) ;
2005-04-17 02:20:36 +04:00
break ;
case ' 1 ' :
2006-03-25 14:08:18 +03:00
ret = cpu_up ( cpu - > sysdev . id ) ;
2005-06-26 01:55:05 +04:00
if ( ! ret )
2005-11-16 11:00:00 +03:00
kobject_uevent ( & dev - > kobj , KOBJ_ONLINE ) ;
2005-04-17 02:20:36 +04:00
break ;
default :
ret = - EINVAL ;
}
2009-11-26 12:59:05 +03:00
cpu_hotplug_driver_unlock ( ) ;
2005-04-17 02:20:36 +04:00
if ( ret > = 0 )
ret = count ;
return ret ;
}
2007-07-31 11:38:16 +04:00
static SYSDEV_ATTR ( online , 0644 , show_online , store_online ) ;
2005-04-17 02:20:36 +04:00
2008-03-05 02:09:05 +03:00
static void __cpuinit register_cpu_control ( struct cpu * cpu )
2005-04-17 02:20:36 +04:00
{
sysdev_create_file ( & cpu - > sysdev , & attr_online ) ;
}
[PATCH] node hotplug: register cpu: remove node struct
With Goto-san's patch, we can add new pgdat/node at runtime. I'm now
considering node-hot-add with cpu + memory on ACPI.
I found acpi container, which describes node, could evaluate cpu before
memory. This means cpu-hot-add occurs before memory hot add.
In most part, cpu-hot-add doesn't depend on node hot add. But register_cpu(),
which creates symbolic link from node to cpu, requires that node should be
onlined before register_cpu(). When a node is onlined, its pgdat should be
there.
This patch-set holds off creating symbolic link from node to cpu
until node is onlined.
This removes node arguments from register_cpu().
Now, register_cpu() requires 'struct node' as its argument. But the array of
struct node is now unified in driver/base/node.c now (By Goto's node hotplug
patch). We can get struct node in generic way. So, this argument is not
necessary now.
This patch also guarantees add cpu under node only when node is onlined. It
is necessary for node-hot-add vs. cpu-hot-add patch following this.
Moreover, register_cpu calculates cpu->node_id by cpu_to_node() without regard
to its 'struct node *root' argument. This patch removes it.
Also modify callers of register_cpu()/unregister_cpu, whose args are changed
by register-cpu-remove-node-struct patch.
[Brice.Goglin@ens-lyon.org: fix it]
Signed-off-by: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Cc: Yasunori Goto <y-goto@jp.fujitsu.com>
Cc: Ashok Raj <ashok.raj@intel.com>
Cc: Dave Hansen <haveblue@us.ibm.com>
Signed-off-by: Brice Goglin <Brice.Goglin@ens-lyon.org>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-06-27 13:53:41 +04:00
void unregister_cpu ( struct cpu * cpu )
2005-04-17 02:20:36 +04:00
{
2005-10-31 01:59:49 +03:00
int logical_cpu = cpu - > sysdev . id ;
2005-04-17 02:20:36 +04:00
[PATCH] node hotplug: register cpu: remove node struct
With Goto-san's patch, we can add new pgdat/node at runtime. I'm now
considering node-hot-add with cpu + memory on ACPI.
I found acpi container, which describes node, could evaluate cpu before
memory. This means cpu-hot-add occurs before memory hot add.
In most part, cpu-hot-add doesn't depend on node hot add. But register_cpu(),
which creates symbolic link from node to cpu, requires that node should be
onlined before register_cpu(). When a node is onlined, its pgdat should be
there.
This patch-set holds off creating symbolic link from node to cpu
until node is onlined.
This removes node arguments from register_cpu().
Now, register_cpu() requires 'struct node' as its argument. But the array of
struct node is now unified in driver/base/node.c now (By Goto's node hotplug
patch). We can get struct node in generic way. So, this argument is not
necessary now.
This patch also guarantees add cpu under node only when node is onlined. It
is necessary for node-hot-add vs. cpu-hot-add patch following this.
Moreover, register_cpu calculates cpu->node_id by cpu_to_node() without regard
to its 'struct node *root' argument. This patch removes it.
Also modify callers of register_cpu()/unregister_cpu, whose args are changed
by register-cpu-remove-node-struct patch.
[Brice.Goglin@ens-lyon.org: fix it]
Signed-off-by: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Cc: Yasunori Goto <y-goto@jp.fujitsu.com>
Cc: Ashok Raj <ashok.raj@intel.com>
Cc: Dave Hansen <haveblue@us.ibm.com>
Signed-off-by: Brice Goglin <Brice.Goglin@ens-lyon.org>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-06-27 13:53:41 +04:00
unregister_cpu_under_node ( logical_cpu , cpu_to_node ( logical_cpu ) ) ;
2005-04-17 02:20:36 +04:00
sysdev_remove_file ( & cpu - > sysdev , & attr_online ) ;
sysdev_unregister ( & cpu - > sysdev ) ;
2008-05-01 15:35:16 +04:00
per_cpu ( cpu_sys_devices , logical_cpu ) = NULL ;
2005-04-17 02:20:36 +04:00
return ;
}
2009-11-25 20:23:25 +03:00
# ifdef CONFIG_ARCH_CPU_PROBE_RELEASE
2010-03-16 02:33:32 +03:00
static ssize_t cpu_probe_store ( struct sysdev_class * class ,
struct sysdev_class_attribute * attr ,
const char * buf ,
2009-11-25 20:23:25 +03:00
size_t count )
{
return arch_cpu_probe ( buf , count ) ;
}
2010-03-16 02:33:32 +03:00
static ssize_t cpu_release_store ( struct sysdev_class * class ,
struct sysdev_class_attribute * attr ,
const char * buf ,
2009-11-25 20:23:25 +03:00
size_t count )
{
return arch_cpu_release ( buf , count ) ;
}
2010-03-16 02:33:32 +03:00
static SYSDEV_CLASS_ATTR ( probe , S_IWUSR , NULL , cpu_probe_store ) ;
static SYSDEV_CLASS_ATTR ( release , S_IWUSR , NULL , cpu_release_store ) ;
2009-11-25 20:23:25 +03:00
# endif /* CONFIG_ARCH_CPU_PROBE_RELEASE */
2005-04-17 02:20:36 +04:00
# else /* ... !CONFIG_HOTPLUG_CPU */
static inline void register_cpu_control ( struct cpu * cpu )
{
}
# endif /* CONFIG_HOTPLUG_CPU */
2006-01-10 07:51:42 +03:00
# ifdef CONFIG_KEXEC
# include <linux/kexec.h>
2008-07-01 20:48:41 +04:00
static ssize_t show_crash_notes ( struct sys_device * dev , struct sysdev_attribute * attr ,
char * buf )
2006-01-10 07:51:42 +03:00
{
struct cpu * cpu = container_of ( dev , struct cpu , sysdev ) ;
ssize_t rc ;
unsigned long long addr ;
int cpunum ;
cpunum = cpu - > sysdev . id ;
/*
* Might be reading other cpu ' s data based on which cpu read thread
* has been scheduled . But cpu data ( memory ) is allocated once during
* boot up and this data does not change there after . Hence this
* operation should be safe . No locking required .
*/
2009-11-24 09:50:03 +03:00
addr = per_cpu_ptr_to_phys ( per_cpu_ptr ( crash_notes , cpunum ) ) ;
2006-01-10 07:51:42 +03:00
rc = sprintf ( buf , " %Lx \n " , addr ) ;
return rc ;
}
static SYSDEV_ATTR ( crash_notes , 0400 , show_crash_notes , NULL ) ;
# endif
2008-04-08 22:43:04 +04:00
/*
* Print cpu online , possible , present , and system maps
*/
2010-01-05 14:48:00 +03:00
struct cpu_attr {
struct sysdev_class_attribute attr ;
const struct cpumask * const * const map ;
} ;
static ssize_t show_cpus_attr ( struct sysdev_class * class ,
struct sysdev_class_attribute * attr ,
char * buf )
2008-04-08 22:43:04 +04:00
{
2010-01-05 14:48:00 +03:00
struct cpu_attr * ca = container_of ( attr , struct cpu_attr , attr ) ;
int n = cpulist_scnprintf ( buf , PAGE_SIZE - 2 , * ( ca - > map ) ) ;
2008-04-08 22:43:04 +04:00
buf [ n + + ] = ' \n ' ;
buf [ n ] = ' \0 ' ;
return n ;
}
2010-01-05 14:48:00 +03:00
# define _CPU_ATTR(name, map) \
{ _SYSDEV_CLASS_ATTR ( name , 0444 , show_cpus_attr , NULL ) , map }
2008-04-08 22:43:04 +04:00
2010-01-05 14:48:03 +03:00
/* Keep in sync with cpu_sysdev_class_attrs */
2010-01-05 14:48:00 +03:00
static struct cpu_attr cpu_attrs [ ] = {
_CPU_ATTR ( online , & cpu_online_mask ) ,
_CPU_ATTR ( possible , & cpu_possible_mask ) ,
_CPU_ATTR ( present , & cpu_present_mask ) ,
} ;
2008-04-08 22:43:04 +04:00
2008-12-16 07:26:48 +03:00
/*
* Print values for NR_CPUS and offlined cpus
*/
2010-01-05 14:47:58 +03:00
static ssize_t print_cpus_kernel_max ( struct sysdev_class * class ,
struct sysdev_class_attribute * attr , char * buf )
2008-12-16 07:26:48 +03:00
{
2009-01-01 05:08:48 +03:00
int n = snprintf ( buf , PAGE_SIZE - 2 , " %d \n " , NR_CPUS - 1 ) ;
2008-12-16 07:26:48 +03:00
return n ;
}
static SYSDEV_CLASS_ATTR ( kernel_max , 0444 , print_cpus_kernel_max , NULL ) ;
/* arch-optional setting to enable display of offline cpus >= nr_cpu_ids */
unsigned int total_cpus ;
2010-01-05 14:47:58 +03:00
static ssize_t print_cpus_offline ( struct sysdev_class * class ,
struct sysdev_class_attribute * attr , char * buf )
2008-12-16 07:26:48 +03:00
{
int n = 0 , len = PAGE_SIZE - 2 ;
cpumask_var_t offline ;
/* display offline cpus < nr_cpu_ids */
if ( ! alloc_cpumask_var ( & offline , GFP_KERNEL ) )
return - ENOMEM ;
cpumask_complement ( offline , cpu_online_mask ) ;
n = cpulist_scnprintf ( buf , len , offline ) ;
free_cpumask_var ( offline ) ;
/* display offline cpus >= nr_cpu_ids */
if ( total_cpus & & nr_cpu_ids < total_cpus ) {
if ( n & & n < len )
buf [ n + + ] = ' , ' ;
if ( nr_cpu_ids = = total_cpus - 1 )
n + = snprintf ( & buf [ n ] , len - n , " %d " , nr_cpu_ids ) ;
else
n + = snprintf ( & buf [ n ] , len - n , " %d-%d " ,
nr_cpu_ids , total_cpus - 1 ) ;
}
n + = snprintf ( & buf [ n ] , len - n , " \n " ) ;
return n ;
}
static SYSDEV_CLASS_ATTR ( offline , 0444 , print_cpus_offline , NULL ) ;
2005-04-17 02:20:36 +04:00
/*
2007-02-17 21:13:42 +03:00
* register_cpu - Setup a sysfs device for a CPU .
2006-12-07 04:14:10 +03:00
* @ cpu - cpu - > hotpluggable field set to 1 will generate a control file in
* sysfs for this CPU .
2005-04-17 02:20:36 +04:00
* @ num - CPU number to use when creating the device .
*
* Initialize and register the CPU device .
*/
2008-02-06 12:36:28 +03:00
int __cpuinit register_cpu ( struct cpu * cpu , int num )
2005-04-17 02:20:36 +04:00
{
int error ;
cpu - > node_id = cpu_to_node ( num ) ;
cpu - > sysdev . id = num ;
cpu - > sysdev . cls = & cpu_sysdev_class ;
error = sysdev_register ( & cpu - > sysdev ) ;
[PATCH] node hotplug: register cpu: remove node struct
With Goto-san's patch, we can add new pgdat/node at runtime. I'm now
considering node-hot-add with cpu + memory on ACPI.
I found acpi container, which describes node, could evaluate cpu before
memory. This means cpu-hot-add occurs before memory hot add.
In most part, cpu-hot-add doesn't depend on node hot add. But register_cpu(),
which creates symbolic link from node to cpu, requires that node should be
onlined before register_cpu(). When a node is onlined, its pgdat should be
there.
This patch-set holds off creating symbolic link from node to cpu
until node is onlined.
This removes node arguments from register_cpu().
Now, register_cpu() requires 'struct node' as its argument. But the array of
struct node is now unified in driver/base/node.c now (By Goto's node hotplug
patch). We can get struct node in generic way. So, this argument is not
necessary now.
This patch also guarantees add cpu under node only when node is onlined. It
is necessary for node-hot-add vs. cpu-hot-add patch following this.
Moreover, register_cpu calculates cpu->node_id by cpu_to_node() without regard
to its 'struct node *root' argument. This patch removes it.
Also modify callers of register_cpu()/unregister_cpu, whose args are changed
by register-cpu-remove-node-struct patch.
[Brice.Goglin@ens-lyon.org: fix it]
Signed-off-by: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Cc: Yasunori Goto <y-goto@jp.fujitsu.com>
Cc: Ashok Raj <ashok.raj@intel.com>
Cc: Dave Hansen <haveblue@us.ibm.com>
Signed-off-by: Brice Goglin <Brice.Goglin@ens-lyon.org>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-06-27 13:53:41 +04:00
2006-12-07 04:14:10 +03:00
if ( ! error & & cpu - > hotpluggable )
2005-04-17 02:20:36 +04:00
register_cpu_control ( cpu ) ;
2005-10-31 01:59:49 +03:00
if ( ! error )
2008-05-01 15:35:16 +04:00
per_cpu ( cpu_sys_devices , num ) = & cpu - > sysdev ;
[PATCH] node hotplug: register cpu: remove node struct
With Goto-san's patch, we can add new pgdat/node at runtime. I'm now
considering node-hot-add with cpu + memory on ACPI.
I found acpi container, which describes node, could evaluate cpu before
memory. This means cpu-hot-add occurs before memory hot add.
In most part, cpu-hot-add doesn't depend on node hot add. But register_cpu(),
which creates symbolic link from node to cpu, requires that node should be
onlined before register_cpu(). When a node is onlined, its pgdat should be
there.
This patch-set holds off creating symbolic link from node to cpu
until node is onlined.
This removes node arguments from register_cpu().
Now, register_cpu() requires 'struct node' as its argument. But the array of
struct node is now unified in driver/base/node.c now (By Goto's node hotplug
patch). We can get struct node in generic way. So, this argument is not
necessary now.
This patch also guarantees add cpu under node only when node is onlined. It
is necessary for node-hot-add vs. cpu-hot-add patch following this.
Moreover, register_cpu calculates cpu->node_id by cpu_to_node() without regard
to its 'struct node *root' argument. This patch removes it.
Also modify callers of register_cpu()/unregister_cpu, whose args are changed
by register-cpu-remove-node-struct patch.
[Brice.Goglin@ens-lyon.org: fix it]
Signed-off-by: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Cc: Yasunori Goto <y-goto@jp.fujitsu.com>
Cc: Ashok Raj <ashok.raj@intel.com>
Cc: Dave Hansen <haveblue@us.ibm.com>
Signed-off-by: Brice Goglin <Brice.Goglin@ens-lyon.org>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-06-27 13:53:41 +04:00
if ( ! error )
register_cpu_under_node ( num , cpu_to_node ( num ) ) ;
2006-01-10 07:51:42 +03:00
# ifdef CONFIG_KEXEC
if ( ! error )
error = sysdev_create_file ( & cpu - > sysdev , & attr_crash_notes ) ;
# endif
2005-04-17 02:20:36 +04:00
return error ;
}
2006-03-08 10:53:25 +03:00
struct sys_device * get_cpu_sysdev ( unsigned cpu )
2005-10-31 01:59:49 +03:00
{
2008-05-01 15:35:16 +04:00
if ( cpu < nr_cpu_ids & & cpu_possible ( cpu ) )
return per_cpu ( cpu_sys_devices , cpu ) ;
2005-10-31 01:59:49 +03:00
else
return NULL ;
}
EXPORT_SYMBOL_GPL ( get_cpu_sysdev ) ;
2005-04-17 02:20:36 +04:00
int __init cpu_dev_init ( void )
{
2006-06-27 13:54:42 +04:00
int err ;
err = sysdev_class_register ( & cpu_sysdev_class ) ;
# if defined(CONFIG_SCHED_MC) || defined(CONFIG_SCHED_SMT)
if ( ! err )
err = sched_create_sysfs_power_savings_entries ( & cpu_sysdev_class ) ;
# endif
return err ;
2005-04-17 02:20:36 +04:00
}
2010-01-05 14:48:03 +03:00
static struct sysdev_class_attribute * cpu_sysdev_class_attrs [ ] = {
# ifdef CONFIG_ARCH_CPU_PROBE_RELEASE
2010-01-22 02:07:59 +03:00
& attr_probe ,
& attr_release ,
2010-01-05 14:48:03 +03:00
# endif
& cpu_attrs [ 0 ] . attr ,
& cpu_attrs [ 1 ] . attr ,
& cpu_attrs [ 2 ] . attr ,
& attr_kernel_max ,
& attr_offline ,
NULL
} ;