2009-07-01 19:01:17 +02:00
|
|
|
/*
|
|
|
|
* This file is part of the coreboot project.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007-2008 coresystems GmbH
|
|
|
|
* Copyright (C) 2009 Thomas Jourdan <thomas.jourdan@gmail.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; version 2 of
|
|
|
|
* the License.
|
|
|
|
*
|
|
|
|
* 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., 51 Franklin St, Fifth Floor, Boston,
|
|
|
|
* MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <console/console.h>
|
|
|
|
#include <arch/io.h>
|
2010-10-12 19:34:08 +02:00
|
|
|
#include <arch/ioapic.h>
|
2009-07-01 19:01:17 +02:00
|
|
|
#include <arch/smp/mpspec.h>
|
|
|
|
#include <device/pci.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
// Generate MP-table IRQ numbers for PCI devices.
|
|
|
|
#define IO_APIC0 2
|
|
|
|
|
|
|
|
#define INT_A 0
|
|
|
|
#define INT_B 1
|
|
|
|
#define INT_C 2
|
|
|
|
#define INT_D 3
|
|
|
|
#define PCI_IRQ(dev, intLine) (((dev)<<2) | intLine)
|
|
|
|
|
|
|
|
#define PIRQ_A 16
|
|
|
|
#define PIRQ_B 17
|
|
|
|
#define PIRQ_C 18
|
|
|
|
#define PIRQ_D 19
|
|
|
|
#define PIRQ_E 20
|
|
|
|
#define PIRQ_F 21
|
|
|
|
#define PIRQ_G 22
|
|
|
|
#define PIRQ_H 23
|
|
|
|
|
|
|
|
// RCBA
|
|
|
|
#define RCBA 0xF0
|
|
|
|
|
|
|
|
#define RCBA_D31IP 0x3100
|
|
|
|
#define RCBA_D30IP 0x3104
|
|
|
|
#define RCBA_D29IP 0x3108
|
|
|
|
#define RCBA_D28IP 0x310C
|
|
|
|
#define RCBA_D31IR 0x3140
|
|
|
|
#define RCBA_D30IR 0x3142
|
|
|
|
#define RCBA_D29IR 0x3144
|
|
|
|
#define RCBA_D28IR 0x3146
|
|
|
|
|
2010-03-22 17:33:25 +01:00
|
|
|
static void *smp_write_config_table(void *v)
|
2009-07-01 19:01:17 +02:00
|
|
|
{
|
|
|
|
struct mp_config_table *mc;
|
2010-11-22 15:14:56 +01:00
|
|
|
unsigned char bus_chipset, bus_pci;
|
2009-07-01 19:01:17 +02:00
|
|
|
unsigned char bus_pcie_a, bus_pcie_a1, bus_pcie_b;
|
2010-11-22 15:14:56 +01:00
|
|
|
int bus_isa, i;
|
2009-07-01 19:01:17 +02:00
|
|
|
uint32_t pin, route;
|
|
|
|
device_t dev;
|
|
|
|
struct resource *res;
|
2014-12-25 03:43:20 +01:00
|
|
|
u8 *rcba;
|
2009-07-01 19:01:17 +02:00
|
|
|
|
|
|
|
dev = dev_find_slot(0, PCI_DEVFN(0x1F,0));
|
|
|
|
res = find_resource(dev, RCBA);
|
|
|
|
if (!res) {
|
2010-04-07 04:30:57 +02:00
|
|
|
return NULL;
|
2009-07-01 19:01:17 +02:00
|
|
|
}
|
2014-12-25 03:43:20 +01:00
|
|
|
rcba = res2mmio(res, 0, 0);
|
2009-07-01 19:01:17 +02:00
|
|
|
|
|
|
|
mc = (void *)(((char *)v) + SMP_FLOATING_TABLE_LEN);
|
Factor out common mptable code to mptable_init().
- Drop sig[], oem[], and productid[] fields in all mptable.c files, no
longer needed. The sig[] is always the same ("PCMP"), the oem[] is
currently also always the same ("COREBOOT"), and productid is being
passed into mptable_init() directly as string now.
- LAPIC_ADDR is passed in as parameter, too. While at the moment it's
always the same value that is passed in, the LAPIC base address could
also be relocated theoretically, so keep it as parameter for now.
- Fix a few productid entries, they were (partially) incorrect:
- DK8S2 (was "DK8X", copypaste)
- 939A785GMH (was "MAHOGANY", copypaste)
- X6DHE-G (was "X6DHE", incomplete board name)
- H8DME-2 (was "H8DMR", copypaste)
- H8QME-2+ (was "H8QME", incomplete board name)
- X6DHE-G2 (was "X6DHE", incomplete board name)
- X6DHR-iG2 (was "X6DHR-iG", incomplete board name)
- GA-M57SLI-S4 (was "M57SLI", incomplete board name)
- KINO-780AM2 (was "KINO", incomplete board name)
- DL145 G1 (was "DL145G1", small fix as per vendor website)
- DL145 G3 (was "TREX", wrong board name)
- DL165 G6 (was "HP DL165 G6", drop vendor)
- S2912 (was "S2895", copypaste)
- VT8454c (was "VIA VT8454C", drop vendor, lower-case "c")
- EPIA-N (was "P4DPE", copypaste)
- pc2500e (was "PC2500", incorrect name)
- S1850 (was "S2850", copy-paste)
- MS-7135 (was "MS7135")
- MS-9282 (was "MS9282")
- MS-9185 (was "MS9185")
- MS-9652 (was "K9ND MS-9652")
- Ultra 40 (was "ultra40")
- E326 (was "E325", copypaste)
- M4A785-M (was "TILAPIA", copypaste)
- P2B-D (was "ASUS P2B-D", drop vendor)
- P2B-DS (was "ASUS P2B-DS", drop vendor)
- Adapt the mptable utility to use mptable_init() too.
Abuild-tested.
Signed-off-by: Uwe Hermann <uwe@hermann-uwe.de>
Acked-by: Patrick Georgi <patrick.georgi@coresystems.de>
git-svn-id: svn://svn.coreboot.org/coreboot/trunk@5987 2b7e53f0-3cfb-0310-b3e9-8179ed1497e1
2010-10-25 17:32:07 +02:00
|
|
|
|
2012-02-16 18:43:25 +01:00
|
|
|
mptable_init(mc, LOCAL_APIC_ADDR);
|
2009-07-01 19:01:17 +02:00
|
|
|
|
|
|
|
smp_write_processors(mc);
|
|
|
|
|
|
|
|
/* Get bus numbers */
|
|
|
|
bus_chipset = 0;
|
|
|
|
|
|
|
|
/* PCI */
|
|
|
|
dev = dev_find_slot(0, PCI_DEVFN(0x1E,0));
|
|
|
|
if (dev) {
|
|
|
|
bus_pci = pci_read_config8(dev, PCI_SECONDARY_BUS);
|
|
|
|
} else {
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "ERROR - could not find PCI 0:1e.0, using defaults\n");
|
2009-07-01 19:01:17 +02:00
|
|
|
bus_pci = 6;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev = dev_find_slot(0, PCI_DEVFN(2,0));
|
|
|
|
if(dev) {
|
|
|
|
bus_pcie_a = pci_read_config8(dev, PCI_SECONDARY_BUS);
|
|
|
|
} else {
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "ERROR - could not find PCIe Port A 0:2.0, using defaults\n");
|
2009-07-01 19:01:17 +02:00
|
|
|
bus_pcie_a = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev = dev_find_slot(0, PCI_DEVFN(3,0));
|
|
|
|
if(dev) {
|
|
|
|
bus_pcie_a1 = pci_read_config8(dev, PCI_SECONDARY_BUS);
|
|
|
|
} else {
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "ERROR - could not find PCIe Port B 0:3.0, using defaults\n");
|
2009-07-01 19:01:17 +02:00
|
|
|
bus_pcie_a1 = 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev = dev_find_slot(0, PCI_DEVFN(0x1C,0));
|
|
|
|
if(dev) {
|
|
|
|
bus_pcie_b = pci_read_config8(dev, PCI_SECONDARY_BUS);
|
|
|
|
} else {
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "ERROR - could not find PCIe Port B 0:3.0, using defaults\n");
|
2009-07-01 19:01:17 +02:00
|
|
|
bus_pcie_b = 3;
|
|
|
|
}
|
|
|
|
|
2010-11-22 15:14:56 +01:00
|
|
|
mptable_write_buses(mc, NULL, &bus_isa);
|
2009-07-01 19:01:17 +02:00
|
|
|
|
|
|
|
/*I/O APICs: APIC ID Version State Address*/
|
2014-12-25 03:43:20 +01:00
|
|
|
smp_write_ioapic(mc, 2, 0x20, VIO_APIC_VADDR);
|
2010-10-31 20:37:50 +01:00
|
|
|
|
2010-05-20 17:28:19 +02:00
|
|
|
mptable_add_isa_interrupts(mc, bus_isa, IO_APIC0, 0);
|
2009-07-01 19:01:17 +02:00
|
|
|
|
|
|
|
/*Local Ints: Type Polarity Trigger Bus ID IRQ APIC ID PIN#*/
|
mptable: Refactor lintsrc generation
We copied pretty much the same code for generating mptable entries for
local interrupts (with some notable exceptions).
This change moves these lines into a generic function "mptable_lintsrc"
and makes use of it in many places.
The remaining uses of smp_write_lintsrc should be reviewed and replaced
by mptable_lintsrc calls where possible, and smp_write_lintsrc made static.
This patch was generated using Coccinelle:
@@
expression mc;
expression isa_bus;
@@
-smp_write_lintsrc(mc, mp_ExtINT, MP_IRQ_TRIGGER_DEFAULT|MP_IRQ_POLARITY_DEFAULT, isa_bus, 0x0, MP_APIC_ALL, 0x0);
-smp_write_lintsrc(mc, mp_NMI, MP_IRQ_TRIGGER_DEFAULT|MP_IRQ_POLARITY_DEFAULT, isa_bus, 0x0, MP_APIC_ALL, 0x1);
+mptable_lintsrc(mc, isa_bus);
@@
expression mc;
expression isa_bus;
@@
-smp_write_lintsrc(mc, mp_ExtINT, MP_IRQ_TRIGGER_EDGE|MP_IRQ_POLARITY_HIGH, isa_bus, 0x0, MP_APIC_ALL, 0x0);
-smp_write_lintsrc(mc, mp_NMI, MP_IRQ_TRIGGER_EDGE|MP_IRQ_POLARITY_HIGH, isa_bus, 0x0, MP_APIC_ALL, 0x1);
+mptable_lintsrc(mc, isa_bus);
@m@
identifier mc;
expression BUS;
@@
-#define IO_LOCAL_INT(type, intr, apicid, pin) smp_write_lintsrc(mc, (type), MP_IRQ_TRIGGER_EDGE | MP_IRQ_POLARITY_HIGH, BUS, (intr), (apicid), (pin));
...
-IO_LOCAL_INT(mp_ExtINT, 0x0, MP_APIC_ALL, 0x0);
-IO_LOCAL_INT(mp_NMI, 0x0, MP_APIC_ALL, 0x1);
+mptable_lintsrc(mc, BUS);
Change-Id: I97421f820cd039f5fd753cb0da5c1cca68819bb4
Signed-off-by: Patrick Georgi <patrick@georgi-clan.de>
Reviewed-on: http://review.coreboot.org/244
Tested-by: build bot (Jenkins)
Reviewed-by: Marc Jones <marcj303@gmail.com>
2011-10-07 21:42:52 +02:00
|
|
|
mptable_lintsrc(mc, bus_isa);
|
2009-07-01 19:01:17 +02:00
|
|
|
|
|
|
|
/* Internal PCI device for i3100 */
|
|
|
|
|
|
|
|
/* EDMA
|
|
|
|
*/
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_chipset, PCI_IRQ(1, INT_A), IO_APIC0, PIRQ_A);
|
|
|
|
|
|
|
|
/* PCIe Port A
|
|
|
|
*/
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_chipset, PCI_IRQ(2, INT_A), IO_APIC0, PIRQ_A);
|
|
|
|
|
|
|
|
/* PCIe Port A1
|
|
|
|
*/
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_chipset, PCI_IRQ(3, INT_A), IO_APIC0, PIRQ_A);
|
|
|
|
|
|
|
|
/* PCIe Port B
|
|
|
|
*/
|
|
|
|
for(i = 0; i < 4; i++) {
|
2010-01-16 18:53:38 +01:00
|
|
|
pin = (read32(rcba + RCBA_D28IP) >> (i * 4)) & 0x0F;
|
2009-07-01 19:01:17 +02:00
|
|
|
if(pin > 0) {
|
|
|
|
pin -= 1;
|
2010-01-16 18:53:38 +01:00
|
|
|
route = PIRQ_A + ((read16(rcba + RCBA_D28IR) >> (pin * 4)) & 0x07);
|
2009-07-01 19:01:17 +02:00
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_chipset, PCI_IRQ(28, pin), IO_APIC0, route);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* USB 1.1 : device 29, function 0, 1
|
|
|
|
*/
|
|
|
|
for(i = 0; i < 2; i++) {
|
2010-01-16 18:53:38 +01:00
|
|
|
pin = (read32(rcba + RCBA_D29IP) >> (i * 4)) & 0x0F;
|
2009-07-01 19:01:17 +02:00
|
|
|
if(pin > 0) {
|
|
|
|
pin -= 1;
|
2010-01-16 18:53:38 +01:00
|
|
|
route = PIRQ_A + ((read16(rcba + RCBA_D29IR) >> (pin * 4)) & 0x07);
|
2009-07-01 19:01:17 +02:00
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_chipset, PCI_IRQ(29, pin), IO_APIC0, route);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* USB 2.0 : device 29, function 7
|
|
|
|
*/
|
2010-01-16 18:53:38 +01:00
|
|
|
pin = (read32(rcba + RCBA_D29IP) >> (7 * 4)) & 0x0F;
|
2009-07-01 19:01:17 +02:00
|
|
|
if(pin > 0) {
|
|
|
|
pin -= 1;
|
2010-01-16 18:53:38 +01:00
|
|
|
route = PIRQ_A + ((read16(rcba + RCBA_D29IR) >> (pin * 4)) & 0x07);
|
2009-07-01 19:01:17 +02:00
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_chipset, PCI_IRQ(29, pin), IO_APIC0, route);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* SATA : device 31 function 2
|
|
|
|
SMBus : device 31 function 3
|
|
|
|
Performance counters : device 31 function 4
|
|
|
|
*/
|
|
|
|
for(i = 2; i < 5; i++) {
|
2010-01-16 18:53:38 +01:00
|
|
|
pin = (read32(rcba + RCBA_D31IP) >> (i * 4)) & 0x0F;
|
2009-07-01 19:01:17 +02:00
|
|
|
if(pin > 0) {
|
|
|
|
pin -= 1;
|
2010-01-16 18:53:38 +01:00
|
|
|
route = PIRQ_A + ((read16(rcba + RCBA_D31IR) >> (pin * 4)) & 0x07);
|
2009-07-01 19:01:17 +02:00
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_chipset, PCI_IRQ(31, pin), IO_APIC0, route);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* SLOTS */
|
|
|
|
|
|
|
|
/* PCIe 4x slot A
|
|
|
|
*/
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_a, PCI_IRQ(0, INT_A), IO_APIC0, PIRQ_A);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_a, PCI_IRQ(0, INT_B), IO_APIC0, PIRQ_B);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_a, PCI_IRQ(0, INT_C), IO_APIC0, PIRQ_C);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_a, PCI_IRQ(0, INT_D), IO_APIC0, PIRQ_D);
|
|
|
|
|
|
|
|
/* PCIe 4x slot A1
|
|
|
|
*/
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_a1, PCI_IRQ(0, INT_A), IO_APIC0, PIRQ_A);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_a1, PCI_IRQ(0, INT_B), IO_APIC0, PIRQ_B);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_a1, PCI_IRQ(0, INT_C), IO_APIC0, PIRQ_C);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_a1, PCI_IRQ(0, INT_D), IO_APIC0, PIRQ_D);
|
|
|
|
|
|
|
|
/* PCIe 4x slot B
|
|
|
|
*/
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_b, PCI_IRQ(0, INT_A), IO_APIC0, PIRQ_A);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_b, PCI_IRQ(0, INT_B), IO_APIC0, PIRQ_B);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_b, PCI_IRQ(0, INT_C), IO_APIC0, PIRQ_C);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pcie_b, PCI_IRQ(0, INT_D), IO_APIC0, PIRQ_D);
|
|
|
|
|
|
|
|
/* PCI slot
|
|
|
|
*/
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pci, PCI_IRQ(0, INT_A), IO_APIC0, PIRQ_A);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pci, PCI_IRQ(0, INT_B), IO_APIC0, PIRQ_B);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pci, PCI_IRQ(0, INT_C), IO_APIC0, PIRQ_C);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, bus_pci, PCI_IRQ(0, INT_D), IO_APIC0, PIRQ_D);
|
|
|
|
|
|
|
|
/* There is no extension information... */
|
|
|
|
|
|
|
|
/* Compute the checksums */
|
2011-10-07 23:01:55 +02:00
|
|
|
return mptable_finalize(mc);
|
2009-07-01 19:01:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned long write_smp_table(unsigned long addr)
|
|
|
|
{
|
|
|
|
void *v;
|
2011-10-07 22:41:07 +02:00
|
|
|
v = smp_write_floating_table(addr, 0);
|
2009-07-01 19:01:17 +02:00
|
|
|
return (unsigned long)smp_write_config_table(v);
|
|
|
|
}
|