mpparse.c 25.5 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2
 *	Intel Multiprocessor Specification 1.1 and 1.4
L
Linus Torvalds 已提交
3 4
 *	compliant MP-table parsing routines.
 *
5
 *	(c) 1995 Alan Cox, Building #3 <alan@lxorguk.ukuu.org.uk>
I
Ingo Molnar 已提交
6
 *	(c) 1998, 1999, 2000, 2009 Ingo Molnar <mingo@redhat.com>
7
 *      (c) 2008 Alexey Starikovskiy <astarikovskiy@suse.de>
L
Linus Torvalds 已提交
8 9 10 11 12 13 14 15 16
 */

#include <linux/mm.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <linux/bootmem.h>
#include <linux/kernel_stat.h>
#include <linux/mc146818rtc.h>
#include <linux/bitops.h>
17 18
#include <linux/acpi.h>
#include <linux/module.h>
19
#include <linux/smp.h>
L
Linus Torvalds 已提交
20 21 22

#include <asm/mtrr.h>
#include <asm/mpspec.h>
23
#include <asm/pgalloc.h>
L
Linus Torvalds 已提交
24
#include <asm/io_apic.h>
25
#include <asm/proto.h>
26
#include <asm/bios_ebda.h>
Y
Yinghai Lu 已提交
27 28
#include <asm/e820.h>
#include <asm/trampoline.h>
Y
Yinghai Lu 已提交
29
#include <asm/setup.h>
30
#include <asm/smp.h>
L
Linus Torvalds 已提交
31

I
Ingo Molnar 已提交
32
#include <asm/apic.h>
L
Linus Torvalds 已提交
33 34 35 36 37 38 39 40 41 42 43 44 45 46
/*
 * Checksum an MP configuration block.
 */

static int __init mpf_checksum(unsigned char *mp, int len)
{
	int sum = 0;

	while (len--)
		sum += *mp++;

	return sum & 0xFF;
}

47
static void __init MP_processor_info(struct mpc_cpu *m)
48 49
{
	int apicid;
50
	char *bootup_cpu = "";
51

52
	if (!(m->cpuflag & CPU_ENABLED)) {
53
		disabled_cpus++;
L
Linus Torvalds 已提交
54
		return;
55
	}
56 57 58

	if (x86_quirks->mpc_apic_id)
		apicid = x86_quirks->mpc_apic_id(m);
59
	else
60
		apicid = m->apicid;
61

62
	if (m->cpuflag & CPU_BOOTPROCESSOR) {
63
		bootup_cpu = " (Bootup-CPU)";
64
		boot_cpu_physical_apicid = m->apicid;
L
Linus Torvalds 已提交
65 66
	}

67 68
	printk(KERN_INFO "Processor #%d%s\n", m->apicid, bootup_cpu);
	generic_processor_info(apicid, m->apicver);
L
Linus Torvalds 已提交
69 70
}

71
#ifdef CONFIG_X86_IO_APIC
72
static void __init MP_bus_info(struct mpc_bus *m)
L
Linus Torvalds 已提交
73 74
{
	char str[7];
75
	memcpy(str, m->bustype, 6);
L
Linus Torvalds 已提交
76 77
	str[6] = 0;

78 79 80
	if (x86_quirks->mpc_oem_bus_info)
		x86_quirks->mpc_oem_bus_info(m, str);
	else
81
		apic_printk(APIC_VERBOSE, "Bus #%d is %s\n", m->busid, str);
L
Linus Torvalds 已提交
82

83
#if MAX_MP_BUSSES < 256
84
	if (m->busid >= MAX_MP_BUSSES) {
85
		printk(KERN_WARNING "MP table busid value (%d) for bustype %s "
86
		       " is too large, max. supported is %d\n",
87
		       m->busid, str, MAX_MP_BUSSES - 1);
88 89
		return;
	}
90
#endif
91

92
	if (strncmp(str, BUSTYPE_ISA, sizeof(BUSTYPE_ISA) - 1) == 0) {
93
		set_bit(m->busid, mp_bus_not_pci);
94
#if defined(CONFIG_EISA) || defined(CONFIG_MCA)
95
		mp_bus_id_to_type[m->busid] = MP_BUS_ISA;
96 97
#endif
	} else if (strncmp(str, BUSTYPE_PCI, sizeof(BUSTYPE_PCI) - 1) == 0) {
98 99 100
		if (x86_quirks->mpc_oem_pci_bus)
			x86_quirks->mpc_oem_pci_bus(m);

101
		clear_bit(m->busid, mp_bus_not_pci);
102
#if defined(CONFIG_EISA) || defined(CONFIG_MCA)
103
		mp_bus_id_to_type[m->busid] = MP_BUS_PCI;
104
	} else if (strncmp(str, BUSTYPE_EISA, sizeof(BUSTYPE_EISA) - 1) == 0) {
105
		mp_bus_id_to_type[m->busid] = MP_BUS_EISA;
106
	} else if (strncmp(str, BUSTYPE_MCA, sizeof(BUSTYPE_MCA) - 1) == 0) {
107
		mp_bus_id_to_type[m->busid] = MP_BUS_MCA;
108
#endif
109 110
	} else
		printk(KERN_WARNING "Unknown bustype %s - ignoring\n", str);
L
Linus Torvalds 已提交
111
}
112

113 114 115 116 117 118 119 120 121 122 123 124 125 126 127
static int bad_ioapic(unsigned long address)
{
	if (nr_ioapics >= MAX_IO_APICS) {
		printk(KERN_ERR "ERROR: Max # of I/O APICs (%d) exceeded "
		       "(found %d)\n", MAX_IO_APICS, nr_ioapics);
		panic("Recompile kernel with bigger MAX_IO_APICS!\n");
	}
	if (!address) {
		printk(KERN_ERR "WARNING: Bogus (zero) I/O APIC address"
		       " found in table, skipping!\n");
		return 1;
	}
	return 0;
}

