cpu.c 6.6 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7
/*
 * drivers/base/cpu.c - basic CPU class support
 */

#include <linux/sysdev.h>
#include <linux/module.h>
#include <linux/init.h>
A
Al Viro 已提交
8
#include <linux/sched.h>
L
Linus Torvalds 已提交
9 10 11
#include <linux/cpu.h>
#include <linux/topology.h>
#include <linux/device.h>
12
#include <linux/node.h>
L
Linus Torvalds 已提交
13

14
#include "base.h"
L
Linus Torvalds 已提交
15 16

struct sysdev_class cpu_sysdev_class = {
17
	.name = "cpu",
L
Linus Torvalds 已提交
18 19 20
};
EXPORT_SYMBOL(cpu_sysdev_class);

21
static DEFINE_PER_CPU(struct sys_device *, cpu_sys_devices);
22

L
Linus Torvalds 已提交
23
#ifdef CONFIG_HOTPLUG_CPU
24 25
static ssize_t show_online(struct sys_device *dev, struct sysdev_attribute *attr,
			   char *buf)
L
Linus Torvalds 已提交
26 27 28 29 30 31
{
	struct cpu *cpu = container_of(dev, struct cpu, sysdev);

	return sprintf(buf, "%u\n", !!cpu_online(cpu->sysdev.id));
}

32 33
static ssize_t __ref store_online(struct sys_device *dev, struct sysdev_attribute *attr,
				 const char *buf, size_t count)
L
Linus Torvalds 已提交
34 35 36 37
{
	struct cpu *cpu = container_of(dev, struct cpu, sysdev);
	ssize_t ret;

38
	cpu_hotplug_driver_lock();
L
Linus Torvalds 已提交
39 40 41 42
	switch (buf[0]) {
	case '0':
		ret = cpu_down(cpu->sysdev.id);
		if (!ret)
43
			kobject_uevent(&dev->kobj, KOBJ_OFFLINE);
L
Linus Torvalds 已提交
44 45
		break;
	case '1':
46
		ret = cpu_up(cpu->sysdev.id);
47
		if (!ret)
48
			kobject_uevent(&dev->kobj, KOBJ_ONLINE);
L
Linus Torvalds 已提交
49 50 51 52
		break;
	default:
		ret = -EINVAL;
	}
53
	cpu_hotplug_driver_unlock();
L
Linus Torvalds 已提交
54 55 56 57 58

	if (ret >= 0)
		ret = count;
	return ret;
}
U
Ulrich Drepper 已提交
59
static SYSDEV_ATTR(online, 0644, show_online, store_online);
L
Linus Torvalds 已提交
60

61
static void __cpuinit register_cpu_control(struct cpu *cpu)
L
Linus Torvalds 已提交
62 63 64
{
	sysdev_create_file(&cpu->sysdev, &attr_online);
}
65
void unregister_cpu(struct cpu *cpu)
L
Linus Torvalds 已提交
66
{
67
	int logical_cpu = cpu->sysdev.id;
L
Linus Torvalds 已提交
68

69 70
	unregister_cpu_under_node(logical_cpu, cpu_to_node(logical_cpu));

L
Linus Torvalds 已提交
71 72 73
	sysdev_remove_file(&cpu->sysdev, &attr_online);

	sysdev_unregister(&cpu->sysdev);
74
	per_cpu(cpu_sys_devices, logical_cpu) = NULL;
L
Linus Torvalds 已提交
75 76
	return;
}
77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108

#ifdef CONFIG_ARCH_CPU_PROBE_RELEASE
static ssize_t cpu_probe_store(struct class *class, const char *buf,
			       size_t count)
{
	return arch_cpu_probe(buf, count);
}

static ssize_t cpu_release_store(struct class *class, const char *buf,
				 size_t count)
{
	return arch_cpu_release(buf, count);
}

static CLASS_ATTR(probe, S_IWUSR, NULL, cpu_probe_store);
static CLASS_ATTR(release, S_IWUSR, NULL, cpu_release_store);

int __init cpu_probe_release_init(void)
{
	int rc;

	rc = sysfs_create_file(&cpu_sysdev_class.kset.kobj,
			       &class_attr_probe.attr);
	if (!rc)
		rc = sysfs_create_file(&cpu_sysdev_class.kset.kobj,
				       &class_attr_release.attr);

	return rc;
}
device_initcall(cpu_probe_release_init);
#endif /* CONFIG_ARCH_CPU_PROBE_RELEASE */

L
Linus Torvalds 已提交
109 110 111 112 113 114
#else /* ... !CONFIG_HOTPLUG_CPU */
static inline void register_cpu_control(struct cpu *cpu)
{
}
#endif /* CONFIG_HOTPLUG_CPU */

