|
@@ -25,6 +25,8 @@
|
|
|
#include <asm/proto.h>
|
|
|
#include <asm/acpi.h>
|
|
|
#include <asm/bios_ebda.h>
|
|
|
+#include <asm/e820.h>
|
|
|
+#include <asm/trampoline.h>
|
|
|
|
|
|
#include <mach_apic.h>
|
|
|
#ifdef CONFIG_X86_32
|
|
@@ -161,20 +163,81 @@ static void __init MP_ioapic_info(struct mpc_config_ioapic *m)
|
|
|
nr_ioapics++;
|
|
|
}
|
|
|
|
|
|
-static void __init MP_intsrc_info(struct mpc_config_intsrc *m)
|
|
|
+static void print_MP_intsrc_info(struct mpc_config_intsrc *m)
|
|
|
{
|
|
|
- printk(KERN_INFO "Int: type %d, pol %d, trig %d, bus %02x,"
|
|
|
+ printk(KERN_CONT "Int: type %d, pol %d, trig %d, bus %02x,"
|
|
|
" IRQ %02x, APIC ID %x, APIC INT %02x\n",
|
|
|
m->mpc_irqtype, m->mpc_irqflag & 3,
|
|
|
(m->mpc_irqflag >> 2) & 3, m->mpc_srcbus,
|
|
|
m->mpc_srcbusirq, m->mpc_dstapic, m->mpc_dstirq);
|
|
|
- mp_irqs[mp_irq_entries].mp_dstapic = m->mpc_dstapic;
|
|
|
- mp_irqs[mp_irq_entries].mp_type = m->mpc_type;
|
|
|
- mp_irqs[mp_irq_entries].mp_irqtype = m->mpc_irqtype;
|
|
|
- mp_irqs[mp_irq_entries].mp_irqflag = m->mpc_irqflag;
|
|
|
- mp_irqs[mp_irq_entries].mp_srcbus = m->mpc_srcbus;
|
|
|
- mp_irqs[mp_irq_entries].mp_srcbusirq = m->mpc_srcbusirq;
|
|
|
- mp_irqs[mp_irq_entries].mp_dstirq = m->mpc_dstirq;
|
|
|
+}
|
|
|
+
|
|
|
+static void __init print_mp_irq_info(struct mp_config_intsrc *mp_irq)
|
|
|
+{
|
|
|
+ printk(KERN_CONT "Int: type %d, pol %d, trig %d, bus %02x,"
|
|
|
+ " IRQ %02x, APIC ID %x, APIC INT %02x\n",
|
|
|
+ mp_irq->mp_irqtype, mp_irq->mp_irqflag & 3,
|
|
|
+ (mp_irq->mp_irqflag >> 2) & 3, mp_irq->mp_srcbus,
|
|
|
+ mp_irq->mp_srcbusirq, mp_irq->mp_dstapic, mp_irq->mp_dstirq);
|
|
|
+}
|
|
|
+
|
|
|
+static void assign_to_mp_irq(struct mpc_config_intsrc *m,
|
|
|
+ struct mp_config_intsrc *mp_irq)
|
|
|
+{
|
|
|
+ mp_irq->mp_dstapic = m->mpc_dstapic;
|
|
|
+ mp_irq->mp_type = m->mpc_type;
|
|
|
+ mp_irq->mp_irqtype = m->mpc_irqtype;
|
|
|
+ mp_irq->mp_irqflag = m->mpc_irqflag;
|
|
|
+ mp_irq->mp_srcbus = m->mpc_srcbus;
|
|
|
+ mp_irq->mp_srcbusirq = m->mpc_srcbusirq;
|
|
|
+ mp_irq->mp_dstirq = m->mpc_dstirq;
|
|
|
+}
|
|
|
+
|
|
|
+static void __init assign_to_mpc_intsrc(struct mp_config_intsrc *mp_irq,
|
|
|
+ struct mpc_config_intsrc *m)
|
|
|
+{
|
|
|
+ m->mpc_dstapic = mp_irq->mp_dstapic;
|
|
|
+ m->mpc_type = mp_irq->mp_type;
|
|
|
+ m->mpc_irqtype = mp_irq->mp_irqtype;
|
|
|
+ m->mpc_irqflag = mp_irq->mp_irqflag;
|
|
|
+ m->mpc_srcbus = mp_irq->mp_srcbus;
|
|
|
+ m->mpc_srcbusirq = mp_irq->mp_srcbusirq;
|
|
|
+ m->mpc_dstirq = mp_irq->mp_dstirq;
|
|
|
+}
|
|
|
+
|
|
|
+static int mp_irq_mpc_intsrc_cmp(struct mp_config_intsrc *mp_irq,
|
|
|
+ struct mpc_config_intsrc *m)
|
|
|
+{
|
|
|
+ if (mp_irq->mp_dstapic != m->mpc_dstapic)
|
|
|
+ return 1;
|
|
|
+ if (mp_irq->mp_type != m->mpc_type)
|
|
|
+ return 2;
|
|
|
+ if (mp_irq->mp_irqtype != m->mpc_irqtype)
|
|
|
+ return 3;
|
|
|
+ if (mp_irq->mp_irqflag != m->mpc_irqflag)
|
|
|
+ return 4;
|
|
|
+ if (mp_irq->mp_srcbus != m->mpc_srcbus)
|
|
|
+ return 5;
|
|
|
+ if (mp_irq->mp_srcbusirq != m->mpc_srcbusirq)
|
|
|
+ return 6;
|
|
|
+ if (mp_irq->mp_dstirq != m->mpc_dstirq)
|
|
|
+ return 7;
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+void MP_intsrc_info(struct mpc_config_intsrc *m)
|
|
|
+{
|
|
|
+ int i;
|
|
|
+
|
|
|
+ print_MP_intsrc_info(m);
|
|
|
+
|
|
|
+ for (i = 0; i < mp_irq_entries; i++) {
|
|
|
+ if (!mp_irq_mpc_intsrc_cmp(&mp_irqs[i], m))
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ assign_to_mp_irq(m, &mp_irqs[mp_irq_entries]);
|
|
|
if (++mp_irq_entries == MAX_IRQ_SOURCES)
|
|
|
panic("Max # of irq sources exceeded!!\n");
|
|
|
}
|
|
@@ -268,12 +331,9 @@ static inline void mps_oem_check(struct mp_config_table *mpc, char *oem,
|
|
|
* Read/parse the MPC
|
|
|
*/
|
|
|
|
|
|
-static int __init smp_read_mpc(struct mp_config_table *mpc, unsigned early)
|
|
|
+static int __init smp_check_mpc(struct mp_config_table *mpc, char *oem,
|
|
|
+ char *str)
|
|
|
{
|
|
|
- char str[16];
|
|
|
- char oem[10];
|
|
|
- int count = sizeof(*mpc);
|
|
|
- unsigned char *mpt = ((unsigned char *)mpc) + count;
|
|
|
|
|
|
if (memcmp(mpc->mpc_signature, MPC_SIGNATURE, 4)) {
|
|
|
printk(KERN_ERR "MPTABLE: bad signature [%c%c%c%c]!\n",
|
|
@@ -301,13 +361,28 @@ static int __init smp_read_mpc(struct mp_config_table *mpc, unsigned early)
|
|
|
memcpy(str, mpc->mpc_productid, 12);
|
|
|
str[12] = 0;
|
|
|
|
|
|
-#ifdef CONFIG_X86_32
|
|
|
- mps_oem_check(mpc, oem, str);
|
|
|
-#endif
|
|
|
printk(KERN_INFO "MPTABLE: Product ID: %s\n", str);
|
|
|
|
|
|
printk(KERN_INFO "MPTABLE: APIC at: 0x%X\n", mpc->mpc_lapic);
|
|
|
|
|
|
+ return 1;
|
|
|
+}
|
|
|
+
|
|
|
+static int __init smp_read_mpc(struct mp_config_table *mpc, unsigned early)
|
|
|
+{
|
|
|
+ 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
|
|
|
+ mps_oem_check(mpc, oem, str);
|
|
|
+#endif
|
|
|
+
|
|
|
/* save the local APIC address, it might be non-default */
|
|
|
if (!acpi_lapic)
|
|
|
mp_lapic_addr = mpc->mpc_lapic;
|
|
@@ -785,3 +860,295 @@ void __init find_smp_config(void)
|
|
|
{
|
|
|
__find_smp_config(1);
|
|
|
}
|
|
|
+
|
|
|
+#ifdef CONFIG_X86_IO_APIC
|
|
|
+static u8 __initdata irq_used[MAX_IRQ_SOURCES];
|
|
|
+
|
|
|
+static int __init get_MP_intsrc_index(struct mpc_config_intsrc *m)
|
|
|
+{
|
|
|
+ int i;
|
|
|
+
|
|
|
+ if (m->mpc_irqtype != mp_INT)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ if (m->mpc_irqflag != 0x0f)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ /* not legacy */
|
|
|
+
|
|
|
+ for (i = 0; i < mp_irq_entries; i++) {
|
|
|
+ if (mp_irqs[i].mp_irqtype != mp_INT)
|
|
|
+ continue;
|
|
|
+
|
|
|
+ if (mp_irqs[i].mp_irqflag != 0x0f)
|
|
|
+ continue;
|
|
|
+
|
|
|
+ if (mp_irqs[i].mp_srcbus != m->mpc_srcbus)
|
|
|
+ continue;
|
|
|
+ if (mp_irqs[i].mp_srcbusirq != m->mpc_srcbusirq)
|
|
|
+ continue;
|
|
|
+ if (irq_used[i]) {
|
|
|
+ /* already claimed */
|
|
|
+ return -2;
|
|
|
+ }
|
|
|
+ irq_used[i] = 1;
|
|
|
+ return i;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* not found */
|
|
|
+ return -1;
|
|
|
+}
|
|
|
+
|
|
|
+#define SPARE_SLOT_NUM 20
|
|
|
+
|
|
|
+static struct mpc_config_intsrc __initdata *m_spare[SPARE_SLOT_NUM];
|
|
|
+#endif
|
|
|
+
|
|
|
+static int __init replace_intsrc_all(struct mp_config_table *mpc,
|
|
|
+ unsigned long mpc_new_phys,
|
|
|
+ unsigned long mpc_new_length)
|
|
|
+{
|
|
|
+#ifdef CONFIG_X86_IO_APIC
|
|
|
+ int i;
|
|
|
+ int nr_m_spare = 0;
|
|
|
+#endif
|
|
|
+
|
|
|
+ int count = sizeof(*mpc);
|
|
|
+ unsigned char *mpt = ((unsigned char *)mpc) + count;
|
|
|
+
|
|
|
+ printk(KERN_INFO "mpc_length %x\n", mpc->mpc_length);
|
|
|
+ while (count < mpc->mpc_length) {
|
|
|
+ switch (*mpt) {
|
|
|
+ case MP_PROCESSOR:
|
|
|
+ {
|
|
|
+ struct mpc_config_processor *m =
|
|
|
+ (struct mpc_config_processor *)mpt;
|
|
|
+ mpt += sizeof(*m);
|
|
|
+ count += sizeof(*m);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ case MP_BUS:
|
|
|
+ {
|
|
|
+ struct mpc_config_bus *m =
|
|
|
+ (struct mpc_config_bus *)mpt;
|
|
|
+ mpt += sizeof(*m);
|
|
|
+ count += sizeof(*m);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ case MP_IOAPIC:
|
|
|
+ {
|
|
|
+ mpt += sizeof(struct mpc_config_ioapic);
|
|
|
+ count += sizeof(struct mpc_config_ioapic);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ case MP_INTSRC:
|
|
|
+ {
|
|
|
+#ifdef CONFIG_X86_IO_APIC
|
|
|
+ struct mpc_config_intsrc *m =
|
|
|
+ (struct mpc_config_intsrc *)mpt;
|
|
|
+
|
|
|
+ printk(KERN_INFO "OLD ");
|
|
|
+ print_MP_intsrc_info(m);
|
|
|
+ i = get_MP_intsrc_index(m);
|
|
|
+ if (i > 0) {
|
|
|
+ assign_to_mpc_intsrc(&mp_irqs[i], m);
|
|
|
+ printk(KERN_INFO "NEW ");
|
|
|
+ print_mp_irq_info(&mp_irqs[i]);
|
|
|
+ } else if (!i) {
|
|
|
+ /* legacy, do nothing */
|
|
|
+ } else 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++;
|
|
|
+ }
|
|
|
+#endif
|
|
|
+ mpt += sizeof(struct mpc_config_intsrc);
|
|
|
+ count += sizeof(struct mpc_config_intsrc);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ case MP_LINTSRC:
|
|
|
+ {
|
|
|
+ struct mpc_config_lintsrc *m =
|
|
|
+ (struct mpc_config_lintsrc *)mpt;
|
|
|
+ mpt += sizeof(*m);
|
|
|
+ count += sizeof(*m);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ default:
|
|
|
+ /* wrong mptable */
|
|
|
+ printk(KERN_ERR "Your mptable is wrong, contact your HW vendor!\n");
|
|
|
+ printk(KERN_ERR "type %x\n", *mpt);
|
|
|
+ print_hex_dump(KERN_ERR, " ", DUMP_PREFIX_ADDRESS, 16,
|
|
|
+ 1, mpc, mpc->mpc_length, 1);
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+#ifdef CONFIG_X86_IO_APIC
|
|
|
+ for (i = 0; i < mp_irq_entries; i++) {
|
|
|
+ if (irq_used[i])
|
|
|
+ continue;
|
|
|
+
|
|
|
+ if (mp_irqs[i].mp_irqtype != mp_INT)
|
|
|
+ continue;
|
|
|
+
|
|
|
+ if (mp_irqs[i].mp_irqflag != 0x0f)
|
|
|
+ continue;
|
|
|
+
|
|
|
+ if (nr_m_spare > 0) {
|
|
|
+ printk(KERN_INFO "*NEW* found ");
|
|
|
+ nr_m_spare--;
|
|
|
+ assign_to_mpc_intsrc(&mp_irqs[i], m_spare[nr_m_spare]);
|
|
|
+ m_spare[nr_m_spare] = NULL;
|
|
|
+ } else {
|
|
|
+ struct mpc_config_intsrc *m =
|
|
|
+ (struct mpc_config_intsrc *)mpt;
|
|
|
+ count += sizeof(struct mpc_config_intsrc);
|
|
|
+ if (!mpc_new_phys) {
|
|
|
+ printk(KERN_INFO "No spare slots, try to append...take your risk, new mpc_length %x\n", count);
|
|
|
+ } else {
|
|
|
+ if (count <= mpc_new_length)
|
|
|
+ printk(KERN_INFO "No spare slots, try to append..., new mpc_length %x\n", count);
|
|
|
+ else {
|
|
|
+ printk(KERN_ERR "mpc_new_length %lx is too small\n", mpc_new_length);
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ assign_to_mpc_intsrc(&mp_irqs[i], m);
|
|
|
+ mpc->mpc_length = count;
|
|
|
+ mpt += sizeof(struct mpc_config_intsrc);
|
|
|
+ }
|
|
|
+ print_mp_irq_info(&mp_irqs[i]);
|
|
|
+ }
|
|
|
+#endif
|
|
|
+out:
|
|
|
+ /* update checksum */
|
|
|
+ mpc->mpc_checksum = 0;
|
|
|
+ mpc->mpc_checksum -= mpf_checksum((unsigned char *)mpc,
|
|
|
+ mpc->mpc_length);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+int __initdata enable_update_mptable;
|
|
|
+
|
|
|
+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];
|
|
|
+ struct intel_mp_floating *mpf;
|
|
|
+ struct mp_config_table *mpc;
|
|
|
+ struct mp_config_table *mpc_new;
|
|
|
+
|
|
|
+ if (!enable_update_mptable)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ mpf = mpf_found;
|
|
|
+ if (!mpf)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Now see if we need to go further.
|
|
|
+ */
|
|
|
+ if (mpf->mpf_feature1 != 0)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ if (!mpf->mpf_physptr)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ mpc = phys_to_virt(mpf->mpf_physptr);
|
|
|
+
|
|
|
+ if (!smp_check_mpc(mpc, oem, str))
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ printk(KERN_INFO "mpf: %lx\n", virt_to_phys(mpf));
|
|
|
+ printk(KERN_INFO "mpf_physptr: %x\n", mpf->mpf_physptr);
|
|
|
+
|
|
|
+ if (mpc_new_phys && mpc->mpc_length > mpc_new_length) {
|
|
|
+ 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 */
|
|
|
+ mpc->mpc_checksum = 0;
|
|
|
+ old = mpf_checksum((unsigned char *)mpc, mpc->mpc_length);
|
|
|
+ mpc->mpc_checksum = 0xff;
|
|
|
+ new = mpf_checksum((unsigned char *)mpc, mpc->mpc_length);
|
|
|
+ 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 {
|
|
|
+ mpf->mpf_physptr = mpc_new_phys;
|
|
|
+ mpc_new = phys_to_virt(mpc_new_phys);
|
|
|
+ memcpy(mpc_new, mpc, mpc->mpc_length);
|
|
|
+ mpc = mpc_new;
|
|
|
+ /* check if we can modify that */
|
|
|
+ if (mpc_new_phys - mpf->mpf_physptr) {
|
|
|
+ struct intel_mp_floating *mpf_new;
|
|
|
+ /* 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;
|
|
|
+ mpf->mpf_physptr = mpc_new_phys;
|
|
|
+ }
|
|
|
+ mpf->mpf_checksum = 0;
|
|
|
+ mpf->mpf_checksum -= mpf_checksum((unsigned char *)mpf, 16);
|
|
|
+ printk(KERN_INFO "mpf_physptr new: %x\n", mpf->mpf_physptr);
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * 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);
|