128
static void __init MP_ioapic_info(struct mpc_ioapic *m)
L
Linus Torvalds 已提交
129
{
130
	if (!(m->flags & MPC_APIC_USABLE))
L
Linus Torvalds 已提交
131 132
		return;

133
	printk(KERN_INFO "I/O APIC #%d Version %d at 0x%X.\n",
134
	       m->apicid, m->apicver, m->apicaddr);
135

136
	if (bad_ioapic(m->apicaddr))
L
Linus Torvalds 已提交
137
		return;
138

139 140 141 142 143
	mp_ioapics[nr_ioapics].apicaddr = m->apicaddr;
	mp_ioapics[nr_ioapics].apicid = m->apicid;
	mp_ioapics[nr_ioapics].type = m->type;
	mp_ioapics[nr_ioapics].apicver = m->apicver;
	mp_ioapics[nr_ioapics].flags = m->flags;
L
Linus Torvalds 已提交
144 145 146
	nr_ioapics++;
}

147
static void print_MP_intsrc_info(struct mpc_intsrc *m)
L
Linus Torvalds 已提交
148
{
149
	apic_printk(APIC_VERBOSE, "Int: type %d, pol %d, trig %d, bus %02x,"
L
Linus Torvalds 已提交
150
		" IRQ %02x, APIC ID %x, APIC INT %02x\n",
151 152
		m->irqtype, m->irqflag & 3, (m->irqflag >> 2) & 3, m->srcbus,
		m->srcbusirq, m->dstapic, m->dstirq);
Y
Yinghai Lu 已提交
153 154
}

155
static void __init print_mp_irq_info(struct mpc_intsrc *mp_irq)
Y
Yinghai Lu 已提交
156
{
157
	apic_printk(APIC_VERBOSE, "Int: type %d, pol %d, trig %d, bus %02x,"
Y
Yinghai Lu 已提交
158
		" IRQ %02x, APIC ID %x, APIC INT %02x\n",
159 160 161
		mp_irq->irqtype, mp_irq->irqflag & 3,
		(mp_irq->irqflag >> 2) & 3, mp_irq->srcbus,
		mp_irq->srcbusirq, mp_irq->dstapic, mp_irq->dstirq);
Y
Yinghai Lu 已提交
162 163
}

164
static void __init assign_to_mp_irq(struct mpc_intsrc *m,
165
				    struct mpc_intsrc *mp_irq)
Y
Yinghai Lu 已提交
166
{
167 168 169 170 171 172 173
	mp_irq->dstapic = m->dstapic;
	mp_irq->type = m->type;
	mp_irq->irqtype = m->irqtype;
	mp_irq->irqflag = m->irqflag;
	mp_irq->srcbus = m->srcbus;
	mp_irq->srcbusirq = m->srcbusirq;
	mp_irq->dstirq = m->dstirq;
Y
Yinghai Lu 已提交
174 175
}

176
static void __init assign_to_mpc_intsrc(struct mpc_intsrc *mp_irq,
177
					struct mpc_intsrc *m)
Y
Yinghai Lu 已提交
178
{
179 180 181 182 183 184 185
	m->dstapic = mp_irq->dstapic;
	m->type = mp_irq->type;
	m->irqtype = mp_irq->irqtype;
	m->irqflag = mp_irq->irqflag;
	m->srcbus = mp_irq->srcbus;
	m->srcbusirq = mp_irq->srcbusirq;
	m->dstirq = mp_irq->dstirq;
Y
Yinghai Lu 已提交
186 187
}

188
static int __init mp_irq_mpc_intsrc_cmp(struct mpc_intsrc *mp_irq,
189
					struct mpc_intsrc *m)
Y
Yinghai Lu 已提交
190
{
191
	if (mp_irq->dstapic != m->dstapic)
Y
Yinghai Lu 已提交
192
		return 1;
193
	if (mp_irq->type != m->type)
Y
Yinghai Lu 已提交
194
		return 2;
195
	if (mp_irq->irqtype != m->irqtype)
Y
Yinghai Lu 已提交
196
		return 3;
197
	if (mp_irq->irqflag != m->irqflag)
Y
Yinghai Lu 已提交
198
		return 4;
199
	if (mp_irq->srcbus != m->srcbus)
Y
Yinghai Lu 已提交
200
		return 5;
201
	if (mp_irq->srcbusirq != m->srcbusirq)
Y
Yinghai Lu 已提交
202
		return 6;
203
	if (mp_irq->dstirq != m->dstirq)
Y
Yinghai Lu 已提交
204 205 206 207 208
		return 7;

	return 0;
}

209
static void __init MP_intsrc_info(struct mpc_intsrc *m)
Y
Yinghai Lu 已提交
210 211 212 213 214
{
	int i;

	print_MP_intsrc_info(m);

215 216 217 218
	for (i = 0; i < mp_irq_entries; i++) {
		if (!mp_irq_mpc_intsrc_cmp(&mp_irqs[i], m))
			return;
	}
Y
Yinghai Lu 已提交
219 220

	assign_to_mp_irq(m, &mp_irqs[mp_irq_entries]);
L
Linus Torvalds 已提交
221 222 223
	if (++mp_irq_entries == MAX_IRQ_SOURCES)
		panic("Max # of irq sources exceeded!!\n");
}
224 225 226 227 228
#else /* CONFIG_X86_IO_APIC */
static inline void __init MP_bus_info(struct mpc_bus *m) {}
static inline void __init MP_ioapic_info(struct mpc_ioapic *m) {}
static inline void __init MP_intsrc_info(struct mpc_intsrc *m) {}
#endif /* CONFIG_X86_IO_APIC */
L
Linus Torvalds 已提交
229

