numa.c 6.6 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35
/*
 *  acpi_numa.c - ACPI NUMA support
 *
 *  Copyright (C) 2002 Takayoshi Kochi <t-kochi@bq.jp.nec.com>
 *
 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 2 of the License, or
 *  (at your option) any later version.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with this program; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 *
 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 *
 */
#include <linux/module.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/types.h>
#include <linux/errno.h>
#include <linux/acpi.h>
#include <acpi/acpi_bus.h>
#include <acpi/acmacros.h>

#define ACPI_NUMA	0x80000000
#define _COMPONENT	ACPI_NUMA
L
Len Brown 已提交
36
ACPI_MODULE_NAME("numa")
L
Linus Torvalds 已提交
37

38 39 40 41 42 43 44 45 46 47
static nodemask_t nodes_found_map = NODE_MASK_NONE;
#define PXM_INVAL	-1
#define NID_INVAL	-1

/* maps to convert between proximity domain and logical node ID */
int __cpuinitdata pxm_to_node_map[MAX_PXM_DOMAINS]
				= { [0 ... MAX_PXM_DOMAINS - 1] = NID_INVAL };
int __cpuinitdata node_to_pxm_map[MAX_NUMNODES]
				= { [0 ... MAX_NUMNODES - 1] = PXM_INVAL };

48
extern int __init acpi_table_parse_madt_family(char *id,
L
Len Brown 已提交
49 50 51 52
					       unsigned long madt_size,
					       int entry_id,
					       acpi_madt_entry_handler handler,
					       unsigned int max_entries);
L
Linus Torvalds 已提交
53

54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91
int __cpuinit pxm_to_node(int pxm)
{
	if (pxm < 0)
		return NID_INVAL;
	return pxm_to_node_map[pxm];
}

int __cpuinit node_to_pxm(int node)
{
	if (node < 0)
		return PXM_INVAL;
	return node_to_pxm_map[node];
}

int __cpuinit acpi_map_pxm_to_node(int pxm)
{
	int node = pxm_to_node_map[pxm];

	if (node < 0){
		if (nodes_weight(nodes_found_map) >= MAX_NUMNODES)
			return NID_INVAL;
		node = first_unset_node(nodes_found_map);
		pxm_to_node_map[pxm] = node;
		node_to_pxm_map[node] = pxm;
		node_set(node, nodes_found_map);
	}

	return node;
}

void __cpuinit acpi_unmap_pxm_to_node(int node)
{
	int pxm = node_to_pxm_map[node];
	pxm_to_node_map[pxm] = NID_INVAL;
	node_to_pxm_map[node] = PXM_INVAL;
	node_clear(node, nodes_found_map);
}

92
void __init acpi_table_print_srat_entry(struct acpi_subtable_header * header)
L
Linus Torvalds 已提交
93 94
{

L
Len Brown 已提交
95
	ACPI_FUNCTION_NAME("acpi_table_print_srat_entry");
L
Linus Torvalds 已提交
96 97 98 99 100 101

	if (!header)
		return;

	switch (header->type) {

102
	case ACPI_SRAT_TYPE_CPU_AFFINITY:
L
Linus Torvalds 已提交
103
#ifdef ACPI_DEBUG_OUTPUT
L
Len Brown 已提交
104
		{
105 106
			struct acpi_srat_cpu_affinity *p =
			    (struct acpi_srat_cpu_affinity *)header;
L
Len Brown 已提交
107 108
			ACPI_DEBUG_PRINT((ACPI_DB_INFO,
					  "SRAT Processor (id[0x%02x] eid[0x%02x]) in proximity domain %d %s\n",
109 110 111 112
					  p->apic_id, p->local_sapic_eid,
					  p->proximity_domain_lo,
					  (p->flags & ACPI_SRAT_CPU_ENABLED)?
					  "enabled" : "disabled"));
L
Len Brown 已提交
113 114
		}
#endif				/* ACPI_DEBUG_OUTPUT */
L
Linus Torvalds 已提交
115 116
		break;

117
	case ACPI_SRAT_TYPE_MEMORY_AFFINITY:
L
Linus Torvalds 已提交
118
#ifdef ACPI_DEBUG_OUTPUT
L
Len Brown 已提交
119
		{
120 121
			struct acpi_srat_mem_affinity *p =
			    (struct acpi_srat_mem_affinity *)header;
L
Len Brown 已提交
122
			ACPI_DEBUG_PRINT((ACPI_DB_INFO,
123 124 125
					  "SRAT Memory (0x%lx length 0x%lx type 0x%x) in proximity domain %d %s%s\n",
					  (unsigned long)p->base_address,
					  (unsigned long)p->length,
L
Len Brown 已提交
126
					  p->memory_type, p->proximity_domain,
127 128 129 130
					  (p->flags & ACPI_SRAT_MEM_ENABLED)?
					  "enabled" : "disabled",
					  (p->flags & ACPI_SRAT_MEM_HOT_PLUGGABLE)?
					  " hot-pluggable" : ""));
L
Len Brown 已提交
131 132
		}
#endif				/* ACPI_DEBUG_OUTPUT */
L
Linus Torvalds 已提交
133 134 135
		break;

	default:
L
Len Brown 已提交
136 137 138
		printk(KERN_WARNING PREFIX
		       "Found unsupported SRAT entry (type = 0x%x)\n",
		       header->type);
L
Linus Torvalds 已提交
139 140 141 142
		break;
	}
}

