processor_core.c 5.0 KB
Newer Older
1 2 3 4 5 6 7
/*
 * Copyright (C) 2005 Intel Corporation
 * Copyright (C) 2009 Hewlett-Packard Development Company, L.P.
 *
 *	Alex Chiang <achiang@hp.com>
 *	- Unified x86/ia64 implementations
 */
8
#include <linux/export.h>
9
#include <linux/acpi.h>
A
Alex Chiang 已提交
10 11 12
#include <acpi/processor.h>

#define _COMPONENT		ACPI_PROCESSOR_COMPONENT
13
ACPI_MODULE_NAME("processor_core");
A
Alex Chiang 已提交
14

15 16 17 18
static int map_lapic_id(struct acpi_subtable_header *entry,
		 u32 acpi_id, int *apic_id)
{
	struct acpi_madt_local_apic *lapic =
19
		container_of(entry, struct acpi_madt_local_apic, header);
20 21

	if (!(lapic->lapic_flags & ACPI_MADT_ENABLED))
22
		return -ENODEV;
23 24

	if (lapic->processor_id != acpi_id)
25
		return -EINVAL;
26 27

	*apic_id = lapic->id;
28
	return 0;
29 30 31 32 33 34
}

static int map_x2apic_id(struct acpi_subtable_header *entry,
			 int device_declaration, u32 acpi_id, int *apic_id)
{
	struct acpi_madt_local_x2apic *apic =
35
		container_of(entry, struct acpi_madt_local_x2apic, header);
36 37

	if (!(apic->lapic_flags & ACPI_MADT_ENABLED))
38
		return -ENODEV;
39

40 41
	if (device_declaration && (apic->uid == acpi_id)) {
		*apic_id = apic->local_apic_id;
42
		return 0;
43 44
	}

45
	return -EINVAL;
46 47 48 49 50 51
}

static int map_lsapic_id(struct acpi_subtable_header *entry,
		int device_declaration, u32 acpi_id, int *apic_id)
{
	struct acpi_madt_local_sapic *lsapic =
52
		container_of(entry, struct acpi_madt_local_sapic, header);
53 54

	if (!(lsapic->lapic_flags & ACPI_MADT_ENABLED))
55
		return -ENODEV;
56 57

	if (device_declaration) {
58
		if ((entry->length < 16) || (lsapic->uid != acpi_id))
59
			return -EINVAL;
60
	} else if (lsapic->processor_id != acpi_id)
61
		return -EINVAL;
62

63
	*apic_id = (lsapic->id << 8) | lsapic->eid;
64
	return 0;
65 66 67 68 69
}

static int map_madt_entry(int type, u32 acpi_id)
{
	unsigned long madt_end, entry;
70 71
	static struct acpi_table_madt *madt;
	static int read_madt;
72 73
	int apic_id = -1;

74 75 76 77 78 79 80
	if (!read_madt) {
		if (ACPI_FAILURE(acpi_get_table(ACPI_SIG_MADT, 0,
					(struct acpi_table_header **)&madt)))
			madt = NULL;
		read_madt++;
	}

81 82 83 84 85 86 87 88 89 90 91 92 93
	if (!madt)
		return apic_id;

	entry = (unsigned long)madt;
	madt_end = entry + madt->header.length;

	/* Parse all entries looking for a match. */

	entry += sizeof(struct acpi_table_madt);
	while (entry + sizeof(struct acpi_subtable_header) < madt_end) {
		struct acpi_subtable_header *header =
			(struct acpi_subtable_header *)entry;
		if (header->type == ACPI_MADT_TYPE_LOCAL_APIC) {
94
			if (!map_lapic_id(header, acpi_id, &apic_id))
95 96
				break;
		} else if (header->type == ACPI_MADT_TYPE_LOCAL_X2APIC) {
97
			if (!map_x2apic_id(header, type, acpi_id, &apic_id))
98 99
				break;
		} else if (header->type == ACPI_MADT_TYPE_LOCAL_SAPIC) {
100
			if (!map_lsapic_id(header, type, acpi_id, &apic_id))
101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127
				break;
		}
		entry += header->length;
	}
	return apic_id;
}