230

231
static void __init MP_lintsrc_info(struct mpc_lintsrc *m)
L
Linus Torvalds 已提交
232
{
233
	apic_printk(APIC_VERBOSE, "Lint: type %d, pol %d, trig %d, bus %02x,"
L
Linus Torvalds 已提交
234
		" IRQ %02x, APIC ID %x, APIC LINT %02x\n",
235 236
		m->irqtype, m->irqflag & 3, (m->irqflag >> 2) & 3, m->srcbusid,
		m->srcbusirq, m->destapic, m->destapiclint);
L
Linus Torvalds 已提交
237 238 239 240 241 242
}

/*
 * Read/parse the MPC
 */

243
static int __init smp_check_mpc(struct mpc_table *mpc, char *oem, char *str)
L
Linus Torvalds 已提交
244 245
{

246
	if (memcmp(mpc->signature, MPC_SIGNATURE, 4)) {
247
		printk(KERN_ERR "MPTABLE: bad signature [%c%c%c%c]!\n",
248 249
		       mpc->signature[0], mpc->signature[1],
		       mpc->signature[2], mpc->signature[3]);
L
Linus Torvalds 已提交
250 251
		return 0;
	}
252
	if (mpf_checksum((unsigned char *)mpc, mpc->length)) {
253
		printk(KERN_ERR "MPTABLE: checksum error!\n");
L
Linus Torvalds 已提交
254 255
		return 0;
	}
256
	if (mpc->spec != 0x01 && mpc->spec != 0x04) {
257
		printk(KERN_ERR "MPTABLE: bad table version (%d)!!\n",
258
		       mpc->spec);
L
Linus Torvalds 已提交
259 260
		return 0;
	}
261
	if (!mpc->lapic) {
262
		printk(KERN_ERR "MPTABLE: null local APIC address!\n");
L
Linus Torvalds 已提交
263 264
		return 0;
	}
265
	memcpy(oem, mpc->oem, 8);
266
	oem[8] = 0;
267
	printk(KERN_INFO "MPTABLE: OEM ID: %s\n", oem);
L
Linus Torvalds 已提交
268

269
	memcpy(str, mpc->productid, 12);
270
	str[12] = 0;
L
Linus Torvalds 已提交
271

272
	printk(KERN_INFO "MPTABLE: Product ID: %s\n", str);
L
Linus Torvalds 已提交
273

274
	printk(KERN_INFO "MPTABLE: APIC at: 0x%X\n", mpc->lapic);
L
Linus Torvalds 已提交
275

Y
Yinghai Lu 已提交
276 277 278
	return 1;
}

279 280 281 282 283 284
static void skip_entry(unsigned char **ptr, int *count, int size)
{
	*ptr += size;
	*count += size;
}

285 286 287 288 289 290 291 292
static void __init smp_dump_mptable(struct mpc_table *mpc, unsigned char *mpt)
{
	printk(KERN_ERR "Your mptable is wrong, contact your HW vendor!\n"
		"type %x\n", *mpt);
	print_hex_dump(KERN_ERR, "  ", DUMP_PREFIX_ADDRESS, 16,
			1, mpc, mpc->length, 1);
}

293
static int __init smp_read_mpc(struct mpc_table *mpc, unsigned early)
Y
Yinghai Lu 已提交
294 295 296 297 298 299 300 301 302 303 304
{
	char str[16];
	char oem[10];

	int count = sizeof(*mpc);
	unsigned char *mpt = ((unsigned char *)mpc) + count;

	if (!smp_check_mpc(mpc, oem, str))
		return 0;

#ifdef CONFIG_X86_32
305
	generic_mps_oem_check(mpc, oem, str);
Y
Yinghai Lu 已提交
306
#endif
307
	/* save the local APIC address, it might be non-default */
L
Linus Torvalds 已提交
308
	if (!acpi_lapic)
309
		mp_lapic_addr = mpc->lapic;
L
Linus Torvalds 已提交
310

311 312 313
	if (early)
		return 1;

314 315 316
	if (mpc->oemptr && x86_quirks->smp_read_mpc_oem) {
		struct mpc_oemtable *oem_table = (void *)(long)mpc->oemptr;
		x86_quirks->smp_read_mpc_oem(oem_table, mpc->oemsize);
317 318
	}

L
Linus Torvalds 已提交
319
	/*
320
	 *      Now process the configuration blocks.
L
Linus Torvalds 已提交
321
	 */
322 323 324
	if (x86_quirks->mpc_record)
		*x86_quirks->mpc_record = 0;

325
	while (count < mpc->length) {
326 327
		switch (*mpt) {
		case MP_PROCESSOR:
328 329
			/* ACPI may have already provided this data */
			if (!acpi_lapic)
330
				MP_processor_info((struct mpc_cpu *)mpt);
331 332
			skip_entry(&mpt, &count, sizeof(struct mpc_cpu));
			break;
333
		case MP_BUS:
334
			MP_bus_info((struct mpc_bus *)mpt);
335 336
			skip_entry(&mpt, &count, sizeof(struct mpc_bus));
			break;
337
		case MP_IOAPIC:
338
			MP_ioapic_info((struct mpc_ioapic *)mpt);
339 340
			skip_entry(&mpt, &count, sizeof(struct mpc_ioapic));
			break;
341
		case MP_INTSRC:
342
			MP_intsrc_info((struct mpc_intsrc *)mpt);
343 344
			skip_entry(&mpt, &count, sizeof(struct mpc_intsrc));
			break;
345
		case MP_LINTSRC:
346
			MP_lintsrc_info((struct mpc_lintsrc *)mpt);
347 348
			skip_entry(&mpt, &count, sizeof(struct mpc_lintsrc));
			break;
349
		default:
Y
Yinghai Lu 已提交
350
			/* wrong mptable */
351
			smp_dump_mptable(mpc, mpt);
352
			count = mpc->length;
Y
Yinghai Lu 已提交
353
			break;
L
Linus Torvalds 已提交
354
		}
355 356
		if (x86_quirks->mpc_record)
			(*x86_quirks->mpc_record)++;
L
Linus Torvalds 已提交
357
	}
358

359 360
#ifdef CONFIG_X86_BIGSMP
	generic_bigsmp_probe();
361 362
#endif

363 364 365
	if (apic->setup_apic_routing)
		apic->setup_apic_routing();

L
Linus Torvalds 已提交
366
	if (!num_processors)
367
		printk(KERN_ERR "MPTABLE: no processors registered!\n");
L
Linus Torvalds 已提交
368 369 370
	return num_processors;
}

