2009-04-22 22:34:05 +02:00
|
|
|
/*
|
|
|
|
* This file is part of the coreboot project.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2001 Eric W.Biederman<ebiderman@lnxi.com>
|
|
|
|
*
|
|
|
|
* Copyright (C) 2006 AMD
|
|
|
|
* Written by Yinghai Lu <yinghailu@gmail.com> for AMD.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007 University of Mannheim
|
|
|
|
* Written by Philipp Degler <pdegler@rumms.uni-mannheim.e> for Uni of Mannheim
|
|
|
|
*
|
|
|
|
* Copyright (C) 2009 University of Heidelberg
|
|
|
|
* Written by Mondrian Nuessle <nuessle@uni-hd.de> for Uni of Heidelberg
|
|
|
|
*
|
|
|
|
* 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <console/console.h>
|
|
|
|
#include <arch/smp/mpspec.h>
|
|
|
|
#include <arch/io.h>
|
|
|
|
#include <device/pci.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#if CONFIG_LOGICAL_CPUS==1
|
2010-03-29 16:45:36 +02:00
|
|
|
#include <cpu/amd/multicore.h>
|
2009-04-22 22:34:05 +02:00
|
|
|
#endif
|
|
|
|
#include <cpu/amd/amdk8_sysconf.h>
|
|
|
|
#include "mb_sysconf.h"
|
|
|
|
|
2010-03-22 17:33:25 +01:00
|
|
|
static void *smp_write_config_table(void *v)
|
2009-04-22 22:34:05 +02:00
|
|
|
{
|
|
|
|
struct mp_config_table *mc;
|
|
|
|
struct mb_sysconf_t *m;
|
|
|
|
|
|
|
|
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
|
|
|
mptable_init(mc, "TREX ", LAPIC_ADDR);
|
2009-04-22 22:34:05 +02:00
|
|
|
|
|
|
|
smp_write_processors(mc);
|
|
|
|
|
|
|
|
get_bus_conf();
|
|
|
|
m = sysconf.mb;
|
|
|
|
|
|
|
|
/*Bus: Bus ID Type*/
|
|
|
|
/* define bus and isa numbers */
|
2010-04-14 12:12:23 +02:00
|
|
|
#if 0
|
|
|
|
unsigned char bus_num;
|
|
|
|
for(bus_num = 0; bus_num < m->bus_isa; bus_num++) {
|
2009-04-22 22:34:05 +02:00
|
|
|
smp_write_bus(mc, bus_num, "PCI ");
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "writing bus %d as PCI...\n",bus_num);
|
2009-04-22 22:34:05 +02:00
|
|
|
}
|
2010-04-14 12:12:23 +02:00
|
|
|
#endif
|
2009-04-22 22:34:05 +02:00
|
|
|
smp_write_bus(mc, 0, "PCI ");
|
|
|
|
smp_write_bus(mc, 1, "PCI ");
|
|
|
|
smp_write_bus(mc, 7, "PCI ");
|
|
|
|
smp_write_bus(mc, 8, "PCI ");
|
|
|
|
|
|
|
|
smp_write_bus(mc,m->bus_isa, "ISA ");
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "writing %d as ISA...\n",m->bus_isa);
|
2009-04-22 22:34:05 +02:00
|
|
|
|
|
|
|
/*I/O APICs: APIC ID Version State Address*/
|
|
|
|
{
|
|
|
|
device_t dev = 0;
|
|
|
|
int i;
|
|
|
|
struct resource *res;
|
|
|
|
for(i=0; i<3; i++) {
|
|
|
|
dev = dev_find_device(0x1166, 0x0235, dev);
|
|
|
|
if (dev) {
|
|
|
|
res = find_resource(dev, PCI_BASE_ADDRESS_0);
|
|
|
|
if (res) {
|
2010-03-22 17:33:25 +01:00
|
|
|
printk(BIOS_DEBUG, "APIC %d base address: %llx\n",m->apicid_bcm5785[i], res->base);
|
2009-04-22 22:34:05 +02:00
|
|
|
smp_write_ioapic(mc, m->apicid_bcm5785[i], 0x11, res->base);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* IRQ routing as factory BIOS */
|
|
|
|
outb(0x01, 0xc00); outb(0x0A, 0xc01);
|
|
|
|
outb(0x17, 0xc00); outb(0x05, 0xc01);
|
|
|
|
/* outb(0x2E, 0xc00); outb(0x0B, 0xc01); */
|
|
|
|
/* outb(0x07, 0xc00); outb(0x07, 0xc01); */
|
|
|
|
outb(0x07, 0xc00); outb(0x0b, 0xc01);
|
|
|
|
|
|
|
|
outb(0x24, 0xc00); outb(0x05, 0xc01);
|
|
|
|
//outb(0x00, 0xc00); outb(0x09, 0xc01);
|
|
|
|
outb(0x02, 0xc00); outb(0x0E, 0xc01);
|
|
|
|
|
|
|
|
// 8259 registers...
|
|
|
|
outb(0xa0, 0x4d0);
|
|
|
|
outb(0x0e, 0x4d1);
|
|
|
|
|
|
|
|
{
|
|
|
|
device_t dev;
|
|
|
|
dev = dev_find_device(0x1166, 0x0205, 0);
|
|
|
|
if(dev) {
|
|
|
|
uint32_t dword;
|
|
|
|
dword = pci_read_config32(dev, 0x64);
|
|
|
|
dword |= (1<<30); // GEVENT14-21 used as PCI IRQ0-7
|
|
|
|
pci_write_config32(dev, 0x64, dword);
|
|
|
|
}
|
|
|
|
// set GEVENT pins to NO OP
|
|
|
|
outb(0x33, 0xcd6); outb(0x00, 0xcd7);
|
|
|
|
outb(0x34, 0xcd6); outb(0x00, 0xcd7);
|
|
|
|
outb(0x35, 0xcd6); outb(0x00, 0xcd7);
|
|
|
|
}
|
|
|
|
|
|
|
|
// hide XIOAPIC PCI configuration space
|
|
|
|
{
|
|
|
|
device_t dev;
|
|
|
|
dev = dev_find_device(0x1166, 0x205, 0);
|
|
|
|
if (dev) {
|
|
|
|
uint32_t dword;
|
|
|
|
dword = pci_read_config32(dev, 0x64);
|
|
|
|
dword |= (1<<26);
|
|
|
|
pci_write_config32(dev, 0x64, dword);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-05-20 17:28:19 +02:00
|
|
|
mptable_add_isa_interrupts(mc, m->bus_isa, m->apicid_bcm5785[0], 0);
|
|
|
|
|
2009-04-22 22:34:05 +02:00
|
|
|
//SATA
|
2010-03-22 12:42:32 +01:00
|
|
|
/* printk(BIOS_DEBUG, "MPTABLE_SATA: bus_id:%d irq:%d apic_id:%d pin:%d\n",m->bus_bcm5785_1, (0x0e<<2)|0, m->apicid_bcm5785[0], 0x7); */
|
2009-04-22 22:34:05 +02:00
|
|
|
/* smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, m->bus_bcm5785_1, (0x0e<<2)|0, m->apicid_bcm5785[0], 0x7); */
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "MPTABLE_SATA: bus_id:%d irq:%d apic_id:%d pin:%d\n",m->bus_bcm5785_1, (0x0e<<2)|0, m->apicid_bcm5785[0], 0xb);
|
2009-04-22 22:34:05 +02:00
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, m->bus_bcm5785_1, (0x0e<<2)|0, m->apicid_bcm5785[0], 0xb);
|
|
|
|
//USB
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "sysconf.sbdn: %d on bus: %x \n",sysconf.sbdn, m->bus_bcm5785_0);
|
2009-04-22 22:34:05 +02:00
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, m->bus_bcm5785_0, (0x03<<2)|0, m->apicid_bcm5785[0], 0xa);
|
|
|
|
|
|
|
|
//VGA
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, m->bus_bcm5785_0, (0x4<<2)|0, m->apicid_bcm5785[1], 0x7);
|
|
|
|
|
|
|
|
//PCIE
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, m->bus_bcm5785_0, (0x6<<2)|0, m->apicid_bcm5785[2], 0xe);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, m->bus_bcm5785_0, (0x7<<2)|0, m->apicid_bcm5785[2], 0xe);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, m->bus_bcm5785_0, (0x8<<2)|0, m->apicid_bcm5785[2], 0xe);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, m->bus_bcm5785_0, (0x9<<2)|0, m->apicid_bcm5785[2], 0xe);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW, m->bus_bcm5785_0, (0xa<<2)|0, m->apicid_bcm5785[2], 0xe);
|
|
|
|
|
|
|
|
//IDE
|
|
|
|
// outb(0x02, 0xc00); outb(0x0e, 0xc01);
|
2010-03-22 12:42:32 +01:00
|
|
|
// printk(BIOS_DEBUG, "MPTABLE_IDE: bus_id:%d irq:%d apic_id:%d pin:%d\n",m->bus_bcm5785_0, ((1+sysconf.sbdn)<<2)|1, m->apicid_bcm5785[0], 0xe);
|
2009-04-22 22:34:05 +02:00
|
|
|
// smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_EDGE|MP_IRQ_POLARITY_HIGH, m->bus_bcm5785_0, (0x02<<2)|1, m->apicid_bcm5785[0], 0xe);
|
|
|
|
|
|
|
|
//onboard Broadcom GbE
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW,8, (4<<2)|0, m->apicid_bcm5785[2], 0x4);
|
|
|
|
smp_write_intsrc(mc, mp_INT, MP_IRQ_TRIGGER_LEVEL|MP_IRQ_POLARITY_LOW,8, (4<<2)|1, m->apicid_bcm5785[2], 0x4);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* enable int */
|
|
|
|
/* why here? must get the BAR and PCI command bit 1 set before enable it ....*/
|
|
|
|
{
|
|
|
|
device_t dev;
|
|
|
|
dev = dev_find_device(0x1166, 0x0205, 0);
|
|
|
|
if(dev) {
|
|
|
|
uint32_t dword;
|
|
|
|
dword = pci_read_config32(dev, 0x6c);
|
|
|
|
dword |= (1<<4); // enable interrupts
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "6ch: %x\n",dword);
|
2009-04-22 22:34:05 +02:00
|
|
|
pci_write_config32(dev, 0x6c, dword);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*Local Ints: Type Polarity Trigger Bus ID IRQ APIC ID PIN#*/
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "m->bus_isa is: %x\n",m->bus_isa);
|
2010-10-27 00:40:16 +02:00
|
|
|
smp_write_lintsrc(mc, mp_ExtINT, MP_IRQ_TRIGGER_EDGE|MP_IRQ_POLARITY_HIGH, m->bus_isa, 0x0, MP_APIC_ALL, 0x0);
|
|
|
|
smp_write_lintsrc(mc, mp_NMI, MP_IRQ_TRIGGER_EDGE|MP_IRQ_POLARITY_HIGH, m->bus_isa , 0x0, MP_APIC_ALL, 0x1);
|
2009-04-22 22:34:05 +02:00
|
|
|
|
|
|
|
//extended table entries
|
|
|
|
smp_write_address_space(mc,0 , ADDRESS_TYPE_IO, 0x0, 0x0, 0x0, 0x0001);
|
|
|
|
smp_write_address_space(mc,0 , ADDRESS_TYPE_MEM, 0x0, 0x7f80, 0x0, 0x5e80);
|
|
|
|
smp_write_address_space(mc,0 , ADDRESS_TYPE_PREFETCH, 0x0, 0xde00, 0x0, 0x0100);
|
|
|
|
smp_write_address_space(mc,0 , ADDRESS_TYPE_MEM, 0x0, 0xdf00, 0x0, 0x1fe0);
|
|
|
|
smp_write_address_space(mc,0 , ADDRESS_TYPE_MEM, 0x1000, 0xfee0, 0xf000, 0x011f);
|
|
|
|
smp_write_address_space(mc,0 , ADDRESS_TYPE_MEM, 0x0, 0x000a, 0x0, 0x0006);
|
|
|
|
smp_write_bus_hierarchy(mc, 9, 0x01, 0);
|
|
|
|
smp_write_compatibility_address_space(mc, 0, ADDRESS_RANGE_ADD, 0);
|
|
|
|
smp_write_compatibility_address_space(mc, 0, ADDRESS_RANGE_ADD, 1);
|
|
|
|
|
|
|
|
|
|
|
|
/* Compute the checksums */
|
|
|
|
mc->mpe_checksum = smp_compute_checksum(smp_next_mpc_entry(mc), mc->mpe_length);
|
|
|
|
mc->mpc_checksum = smp_compute_checksum(mc, mc->mpc_length);
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "Wrote the mp table end at: %p - %p\n",
|
2009-04-22 22:34:05 +02:00
|
|
|
mc, smp_next_mpe_entry(mc));
|
|
|
|
return smp_next_mpe_entry(mc);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned long write_smp_table(unsigned long addr)
|
|
|
|
{
|
|
|
|
void *v;
|
|
|
|
v = smp_write_floating_table(addr);
|
|
|
|
return (unsigned long)smp_write_config_table(v);
|
|
|
|
}
|