115 116 117
#ifdef CONFIG_KEXEC
#include <linux/kexec.h>

118 119
static ssize_t show_crash_notes(struct sys_device *dev, struct sysdev_attribute *attr,
				char *buf)
120 121 122 123 124 125 126 127 128 129 130 131 132 133
{
	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.
	 */
134
	addr = per_cpu_ptr_to_phys(per_cpu_ptr(crash_notes, cpunum));
135 136 137 138 139 140
	rc = sprintf(buf, "%Lx\n", addr);
	return rc;
}
static SYSDEV_ATTR(crash_notes, 0400, show_crash_notes, NULL);
#endif

141 142 143
/*
 * Print cpu online, possible, present, and system maps
 */
144
static ssize_t print_cpus_map(char *buf, const struct cpumask *map)
145
{
146
	int n = cpulist_scnprintf(buf, PAGE_SIZE-2, map);
147 148 149 150 151 152 153 154 155

	buf[n++] = '\n';
	buf[n] = '\0';
	return n;
}

#define	print_cpus_func(type) \
static ssize_t print_cpus_##type(struct sysdev_class *class, char *buf)	\
{									\
156
	return print_cpus_map(buf, cpu_##type##_mask);			\
157
}									\
158
static struct sysdev_class_attribute attr_##type##_map = 		\
159 160 161 162 163 164
	_SYSDEV_CLASS_ATTR(type, 0444, print_cpus_##type, NULL)

print_cpus_func(online);
print_cpus_func(possible);
print_cpus_func(present);

165 166 167 168 169
/*
 * Print values for NR_CPUS and offlined cpus
 */
static ssize_t print_cpus_kernel_max(struct sysdev_class *class, char *buf)
{
170
	int n = snprintf(buf, PAGE_SIZE-2, "%d\n", NR_CPUS - 1);
171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206
	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;

static ssize_t print_cpus_offline(struct sysdev_class *class, char *buf)
{
	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);

207
static struct sysdev_class_attribute *cpu_state_attr[] = {
208 209 210
	&attr_online_map,
	&attr_possible_map,
	&attr_present_map,
211 212
	&attr_kernel_max,
	&attr_offline,
213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229
};

static int cpu_states_init(void)
{
	int i;
	int err = 0;

	for (i = 0;  i < ARRAY_SIZE(cpu_state_attr); i++) {
		int ret;
		ret = sysdev_class_create_file(&cpu_sysdev_class,
						cpu_state_attr[i]);
		if (!err)
			err = ret;
	}
	return err;
}

L
Linus Torvalds 已提交
230
/*
231
 * register_cpu - Setup a sysfs device for a CPU.
232 233
 * @cpu - cpu->hotpluggable field set to 1 will generate a control file in
 *	  sysfs for this CPU.
L
Linus Torvalds 已提交
234 235 236 237
 * @num - CPU number to use when creating the device.
 *
 * Initialize and register the CPU device.
 */
238
int __cpuinit register_cpu(struct cpu *cpu, int num)
L
Linus Torvalds 已提交
239 240 241 242 243 244 245
{
	int error;
	cpu->node_id = cpu_to_node(num);
	cpu->sysdev.id = num;
	cpu->sysdev.cls = &cpu_sysdev_class;

	error = sysdev_register(&cpu->sysdev);
246

247
	if (!error && cpu->hotpluggable)
L
Linus Torvalds 已提交
248
		register_cpu_control(cpu);
249
	if (!error)
250
		per_cpu(cpu_sys_devices, num) = &cpu->sysdev;
251 252
	if (!error)
		register_cpu_under_node(num, cpu_to_node(num));
253 254 255 256 257

#ifdef CONFIG_KEXEC
	if (!error)
		error = sysdev_create_file(&cpu->sysdev, &attr_crash_notes);
#endif
L
Linus Torvalds 已提交
258 259 260
	return error;
}

261
struct sys_device *get_cpu_sysdev(unsigned cpu)
262
{
263 264
	if (cpu < nr_cpu_ids && cpu_possible(cpu))
		return per_cpu(cpu_sys_devices, cpu);
265 266 267 268
	else
		return NULL;
}
EXPORT_SYMBOL_GPL(get_cpu_sysdev);
L
Linus Torvalds 已提交
269 270 271

int __init cpu_dev_init(void)
{
272 273 274
	int err;

	err = sysdev_class_register(&cpu_sysdev_class);
275 276 277
	if (!err)
		err = cpu_states_init();

278 279 280 281 282 283
#if defined(CONFIG_SCHED_MC) || defined(CONFIG_SCHED_SMT)
	if (!err)
		err = sched_create_sysfs_power_savings_entries(&cpu_sysdev_class);
#endif

	return err;
L
Linus Torvalds 已提交
284
}