371 372
#ifdef CONFIG_X86_IO_APIC

L
Linus Torvalds 已提交
373 374 375 376 377 378 379 380 381 382
static int __init ELCR_trigger(unsigned int irq)
{
	unsigned int port;

	port = 0x4d0 + (irq >> 3);
	return (inb(port) >> (irq & 7)) & 1;
}

static void __init construct_default_ioirq_mptable(int mpc_default_type)
{
383
	struct mpc_intsrc intsrc;
L
Linus Torvalds 已提交
384 385 386
	int i;
	int ELCR_fallback = 0;

387 388 389
	intsrc.type = MP_INTSRC;
	intsrc.irqflag = 0;	/* conforming */
	intsrc.srcbus = 0;
390
	intsrc.dstapic = mp_ioapics[0].apicid;
L
Linus Torvalds 已提交
391

392
	intsrc.irqtype = mp_INT;
L
Linus Torvalds 已提交
393 394 395 396 397 398 399 400 401 402

	/*
	 *  If true, we have an ISA/PCI system with no IRQ entries
	 *  in the MP table. To prevent the PCI interrupts from being set up
	 *  incorrectly, we try to use the ELCR. The sanity check to see if
	 *  there is good ELCR data is very simple - IRQ0, 1, 2 and 13 can
	 *  never be level sensitive, so we simply see if the ELCR agrees.
	 *  If it does, we assume it's valid.
	 */
	if (mpc_default_type == 5) {
403 404
		printk(KERN_INFO "ISA/PCI bus type with no IRQ information... "
		       "falling back to ELCR\n");
L
Linus Torvalds 已提交
405

406 407 408 409
		if (ELCR_trigger(0) || ELCR_trigger(1) || ELCR_trigger(2) ||
		    ELCR_trigger(13))
			printk(KERN_ERR "ELCR contains invalid data... "
			       "not using ELCR\n");
L
Linus Torvalds 已提交
410
		else {
411 412
			printk(KERN_INFO
			       "Using ELCR to identify PCI interrupts\n");
L
Linus Torvalds 已提交
413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434
			ELCR_fallback = 1;
		}
	}

	for (i = 0; i < 16; i++) {
		switch (mpc_default_type) {
		case 2:
			if (i == 0 || i == 13)
				continue;	/* IRQ0 & IRQ13 not connected */
			/* fall through */
		default:
			if (i == 2)
				continue;	/* IRQ2 is never connected */
		}

		if (ELCR_fallback) {
			/*
			 *  If the ELCR indicates a level-sensitive interrupt, we
			 *  copy that information over to the MP table in the
			 *  irqflag field (level sensitive, active high polarity).
			 */
			if (ELCR_trigger(i))
435
				intsrc.irqflag = 13;
L
Linus Torvalds 已提交
436
			else
437
				intsrc.irqflag = 0;
L
Linus Torvalds 已提交
438 439
		}

440 441
		intsrc.srcbusirq = i;
		intsrc.dstirq = i ? i : 2;	/* IRQ0 to INTIN2 */
L
Linus Torvalds 已提交
442 443 444
		MP_intsrc_info(&intsrc);
	}

445 446 447
	intsrc.irqtype = mp_ExtINT;
	intsrc.srcbusirq = 0;
	intsrc.dstirq = 0;	/* 8259A to INTIN0 */
L
Linus Torvalds 已提交
448 449 450
	MP_intsrc_info(&intsrc);
}

451

452
static void __init construct_ioapic_table(int mpc_default_type)
L
Linus Torvalds 已提交
453
{
454
	struct mpc_ioapic ioapic;
455
	struct mpc_bus bus;
L
Linus Torvalds 已提交
456

457 458
	bus.type = MP_BUS;
	bus.busid = 0;
L
Linus Torvalds 已提交
459
	switch (mpc_default_type) {
460
	default:
461
		printk(KERN_ERR "???\nUnknown standard configuration %d\n",
462 463 464 465
		       mpc_default_type);
		/* fall through */
	case 1:
	case 5:
466
		memcpy(bus.bustype, "ISA   ", 6);
467 468 469 470
		break;
	case 2:
	case 6:
	case 3:
471
		memcpy(bus.bustype, "EISA  ", 6);
472 473 474
		break;
	case 4:
	case 7:
475
		memcpy(bus.bustype, "MCA   ", 6);
L
Linus Torvalds 已提交
476 477 478
	}
	MP_bus_info(&bus);
	if (mpc_default_type > 4) {
479 480
		bus.busid = 1;
		memcpy(bus.bustype, "PCI   ", 6);
L
Linus Torvalds 已提交
481 482 483
		MP_bus_info(&bus);
	}

484 485 486 487 488
	ioapic.type = MP_IOAPIC;
	ioapic.apicid = 2;
	ioapic.apicver = mpc_default_type > 4 ? 0x10 : 0x01;
	ioapic.flags = MPC_APIC_USABLE;
	ioapic.apicaddr = 0xFEC00000;
L
Linus Torvalds 已提交
489 490 491 492 493 494
	MP_ioapic_info(&ioapic);

	/*
	 * We set up most of the low 16 IO-APIC pins according to MPS rules.
	 */
	construct_default_ioirq_mptable(mpc_default_type);
495 496
}
#else
497
static inline void __init construct_ioapic_table(int mpc_default_type) { }
498
#endif
499 500 501