143
static int __init acpi_parse_slit(struct acpi_table_header *table)
L
Linus Torvalds 已提交
144
{
L
Len Brown 已提交
145 146
	struct acpi_table_slit *slit;
	u32 localities;
L
Linus Torvalds 已提交
147

148
	if (!table)
L
Linus Torvalds 已提交
149 150
		return -EINVAL;

151
	slit = (struct acpi_table_slit *)table;
L
Linus Torvalds 已提交
152 153

	/* downcast just for %llu vs %lu for i386/ia64  */
154
	localities = (u32) slit->locality_count;
L
Linus Torvalds 已提交
155 156 157 158 159 160 161

	acpi_numa_slit_init(slit);

	return 0;
}

static int __init
162
acpi_parse_processor_affinity(struct acpi_subtable_header * header,
L
Len Brown 已提交
163
			      const unsigned long end)
L
Linus Torvalds 已提交
164
{
165
	struct acpi_srat_cpu_affinity *processor_affinity;
L
Linus Torvalds 已提交
166

167
	processor_affinity = (struct acpi_srat_cpu_affinity *)header;
L
Linus Torvalds 已提交
168 169 170 171 172 173 174 175 176 177 178 179
	if (!processor_affinity)
		return -EINVAL;

	acpi_table_print_srat_entry(header);

	/* let architecture-dependent part to do it */
	acpi_numa_processor_affinity_init(processor_affinity);

	return 0;
}

static int __init
180
acpi_parse_memory_affinity(struct acpi_subtable_header * header,
L
Len Brown 已提交
181
			   const unsigned long end)
L
Linus Torvalds 已提交
182
{
183
	struct acpi_srat_mem_affinity *memory_affinity;
L
Linus Torvalds 已提交
184

185
	memory_affinity = (struct acpi_srat_mem_affinity *)header;
L
Linus Torvalds 已提交
186 187 188 189 190 191 192 193 194 195 196
	if (!memory_affinity)
		return -EINVAL;

	acpi_table_print_srat_entry(header);

	/* let architecture-dependent part to do it */
	acpi_numa_memory_affinity_init(memory_affinity);

	return 0;
}

197
static int __init acpi_parse_srat(struct acpi_table_header *table)
L
Linus Torvalds 已提交
198
{
L
Len Brown 已提交
199
	struct acpi_table_srat *srat;
L
Linus Torvalds 已提交
200

201
	if (!table)
L
Linus Torvalds 已提交
202 203
		return -EINVAL;

204
	srat = (struct acpi_table_srat *)table;
L
Linus Torvalds 已提交
205 206 207 208 209

	return 0;
}

int __init
210
acpi_table_parse_srat(enum acpi_srat_type id,
L
Len Brown 已提交
211
		      acpi_madt_entry_handler handler, unsigned int max_entries)
L
Linus Torvalds 已提交
212
{
213
	return acpi_table_parse_madt_family(ACPI_SIG_SRAT,
L
Len Brown 已提交
214 215
					    sizeof(struct acpi_table_srat), id,
					    handler, max_entries);
L
Linus Torvalds 已提交
216 217
}

L
Len Brown 已提交
218
int __init acpi_numa_init(void)
L
Linus Torvalds 已提交
219
{
L
Len Brown 已提交
220
	int result;
L
Linus Torvalds 已提交
221 222

	/* SRAT: Static Resource Affinity Table */
223
	result = acpi_table_parse(ACPI_SIG_SRAT, acpi_parse_srat);
L
Linus Torvalds 已提交
224 225

	if (result > 0) {
226
		result = acpi_table_parse_srat(ACPI_SRAT_TYPE_CPU_AFFINITY,
L
Linus Torvalds 已提交
227 228
					       acpi_parse_processor_affinity,
					       NR_CPUS);
229
		result = acpi_table_parse_srat(ACPI_SRAT_TYPE_MEMORY_AFFINITY, acpi_parse_memory_affinity, NR_NODE_MEMBLKS);	// IA64 specific
L
Linus Torvalds 已提交
230 231 232
	}

	/* SLIT: System Locality Information Table */
233
	result = acpi_table_parse(ACPI_SIG_SLIT, acpi_parse_slit);
L
Linus Torvalds 已提交
234 235 236 237 238

	acpi_numa_arch_fixup();
	return 0;
}

L
Len Brown 已提交
239
int acpi_get_pxm(acpi_handle h)
L
Linus Torvalds 已提交
240 241 242 243 244 245 246 247 248 249
{
	unsigned long pxm;
	acpi_status status;
	acpi_handle handle;
	acpi_handle phandle = h;

	do {
		handle = phandle;
		status = acpi_evaluate_integer(handle, "_PXM", NULL, &pxm);
		if (ACPI_SUCCESS(status))
250
			return pxm;
L
Linus Torvalds 已提交
251
		status = acpi_get_parent(handle, &phandle);
L
Len Brown 已提交
252
	} while (ACPI_SUCCESS(status));
L
Linus Torvalds 已提交
253 254 255
	return -1;
}
EXPORT_SYMBOL(acpi_get_pxm);
256 257 258 259 260 261 262 263 264

int acpi_get_node(acpi_handle *handle)
{
	int pxm, node = -1;

	pxm = acpi_get_pxm(handle);
	if (pxm >= 0)
		node = acpi_map_pxm_to_node(pxm);

265
	return node;
266 267
}
EXPORT_SYMBOL(acpi_get_node);