Commit 64ec66bb authored by David Gibson's avatar David Gibson Committed by Linus Torvalds

[PATCH] ppc64: tweaks to ppc64 cpu sysfs information

Currently the ppc64 sysfs code registers an entry for each possible cpu in
sysfs, rather than just online cpus.  That makes sense, since the sysfs
entries are needed to control onlining of the cpus.  However, this is done
even if CONFIG_HOTPLUG_CPU is not set, or if it is not a hotplug capable
(DLPAR) machine, which is a bit misleading.  Secondly it also registers all
the other sysfs entries (mostly performance monitoring controls) on all
possible cpus, although they are quite meaningless on non-online cpus.

This patch alters the code to only register sysfs directories at boot for
cpus which are either online or could be onlined (cpu is possible, and
CONFIG_HOTPLUG_CPU and an lpar machine).  Furthermore, the entries apart
from 'online' itself and 'physical_id' are only registered for online CPUs
(and deregistered again if a cpu goes offline).

Currently the ppc64 sysfs code registers an entry for each possible cpu in
sysfs, rather than just online cpus.  That makes sense, since the sysfs
entries are needed to control onlining of the cpus.  However, this is done
even if CONFIG_HOTPLUG_CPU is not set, or if it is not a hotplug capable
(DLPAR) machine, which is a bit misleading.  Secondly it also registers all
the other sysfs entries (mostly performance monitoring controls) on all
possible cpus, although they are quite meaningless on non-online cpus.