static inline void __init construct_default_ISA_mptable(int mpc_default_type)
{
502
	struct mpc_cpu processor;
503
	struct mpc_lintsrc lintsrc;
504 505 506 507 508 509 510 511 512 513 514
	int linttypes[2] = { mp_ExtINT, mp_NMI };
	int i;

	/*
	 * local APIC has default address
	 */
	mp_lapic_addr = APIC_DEFAULT_PHYS_BASE;

	/*
	 * 2 CPUs, numbered 0 & 1.
	 */
515
	processor.type = MP_PROCESSOR;
516
	/* Either an integrated APIC or a discrete 82489DX. */
517 518 519
	processor.apicver = mpc_default_type > 4 ? 0x10 : 0x01;
	processor.cpuflag = CPU_ENABLED;
	processor.cpufeature = (boot_cpu_data.x86 << 8) |
520
	    (boot_cpu_data.x86_model << 4) | boot_cpu_data.x86_mask;
521 522 523
	processor.featureflag = boot_cpu_data.x86_capability[0];
	processor.reserved[0] = 0;
	processor.reserved[1] = 0;
524
	for (i = 0; i < 2; i++) {
525
		processor.apicid = i;
526 527 528 529 530
		MP_processor_info(&processor);
	}

	construct_ioapic_table(mpc_default_type);

531 532 533 534 535
	lintsrc.type = MP_LINTSRC;
	lintsrc.irqflag = 0;		/* conforming */
	lintsrc.srcbusid = 0;
	lintsrc.srcbusirq = 0;
	lintsrc.destapic = MP_APIC_ALL;
L
Linus Torvalds 已提交
536
	for (i = 0; i < 2; i++) {
537 538
		lintsrc.irqtype = linttypes[i];
		lintsrc.destapiclint = i;
L
Linus Torvalds 已提交
539 540 541 542
		MP_lintsrc_info(&lintsrc);
	}
}

543
static struct mpf_intel *mpf_found;
L
Linus Torvalds 已提交
544

Y
Yinghai Lu 已提交
545 546 547 548 549 550 551 552 553 554 555 556 557
static unsigned long __init get_mpc_size(unsigned long physptr)
{
	struct mpc_table *mpc;
	unsigned long size;

	mpc = early_ioremap(physptr, PAGE_SIZE);
	size = mpc->length;
	early_iounmap(mpc, PAGE_SIZE);
	apic_printk(APIC_VERBOSE, "  mpc: %lx-%lx\n", physptr, physptr + size);

	return size;
}

558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606
static int __init check_physptr(struct mpf_intel *mpf, unsigned int early)
{
	struct mpc_table *mpc;
	unsigned long size;

	size = get_mpc_size(mpf->physptr);
	mpc = early_ioremap(mpf->physptr, size);
	/*
	 * Read the physical hardware table.  Anything here will
	 * override the defaults.
	 */
	if (!smp_read_mpc(mpc, early)) {
#ifdef CONFIG_X86_LOCAL_APIC
		smp_found_config = 0;
#endif
		printk(KERN_ERR "BIOS bug, MP table errors detected!...\n"
			"... disabling SMP support. (tell your hw vendor)\n");
		early_iounmap(mpc, size);
		return -1;
	}
	early_iounmap(mpc, size);

	if (early)
		return -1;

#ifdef CONFIG_X86_IO_APIC
	/*
	 * If there are no explicit MP IRQ entries, then we are
	 * broken.  We set up most of the low 16 IO-APIC pins to
	 * ISA defaults and hope it will work.
	 */
	if (!mp_irq_entries) {
		struct mpc_bus bus;

		printk(KERN_ERR "BIOS bug, no explicit IRQ entries, "
		       "using default mptable. (tell your hw vendor)\n");

		bus.type = MP_BUS;
		bus.busid = 0;
		memcpy(bus.bustype, "ISA   ", 6);
		MP_bus_info(&bus);

		construct_default_ioirq_mptable(0);
	}
#endif

	return 0;
}

L
Linus Torvalds 已提交
607 608 609
/*
 * Scan the memory blocks for an SMP configuration block.
 */
610
static void __init __get_smp_config(unsigned int early)
L
Linus Torvalds 已提交
611
{
612
	struct mpf_intel *mpf = mpf_found;
L
Linus Torvalds 已提交
613

614 615 616
	if (!mpf)
		return;

617 618
	if (acpi_lapic && early)
		return;
619

L
Linus Torvalds 已提交
620
	/*
621 622
	 * MPS doesn't support hyperthreading, aka only have
	 * thread 0 apic id in MPS table
L
Linus Torvalds 已提交
623
	 */
624
	if (acpi_lapic && acpi_ioapic)
L
Linus Torvalds 已提交
625 626
		return;

627 628 629 630
	if (x86_quirks->mach_get_smp_config) {
		if (x86_quirks->mach_get_smp_config(early))
			return;
	}
631

632
	printk(KERN_INFO "Intel MultiProcessor Specification v1.%d\n",
633
	       mpf->specification);
634
#if defined(CONFIG_X86_LOCAL_APIC) && defined(CONFIG_X86_32)
635
	if (mpf->feature2 & (1 << 7)) {
L
Linus Torvalds 已提交
636 637 638 639 640 641
		printk(KERN_INFO "    IMCR and PIC compatibility mode.\n");
		pic_mode = 1;
	} else {
		printk(KERN_INFO "    Virtual Wire compatibility mode.\n");
		pic_mode = 0;
	}
642
#endif
L
Linus Torvalds 已提交
643 644 645
	/*
	 * Now see if we need to read further.
	 */
646
	if (mpf->feature1 != 0) {
647 648 649 650 651 652 653
		if (early) {
			/*
			 * local APIC has default address
			 */
			mp_lapic_addr = APIC_DEFAULT_PHYS_BASE;
			return;
		}
L
Linus Torvalds 已提交
654

655
		printk(KERN_INFO "Default MP configuration #%d\n",
656 657
		       mpf->feature1);
		construct_default_ISA_mptable(mpf->feature1);
L
Linus Torvalds 已提交
658

659
	} else if (mpf->physptr) {
660
		if (check_physptr(mpf, early))
L
Linus Torvalds 已提交
661 662 663 664
			return;
	} else
		BUG();

665 666
	if (!early)
		printk(KERN_INFO "Processors: %d\n", num_processors);
L
Linus Torvalds 已提交
667 668 669 670 671
	/*
	 * Only use the first configuration found.
	 */
}

672 673 674 675 676 677 678 679 680 681
void __init early_get_smp_config(void)
{
	__get_smp_config(1);
}

void __init get_smp_config(void)
{
	__get_smp_config(0);
}

682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706
static void smp_reserve_bootmem(struct mpf_intel *mpf)
{
	unsigned long size = get_mpc_size(mpf->physptr);
#ifdef CONFIG_X86_32
	/*
	 * We cannot access to MPC table to compute table size yet,
	 * as only few megabytes from the bottom is mapped now.
	 * PC-9800's MPC table places on the very last of physical
	 * memory; so that simply reserving PAGE_SIZE from mpf->physptr
	 * yields BUG() in reserve_bootmem.
	 * also need to make sure physptr is below than max_low_pfn
	 * we don't need reserve the area above max_low_pfn
	 */
	unsigned long end = max_low_pfn * PAGE_SIZE;

	if (mpf->physptr < end) {
		if (mpf->physptr + size > end)
			size = end - mpf->physptr;
		reserve_bootmem_generic(mpf->physptr, size, BOOTMEM_DEFAULT);
	}
#else
	reserve_bootmem_generic(mpf->physptr, size, BOOTMEM_DEFAULT);
#endif
}

707 708
static int __init smp_scan_config(unsigned long base, unsigned long length,
				  unsigned reserve)
L
Linus Torvalds 已提交
709
{
710
	unsigned int *bp = phys_to_virt(base);
711
	struct mpf_intel *mpf;
L
Linus Torvalds 已提交
712

713 714
	apic_printk(APIC_VERBOSE, "Scan SMP from %p for %ld bytes.\n",
			bp, length);
715
	BUILD_BUG_ON(sizeof(*mpf) != 16);
L
Linus Torvalds 已提交
716 717

	while (length > 0) {
718
		mpf = (struct mpf_intel *)bp;
L
Linus Torvalds 已提交
719
		if ((*bp == SMP_MAGIC_IDENT) &&
720
		    (mpf->length == 1) &&
721
		    !mpf_checksum((unsigned char *)bp, 16) &&
722 723
		    ((mpf->specification == 1)
		     || (mpf->specification == 4))) {
724
#ifdef CONFIG_X86_LOCAL_APIC
L
Linus Torvalds 已提交
725
			smp_found_config = 1;
726
#endif
727
			mpf_found = mpf;
728

729 730
			printk(KERN_INFO "found SMP MP-table at [%p] %llx\n",
			       mpf, (u64)virt_to_phys(mpf));
731 732 733

			if (!reserve)
				return 1;
734
			reserve_bootmem_generic(virt_to_phys(mpf), sizeof(*mpf),
735
						BOOTMEM_DEFAULT);
736 737
			if (mpf->physptr)
				smp_reserve_bootmem(mpf);
L
Linus Torvalds 已提交
738

739
			return 1;
L
Linus Torvalds 已提交
740 741 742 743 744 745 746
		}
		bp += 4;
		length -= 16;
	}
	return 0;
}

747
static void __init __find_smp_config(unsigned int reserve)
L
Linus Torvalds 已提交
748 749 750
{
	unsigned int address;

Y
Yinghai Lu 已提交
751 752
	if (x86_quirks->mach_find_smp_config) {
		if (x86_quirks->mach_find_smp_config(reserve))
753 754
			return;
	}
L
Linus Torvalds 已提交
755 756 757 758 759 760 761 762
	/*
	 * FIXME: Linux assumes you have 640K of base ram..
	 * this continues the error...
	 *
	 * 1) Scan the bottom 1K for a signature
	 * 2) Scan the top 1K of base RAM
	 * 3) Scan the 64K of bios
	 */
763 764 765
	if (smp_scan_config(0x0, 0x400, reserve) ||
	    smp_scan_config(639 * 0x400, 0x400, reserve) ||
	    smp_scan_config(0xF0000, 0x10000, reserve))
L
Linus Torvalds 已提交
766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785
		return;
	/*
	 * If it is an SMP machine we should know now, unless the
	 * configuration is in an EISA/MCA bus machine with an
	 * extended bios data area.
	 *
	 * there is a real-mode segmented pointer pointing to the
	 * 4K EBDA area at 0x40E, calculate and scan it here.
	 *
	 * NOTE! There are Linux loaders that will corrupt the EBDA
	 * area, and as such this kind of SMP config may be less
	 * trustworthy, simply because the SMP table may have been
	 * stomped on during early boot. These loaders are buggy and
	 * should be fixed.
	 *
	 * MP1.4 SPEC states to only scan first 1K of 4K EBDA.
	 */

	address = get_bios_ebda();
	if (address)
786 787 788 789 790 791 792 793 794 795 796
		smp_scan_config(address, 0x400, reserve);
}

void __init early_find_smp_config(void)
{
	__find_smp_config(0);
}

void __init find_smp_config(void)
{
	__find_smp_config(1);
L
Linus Torvalds 已提交
797
}
Y
Yinghai Lu 已提交
798 799 800 801

#ifdef CONFIG_X86_IO_APIC
static u8 __initdata irq_used[MAX_IRQ_SOURCES];

802
static int  __init get_MP_intsrc_index(struct mpc_intsrc *m)
Y
Yinghai Lu 已提交
803 804 805
{
	int i;

806
	if (m->irqtype != mp_INT)
Y
Yinghai Lu 已提交
807 808
		return 0;

809
	if (m->irqflag != 0x0f)
Y
Yinghai Lu 已提交
810 811 812 813 814
		return 0;

	/* not legacy */

	for (i = 0; i < mp_irq_entries; i++) {
815
		if (mp_irqs[i].irqtype != mp_INT)
Y
Yinghai Lu 已提交
816 817
			continue;

818
		if (mp_irqs[i].irqflag != 0x0f)
Y
Yinghai Lu 已提交
819 820
			continue;

821
		if (mp_irqs[i].srcbus != m->srcbus)
Y
Yinghai Lu 已提交
822
			continue;
823
		if (mp_irqs[i].srcbusirq != m->srcbusirq)
Y
Yinghai Lu 已提交
824 825 826 827 828 829 830 831 832 833 834 835 836 837 838
			continue;
		if (irq_used[i]) {
			/* already claimed */
			return -2;
		}
		irq_used[i] = 1;
		return i;
	}

	/* not found */
	return -1;
}

#define SPARE_SLOT_NUM 20

839
static struct mpc_intsrc __initdata *m_spare[SPARE_SLOT_NUM];
840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890

static void check_irq_src(struct mpc_intsrc *m, int *nr_m_spare)
{
	int i;

	apic_printk(APIC_VERBOSE, "OLD ");
	print_MP_intsrc_info(m);

	i = get_MP_intsrc_index(m);
	if (i > 0) {
		assign_to_mpc_intsrc(&mp_irqs[i], m);
		apic_printk(APIC_VERBOSE, "NEW ");
		print_mp_irq_info(&mp_irqs[i]);
		return;
	}
	if (!i) {
		/* legacy, do nothing */
		return;
	}
	if (*nr_m_spare < SPARE_SLOT_NUM) {
		/*
		 * not found (-1), or duplicated (-2) are invalid entries,
		 * we need to use the slot later
		 */
		m_spare[*nr_m_spare] = m;
		*nr_m_spare += 1;
	}
}
#else /* CONFIG_X86_IO_APIC */
static inline void check_irq_src(struct mpc_intsrc *m, int *nr_m_spare) {}
#endif /* CONFIG_X86_IO_APIC */

static int check_slot(unsigned long mpc_new_phys, unsigned long mpc_new_length,
		      int count)
{
	if (!mpc_new_phys) {
		pr_info("No spare slots, try to append...take your risk, "
			"new mpc_length %x\n", count);
	} else {
		if (count <= mpc_new_length)
			pr_info("No spare slots, try to append..., "
				"new mpc_length %x\n", count);
		else {
			pr_err("mpc_new_length %lx is too small\n",
				mpc_new_length);
			return -1;
		}
	}

	return 0;
}
Y
Yinghai Lu 已提交
891

892
static int  __init replace_intsrc_all(struct mpc_table *mpc,
Y
Yinghai Lu 已提交
893 894 895 896 897 898 899
					unsigned long mpc_new_phys,
					unsigned long mpc_new_length)
{
#ifdef CONFIG_X86_IO_APIC
	int i;
#endif
	int count = sizeof(*mpc);
900
	int nr_m_spare = 0;
Y
Yinghai Lu 已提交
901 902
	unsigned char *mpt = ((unsigned char *)mpc) + count;

903 904
	printk(KERN_INFO "mpc_length %x\n", mpc->length);
	while (count < mpc->length) {
Y
Yinghai Lu 已提交
905 906
		switch (*mpt) {
		case MP_PROCESSOR:
907 908
			skip_entry(&mpt, &count, sizeof(struct mpc_cpu));
			break;
Y
Yinghai Lu 已提交
909
		case MP_BUS:
910 911
			skip_entry(&mpt, &count, sizeof(struct mpc_bus));
			break;
Y
Yinghai Lu 已提交
912
		case MP_IOAPIC:
913 914
			skip_entry(&mpt, &count, sizeof(struct mpc_ioapic));
			break;
Y
Yinghai Lu 已提交
915
		case MP_INTSRC:
916
			check_irq_src((struct mpc_intsrc *)mpt, &nr_m_spare);
917 918
			skip_entry(&mpt, &count, sizeof(struct mpc_intsrc));
			break;
Y
Yinghai Lu 已提交
919
		case MP_LINTSRC:
920 921
			skip_entry(&mpt, &count, sizeof(struct mpc_lintsrc));
			break;
Y
Yinghai Lu 已提交
922 923
		default:
			/* wrong mptable */
924
			smp_dump_mptable(mpc, mpt);
Y
Yinghai Lu 已提交
925 926 927 928 929 930 931 932 933
			goto out;
		}
	}

#ifdef CONFIG_X86_IO_APIC
	for (i = 0; i < mp_irq_entries; i++) {
		if (irq_used[i])
			continue;

934
		if (mp_irqs[i].irqtype != mp_INT)
Y
Yinghai Lu 已提交
935 936
			continue;

937
		if (mp_irqs[i].irqflag != 0x0f)
Y
Yinghai Lu 已提交
938 939 940
			continue;

		if (nr_m_spare > 0) {
941
			apic_printk(APIC_VERBOSE, "*NEW* found\n");
Y
Yinghai Lu 已提交
942 943 944 945
			nr_m_spare--;
			assign_to_mpc_intsrc(&mp_irqs[i], m_spare[nr_m_spare]);
			m_spare[nr_m_spare] = NULL;
		} else {
946 947
			struct mpc_intsrc *m = (struct mpc_intsrc *)mpt;
			count += sizeof(struct mpc_intsrc);
948 949
			if (!check_slot(mpc_new_phys, mpc_new_length, count))
				goto out;
Y
Yinghai Lu 已提交
950
			assign_to_mpc_intsrc(&mp_irqs[i], m);
951
			mpc->length = count;
952
			mpt += sizeof(struct mpc_intsrc);
Y
Yinghai Lu 已提交
953 954 955 956 957 958
		}
		print_mp_irq_info(&mp_irqs[i]);
	}
#endif
out:
	/* update checksum */
959 960
	mpc->checksum = 0;
	mpc->checksum -= mpf_checksum((unsigned char *)mpc, mpc->length);
Y
Yinghai Lu 已提交
961 962 963 964

	return 0;
}

