diff options
Diffstat (limited to 'drivers/base/cpu.c')
| -rw-r--r-- | drivers/base/cpu.c | 136 | 
1 files changed, 71 insertions, 65 deletions
| diff --git a/drivers/base/cpu.c b/drivers/base/cpu.c index 3d48fc887ef4..a16d20e389f0 100644 --- a/drivers/base/cpu.c +++ b/drivers/base/cpu.c @@ -13,17 +13,21 @@  #include <linux/gfp.h>  #include <linux/slab.h>  #include <linux/percpu.h> +#include <linux/acpi.h>  #include "base.h" -struct bus_type cpu_subsys = { -	.name = "cpu", -	.dev_name = "cpu", -}; -EXPORT_SYMBOL_GPL(cpu_subsys); -  static DEFINE_PER_CPU(struct device *, cpu_sys_devices); +static int cpu_subsys_match(struct device *dev, struct device_driver *drv) +{ +	/* ACPI style match is the only one that may succeed. */ +	if (acpi_driver_match_device(dev, drv)) +		return 1; + +	return 0; +} +  #ifdef CONFIG_HOTPLUG_CPU  static void change_cpu_under_node(struct cpu *cpu,  			unsigned int from_nid, unsigned int to_nid) @@ -34,69 +38,45 @@ static void change_cpu_under_node(struct cpu *cpu,  	cpu->node_id = to_nid;  } -static ssize_t show_online(struct device *dev, -			   struct device_attribute *attr, -			   char *buf) +static int __ref cpu_subsys_online(struct device *dev)  {  	struct cpu *cpu = container_of(dev, struct cpu, dev); +	int cpuid = dev->id; +	int from_nid, to_nid; +	int ret; + +	cpu_hotplug_driver_lock(); + +	from_nid = cpu_to_node(cpuid); +	ret = cpu_up(cpuid); +	/* +	 * When hot adding memory to memoryless node and enabling a cpu +	 * on the node, node number of the cpu may internally change. +	 */ +	to_nid = cpu_to_node(cpuid); +	if (from_nid != to_nid) +		change_cpu_under_node(cpu, from_nid, to_nid); -	return sprintf(buf, "%u\n", !!cpu_online(cpu->dev.id)); +	cpu_hotplug_driver_unlock(); +	return ret;  } -static ssize_t __ref store_online(struct device *dev, -				  struct device_attribute *attr, -				  const char *buf, size_t count) +static int cpu_subsys_offline(struct device *dev)  { -	struct cpu *cpu = container_of(dev, struct cpu, dev); -	int cpuid = cpu->dev.id; -	int from_nid, to_nid; -	ssize_t ret; +	int ret;  	cpu_hotplug_driver_lock(); -	switch (buf[0]) { -	case '0': -		ret = cpu_down(cpuid); -		if (!ret) -			kobject_uevent(&dev->kobj, KOBJ_OFFLINE); -		break; -	case '1': -		from_nid = cpu_to_node(cpuid); -		ret = cpu_up(cpuid); - -		/* -		 * When hot adding memory to memoryless node and enabling a cpu -		 * on the node, node number of the cpu may internally change. -		 */ -		to_nid = cpu_to_node(cpuid); -		if (from_nid != to_nid) -			change_cpu_under_node(cpu, from_nid, to_nid); - -		if (!ret) -			kobject_uevent(&dev->kobj, KOBJ_ONLINE); -		break; -	default: -		ret = -EINVAL; -	} +	ret = cpu_down(dev->id);  	cpu_hotplug_driver_unlock(); - -	if (ret >= 0) -		ret = count;  	return ret;  } -static DEVICE_ATTR(online, 0644, show_online, store_online); -static void __cpuinit register_cpu_control(struct cpu *cpu) -{ -	device_create_file(&cpu->dev, &dev_attr_online); -}  void unregister_cpu(struct cpu *cpu)  {  	int logical_cpu = cpu->dev.id;  	unregister_cpu_under_node(logical_cpu, cpu_to_node(logical_cpu)); -	device_remove_file(&cpu->dev, &dev_attr_online); -  	device_unregister(&cpu->dev);  	per_cpu(cpu_sys_devices, logical_cpu) = NULL;  	return; @@ -122,13 +102,19 @@ static ssize_t cpu_release_store(struct device *dev,  static DEVICE_ATTR(probe, S_IWUSR, NULL, cpu_probe_store);  static DEVICE_ATTR(release, S_IWUSR, NULL, cpu_release_store);  #endif /* CONFIG_ARCH_CPU_PROBE_RELEASE */ - -#else /* ... !CONFIG_HOTPLUG_CPU */ -static inline void register_cpu_control(struct cpu *cpu) -{ -}  #endif /* CONFIG_HOTPLUG_CPU */ +struct bus_type cpu_subsys = { +	.name = "cpu", +	.dev_name = "cpu", +	.match = cpu_subsys_match, +#ifdef CONFIG_HOTPLUG_CPU +	.online = cpu_subsys_online, +	.offline = cpu_subsys_offline, +#endif +}; +EXPORT_SYMBOL_GPL(cpu_subsys); +  #ifdef CONFIG_KEXEC  #include <linux/kexec.h> @@ -164,8 +150,32 @@ static ssize_t show_crash_notes_size(struct device *dev,  	return rc;  }  static DEVICE_ATTR(crash_notes_size, 0400, show_crash_notes_size, NULL); + +static struct attribute *crash_note_cpu_attrs[] = { +	&dev_attr_crash_notes.attr, +	&dev_attr_crash_notes_size.attr, +	NULL +}; + +static struct attribute_group crash_note_cpu_attr_group = { +	.attrs = crash_note_cpu_attrs, +};  #endif +static const struct attribute_group *common_cpu_attr_groups[] = { +#ifdef CONFIG_KEXEC +	&crash_note_cpu_attr_group, +#endif +	NULL +}; + +static const struct attribute_group *hotplugable_cpu_attr_groups[] = { +#ifdef CONFIG_KEXEC +	&crash_note_cpu_attr_group, +#endif +	NULL +}; +  /*   * Print cpu online, possible, present, and system maps   */ @@ -277,24 +287,20 @@ int __cpuinit register_cpu(struct cpu *cpu, int num)  	cpu->dev.id = num;  	cpu->dev.bus = &cpu_subsys;  	cpu->dev.release = cpu_device_release; +	cpu->dev.offline_disabled = !cpu->hotpluggable; +	cpu->dev.offline = !cpu_online(num);  #ifdef CONFIG_ARCH_HAS_CPU_AUTOPROBE  	cpu->dev.bus->uevent = arch_cpu_uevent;  #endif +	cpu->dev.groups = common_cpu_attr_groups; +	if (cpu->hotpluggable) +		cpu->dev.groups = hotplugable_cpu_attr_groups;  	error = device_register(&cpu->dev); -	if (!error && cpu->hotpluggable) -		register_cpu_control(cpu);  	if (!error)  		per_cpu(cpu_sys_devices, num) = &cpu->dev;  	if (!error)  		register_cpu_under_node(num, cpu_to_node(num)); -#ifdef CONFIG_KEXEC -	if (!error) -		error = device_create_file(&cpu->dev, &dev_attr_crash_notes); -	if (!error) -		error = device_create_file(&cpu->dev, -					   &dev_attr_crash_notes_size); -#endif  	return error;  } | 