static int map_mat_entry(acpi_handle handle, int type, u32 acpi_id)
{
	struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
	union acpi_object *obj;
	struct acpi_subtable_header *header;
	int apic_id = -1;

	if (ACPI_FAILURE(acpi_evaluate_object(handle, "_MAT", NULL, &buffer)))
		goto exit;

	if (!buffer.length || !buffer.pointer)
		goto exit;

	obj = buffer.pointer;
	if (obj->type != ACPI_TYPE_BUFFER ||
	    obj->buffer.length < sizeof(struct acpi_subtable_header)) {
		goto exit;
	}

	header = (struct acpi_subtable_header *)obj->buffer.pointer;
128
	if (header->type == ACPI_MADT_TYPE_LOCAL_APIC)
129
		map_lapic_id(header, acpi_id, &apic_id);
130
	else if (header->type == ACPI_MADT_TYPE_LOCAL_SAPIC)
131
		map_lsapic_id(header, type, acpi_id, &apic_id);
132
	else if (header->type == ACPI_MADT_TYPE_LOCAL_X2APIC)
133
		map_x2apic_id(header, type, acpi_id, &apic_id);
134 135

exit:
136
	kfree(buffer.pointer);
137 138 139
	return apic_id;
}

140
int acpi_get_apicid(acpi_handle handle, int type, u32 acpi_id)
141
{
142
	int apic_id;
143 144 145 146

	apic_id = map_mat_entry(handle, type, acpi_id);
	if (apic_id == -1)
		apic_id = map_madt_entry(type, acpi_id);
147 148 149 150 151 152 153 154 155 156

	return apic_id;
}

int acpi_map_cpuid(int apic_id, u32 acpi_id)
{
#ifdef CONFIG_SMP
	int i;
#endif

157 158 159 160 161 162 163 164 165
	if (apic_id == -1) {
		/*
		 * On UP processor, there is no _MAT or MADT table.
		 * So above apic_id is always set to -1.
		 *
		 * BIOS may define multiple CPU handles even for UP processor.
		 * For example,
		 *
		 * Scope (_PR)
166
		 * {
167 168 169 170 171 172
		 *     Processor (CPU0, 0x00, 0x00000410, 0x06) {}
		 *     Processor (CPU1, 0x01, 0x00000410, 0x06) {}
		 *     Processor (CPU2, 0x02, 0x00000410, 0x06) {}
		 *     Processor (CPU3, 0x03, 0x00000410, 0x06) {}
		 * }
		 *
173 174 175
		 * Ignores apic_id and always returns 0 for the processor
		 * handle with acpi id 0 if nr_cpu_ids is 1.
		 * This should be the case if SMP tables are not found.
176 177
		 * Return -1 for other CPU's handle.
		 */
178
		if (nr_cpu_ids <= 1 && acpi_id == 0)
179 180 181 182
			return acpi_id;
		else
			return apic_id;
	}
183

184
#ifdef CONFIG_SMP
185 186 187 188
	for_each_possible_cpu(i) {
		if (cpu_physical_id(i) == apic_id)
			return i;
	}
189 190 191 192 193
#else
	/* In UP kernel, only processor 0 is valid */
	if (apic_id == 0)
		return apic_id;
#endif
194 195
	return -1;
}
196 197 198 199 200 201 202 203 204

int acpi_get_cpuid(acpi_handle handle, int type, u32 acpi_id)
{
	int apic_id;

	apic_id = acpi_get_apicid(handle, type, acpi_id);

	return acpi_map_cpuid(apic_id, acpi_id);
}
205
EXPORT_SYMBOL_GPL(acpi_get_cpuid);