965 966
static int __initdata enable_update_mptable;

Y
Yinghai Lu 已提交
967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004
static int __init update_mptable_setup(char *str)
{
	enable_update_mptable = 1;
	return 0;
}
early_param("update_mptable", update_mptable_setup);

static unsigned long __initdata mpc_new_phys;
static unsigned long mpc_new_length __initdata = 4096;

/* alloc_mptable or alloc_mptable=4k */
static int __initdata alloc_mptable;
static int __init parse_alloc_mptable_opt(char *p)
{
	enable_update_mptable = 1;
	alloc_mptable = 1;
	if (!p)
		return 0;
	mpc_new_length = memparse(p, &p);
	return 0;
}
early_param("alloc_mptable", parse_alloc_mptable_opt);

void __init early_reserve_e820_mpc_new(void)
{
	if (enable_update_mptable && alloc_mptable) {
		u64 startt = 0;
#ifdef CONFIG_X86_TRAMPOLINE
		startt = TRAMPOLINE_BASE;
#endif
		mpc_new_phys = early_reserve_e820(startt, mpc_new_length, 4);
	}
}

static int __init update_mp_table(void)
{
	char str[16];
	char oem[10];
1005
	struct mpf_intel *mpf;
1006
	struct mpc_table *mpc, *mpc_new;
Y
Yinghai Lu 已提交
1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017

	if (!enable_update_mptable)
		return 0;

	mpf = mpf_found;
	if (!mpf)
		return 0;

	/*
	 * Now see if we need to go further.
	 */
1018
	if (mpf->feature1 != 0)
Y
Yinghai Lu 已提交
1019 1020
		return 0;

1021
	if (!mpf->physptr)
Y
Yinghai Lu 已提交
1022 1023
		return 0;

1024
	mpc = phys_to_virt(mpf->physptr);
Y
Yinghai Lu 已提交
1025 1026 1027 1028

	if (!smp_check_mpc(mpc, oem, str))
		return 0;

1029
	printk(KERN_INFO "mpf: %llx\n", (u64)virt_to_phys(mpf));
1030
	printk(KERN_INFO "physptr: %x\n", mpf->physptr);
Y
Yinghai Lu 已提交
1031

1032
	if (mpc_new_phys && mpc->length > mpc_new_length) {
Y
Yinghai Lu 已提交
1033 1034 1035 1036 1037 1038 1039 1040
		mpc_new_phys = 0;
		printk(KERN_INFO "mpc_new_length is %ld, please use alloc_mptable=8k\n",
			 mpc_new_length);
	}

	if (!mpc_new_phys) {
		unsigned char old, new;
		/* check if we can change the postion */
1041 1042 1043 1044
		mpc->checksum = 0;
		old = mpf_checksum((unsigned char *)mpc, mpc->length);
		mpc->checksum = 0xff;
		new = mpf_checksum((unsigned char *)mpc, mpc->length);
Y
Yinghai Lu 已提交
1045 1046 1047 1048 1049 1050
		if (old == new) {
			printk(KERN_INFO "mpc is readonly, please try alloc_mptable instead\n");
			return 0;
		}
		printk(KERN_INFO "use in-positon replacing\n");
	} else {
1051
		mpf->physptr = mpc_new_phys;
Y
Yinghai Lu 已提交
1052
		mpc_new = phys_to_virt(mpc_new_phys);
1053
		memcpy(mpc_new, mpc, mpc->length);
Y
Yinghai Lu 已提交
1054 1055
		mpc = mpc_new;
		/* check if we can modify that */
1056
		if (mpc_new_phys - mpf->physptr) {
1057
			struct mpf_intel *mpf_new;
Y
Yinghai Lu 已提交
1058 1059 1060 1061 1062
			/* steal 16 bytes from [0, 1k) */
			printk(KERN_INFO "mpf new: %x\n", 0x400 - 16);
			mpf_new = phys_to_virt(0x400 - 16);
			memcpy(mpf_new, mpf, 16);
			mpf = mpf_new;
1063
			mpf->physptr = mpc_new_phys;
Y
Yinghai Lu 已提交
1064
		}
1065 1066 1067
		mpf->checksum = 0;
		mpf->checksum -= mpf_checksum((unsigned char *)mpf, 16);
		printk(KERN_INFO "physptr new: %x\n", mpf->physptr);
Y
Yinghai Lu 已提交
1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081
	}

	/*
	 * only replace the one with mp_INT and
	 *	 MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW,
	 * already in mp_irqs , stored by ... and mp_config_acpi_gsi,
	 * may need pci=routeirq for all coverage
	 */
	replace_intsrc_all(mpc, mpc_new_phys, mpc_new_length);

	return 0;
}

late_initcall(update_mp_table);
新手
引导
客服 返回
顶部