This patch alters the code to only register sysfs directories at boot for
cpus which are either online or could be onlined (cpu is possible, and
CONFIG_HOTPLUG_CPU and an lpar machine).  Furthermore, the entries apart
from 'online' itself and 'physical_id' are only registered for online CPUs
(and deregistered again if a cpu goes offline).
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 53a50435
...@@ -7,6 +7,8 @@ ...@@ -7,6 +7,8 @@
#include <linux/sched.h> #include <linux/sched.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/nodemask.h> #include <linux/nodemask.h>
#include <linux/cpumask.h>
#include <linux/notifier.h>
#include <asm/current.h> #include <asm/current.h>
#include <asm/processor.h> #include <asm/processor.h>
...@@ -15,6 +17,8 @@ ...@@ -15,6 +17,8 @@
#include <asm/prom.h> #include <asm/prom.h>
static DEFINE_PER_CPU(struct cpu, cpu_devices);
/* SMT stuff */ /* SMT stuff */
#ifdef CONFIG_PPC_MULTIPLATFORM #ifdef CONFIG_PPC_MULTIPLATFORM
...@@ -255,8 +259,18 @@ static SYSDEV_ATTR(pmc7, 0600, show_pmc7, store_pmc7); ...@@ -255,8 +259,18 @@ static SYSDEV_ATTR(pmc7, 0600, show_pmc7, store_pmc7);
static SYSDEV_ATTR(pmc8, 0600, show_pmc8, store_pmc8); static SYSDEV_ATTR(pmc8, 0600, show_pmc8, store_pmc8);
static SYSDEV_ATTR(purr, 0600, show_purr, NULL); static SYSDEV_ATTR(purr, 0600, show_purr, NULL);
static void __init register_cpu_pmc(struct sys_device *s) static void register_cpu_online(unsigned int cpu)
{ {
struct cpu *c = &per_cpu(cpu_devices, cpu);
struct sys_device *s = &c->sysdev;
#ifndef CONFIG_PPC_ISERIES
if (cur_cpu_spec->cpu_features & CPU_FTR_SMT)
sysdev_create_file(s, &attr_smt_snooze_delay);
#endif
/* PMC stuff */
sysdev_create_file(s, &attr_mmcr0); sysdev_create_file(s, &attr_mmcr0);
sysdev_create_file(s, &attr_mmcr1); sysdev_create_file(s, &attr_mmcr1);
...@@ -279,6 +293,65 @@ static void __init register_cpu_pmc(struct sys_device *s) ...@@ -279,6 +293,65 @@ static void __init register_cpu_pmc(struct sys_device *s)
sysdev_create_file(s, &attr_purr); sysdev_create_file(s, &attr_purr);
} }
#ifdef CONFIG_HOTPLUG_CPU
static void unregister_cpu_online(unsigned int cpu)
{
struct cpu *c = &per_cpu(cpu_devices, cpu);
struct sys_device *s = &c->sysdev;
BUG_ON(c->no_control);
#ifndef CONFIG_PPC_ISERIES
if (cur_cpu_spec->cpu_features & CPU_FTR_SMT)
sysdev_remove_file(s, &attr_smt_snooze_delay);
#endif
/* PMC stuff */
sysdev_remove_file(s, &attr_mmcr0);
sysdev_remove_file(s, &attr_mmcr1);
if (cur_cpu_spec->cpu_features & CPU_FTR_MMCRA)
sysdev_remove_file(s, &attr_mmcra);
sysdev_remove_file(s, &attr_pmc1);
sysdev_remove_file(s, &attr_pmc2);
sysdev_remove_file(s, &attr_pmc3);
sysdev_remove_file(s, &attr_pmc4);
sysdev_remove_file(s, &attr_pmc5);
sysdev_remove_file(s, &attr_pmc6);
if (cur_cpu_spec->cpu_features & CPU_FTR_PMC8) {
sysdev_remove_file(s, &attr_pmc7);
sysdev_remove_file(s, &attr_pmc8);
}
if (cur_cpu_spec->cpu_features & CPU_FTR_SMT)
sysdev_remove_file(s, &attr_purr);
}
#endif /* CONFIG_HOTPLUG_CPU */
static int __devinit sysfs_cpu_notify(struct notifier_block *self,
unsigned long action, void *hcpu)
{
unsigned int cpu = (unsigned int)(long)hcpu;
switch (action) {
case CPU_ONLINE:
register_cpu_online(cpu);
break;
#ifdef CONFIG_HOTPLUG_CPU
case CPU_DEAD:
unregister_cpu_online(cpu);
break;
#endif
}
return NOTIFY_OK;
}
static struct notifier_block __devinitdata sysfs_cpu_nb = {
.notifier_call = sysfs_cpu_notify,
};
/* NUMA stuff */ /* NUMA stuff */
...@@ -308,8 +381,7 @@ static void register_nodes(void) ...@@ -308,8 +381,7 @@ static void register_nodes(void)
} }
#endif #endif
/* Only valid if CPU is present. */
/* Only valid if CPU is online. */
static ssize_t show_physical_id(struct sys_device *dev, char *buf) static ssize_t show_physical_id(struct sys_device *dev, char *buf)
{ {
struct cpu *cpu = container_of(dev, struct cpu, sysdev); struct cpu *cpu = container_of(dev, struct cpu, sysdev);
...@@ -318,9 +390,6 @@ static ssize_t show_physical_id(struct sys_device *dev, char *buf) ...@@ -318,9 +390,6 @@ static ssize_t show_physical_id(struct sys_device *dev, char *buf)
} }
static SYSDEV_ATTR(physical_id, 0444, show_physical_id, NULL); static SYSDEV_ATTR(physical_id, 0444, show_physical_id, NULL);
static DEFINE_PER_CPU(struct cpu, cpu_devices);
static int __init topology_init(void) static int __init topology_init(void)
{ {
int cpu; int cpu;
...@@ -328,6 +397,8 @@ static int __init topology_init(void) ...@@ -328,6 +397,8 @@ static int __init topology_init(void)
register_nodes(); register_nodes();
register_cpu_notifier(&sysfs_cpu_nb);
for_each_cpu(cpu) { for_each_cpu(cpu) {
struct cpu *c = &per_cpu(cpu_devices, cpu); struct cpu *c = &per_cpu(cpu_devices, cpu);
...@@ -341,19 +412,19 @@ static int __init topology_init(void) ...@@ -341,19 +412,19 @@ static int __init topology_init(void)
* CPU. For instance, the boot cpu might never be valid * CPU. For instance, the boot cpu might never be valid
* for hotplugging. * for hotplugging.
*/ */
#ifdef CONFIG_HOTPLUG_CPU
if (systemcfg->platform != PLATFORM_PSERIES_LPAR) if (systemcfg->platform != PLATFORM_PSERIES_LPAR)
#endif
c->no_control = 1; c->no_control = 1;
register_cpu(c, cpu, parent); if (cpu_online(cpu) || (c->no_control == 0)) {
register_cpu(c, cpu, parent);
register_cpu_pmc(&c->sysdev);
sysdev_create_file(&c->sysdev, &attr_physical_id); sysdev_create_file(&c->sysdev, &attr_physical_id);
}
#ifndef CONFIG_PPC_ISERIES if (cpu_online(cpu))
if (cur_cpu_spec->cpu_features & CPU_FTR_SMT) register_cpu_online(cpu);
sysdev_create_file(&c->sysdev, &attr_smt_snooze_delay);
#endif
} }
return 0; return 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment