We define IO_APIC_ADDR in <arch/ioapic.h>, let's use it.
[coreboot.git] / src / southbridge / nvidia / ck804 / ck804_lpc.c
index b06de3e558e1cf91f0c22250b53eb6a1351047f9..7d8c9ad723d85362fda23c7059149def7555ca8e 100644 (file)
@@ -4,6 +4,7 @@
  *  by yhlu@tyan.com
  *  2006.1 yhlu add dest apicid for IRQ0
  */
+
 #include <console/console.h>
 #include <device/device.h>
 #include <device/pci.h>
@@ -14,6 +15,7 @@
 #include <pc80/isa-dma.h>
 #include <bitops.h>
 #include <arch/io.h>
+#include <arch/ioapic.h>
 #include <cpu/x86/lapic.h>
 #include <stdlib.h>
 #include "ck804.h"
 
 #define NMI_OFF 0
 
-struct ioapicreg {
-       unsigned int reg;
-       unsigned int value_low, value_high;
-};
-
-static struct ioapicreg ioapicregvalues[] = {
-#define ALL            (0xff << 24)
-#define NONE           (0)
-#define DISABLED       (1 << 16)
-#define ENABLED                (0 << 16)
-#define TRIGGER_EDGE   (0 << 15)
-#define TRIGGER_LEVEL  (1 << 15)
-#define POLARITY_HIGH  (0 << 13)
-#define POLARITY_LOW   (1 << 13)
-#define PHYSICAL_DEST  (0 << 11)
-#define LOGICAL_DEST   (1 << 11)
-#define ExtINT         (7 << 8)
-#define NMI            (4 << 8)
-#define SMI            (2 << 8)
-#define INT            (1 << 8)
-       /* IO-APIC virtual wire mode configuration */
-       /* mask, trigger, polarity, destination, delivery, vector */
-       {   0, ENABLED | TRIGGER_EDGE | POLARITY_HIGH | PHYSICAL_DEST | ExtINT, NONE},
-       {   1, DISABLED, NONE},
-       {   2, DISABLED, NONE},
-       {   3, DISABLED, NONE},
-       {   4, DISABLED, NONE},
-       {   5, DISABLED, NONE},
-       {   6, DISABLED, NONE},
-       {   7, DISABLED, NONE},
-       {   8, DISABLED, NONE},
-       {   9, DISABLED, NONE},
-       {  10, DISABLED, NONE},
-       {  11, DISABLED, NONE},
-       {  12, DISABLED, NONE},
-       {  13, DISABLED, NONE},
-       {  14, DISABLED, NONE},
-       {  15, DISABLED, NONE},
-       {  16, DISABLED, NONE},
-       {  17, DISABLED, NONE},
-       {  18, DISABLED, NONE},
-       {  19, DISABLED, NONE},
-       {  20, DISABLED, NONE},
-       {  21, DISABLED, NONE},
-       {  22, DISABLED, NONE},
-       {  23, DISABLED, NONE},
-       /* Be careful and don't write past the end... */
-};
-
-static void setup_ioapic(unsigned long ioapic_base)
-{
-       int i;
-       unsigned long value_low, value_high;
-//     unsigned long ioapic_base = 0xfec00000;
-       volatile unsigned long *l;
-       struct ioapicreg *a = ioapicregvalues;
-
-       ioapicregvalues[0].value_high = lapicid()<<(56-32);
-
-       l = (unsigned long *) ioapic_base;
-
-       for (i = 0; i < ARRAY_SIZE(ioapicregvalues);
-            i++, a++) {
-               l[0] = (a->reg * 2) + 0x10;
-               l[4] = a->value_low;
-               value_low = l[4];
-               l[0] = (a->reg *2) + 0x11;
-               l[4] = a->value_high;
-               value_high = l[4];
-               if ((i==0) && (value_low == 0xffffffff)) {
-                       printk_warning("IO APIC not responding.\n");
-                       return;
-               }
-               printk_spew("for IRQ, reg 0x%08x value 0x%08x 0x%08x\n",
-                           a->reg, a->value_low, a->value_high);
-       }
-}
-
 // 0x7a or e3
 #define PREVIOUS_POWER_STATE 0x7A
 
@@ -108,8 +32,8 @@ static void setup_ioapic(unsigned long ioapic_base)
 #define SLOW_CPU_OFF 0
 #define SLOW_CPU__ON 1
 
-#ifndef MAINBOARD_POWER_ON_AFTER_POWER_FAIL
-#define MAINBOARD_POWER_ON_AFTER_POWER_FAIL MAINBOARD_POWER_ON
+#ifndef CONFIG_MAINBOARD_POWER_ON_AFTER_POWER_FAIL
+#define CONFIG_MAINBOARD_POWER_ON_AFTER_POWER_FAIL MAINBOARD_POWER_ON
 #endif
 
 static void lpc_common_init(device_t dev)
@@ -117,20 +41,19 @@ static void lpc_common_init(device_t dev)
        uint8_t byte;
        uint32_t dword;
 
-       /* IO APIC initialization */
+       /* I/O APIC initialization */
        byte = pci_read_config8(dev, 0x74);
-       byte |= (1<<0); // enable APIC
+       byte |= (1 << 0);       /* Enable APIC. */
        pci_write_config8(dev, 0x74, byte);
-       dword = pci_read_config32(dev, PCI_BASE_ADDRESS_1); // 0x14
+       dword = pci_read_config32(dev, PCI_BASE_ADDRESS_1);     /* 0x14 */
 
-       setup_ioapic(dword);
+       setup_ioapic(dword, 0); // Don't rename IOAPIC ID
 
 #if 1
        dword = pci_read_config32(dev, 0xe4);
-       dword |= (1<<23);
+       dword |= (1 << 23);
        pci_write_config32(dev, 0xe4, dword);
 #endif
-
 }
 
 static void lpc_slave_init(device_t dev)
@@ -138,132 +61,126 @@ static void lpc_slave_init(device_t dev)
        lpc_common_init(dev);
 }
 
-static void rom_dummy_write(device_t dev){
+static void rom_dummy_write(device_t dev)
+{
        uint8_t old, new;
        uint8_t *p;
 
        old = pci_read_config8(dev, 0x88);
        new = old | 0xc0;
-       if (new != old) {
+       if (new != old)
                pci_write_config8(dev, 0x88, new);
-       }
-       //  enable write
+       /* Enable write. */
        old = pci_read_config8(dev, 0x6d);
        new = old | 0x01;
-       if (new != old) {
+       if (new != old)
                pci_write_config8(dev, 0x6d, new);
-       }
 
-       /* dummy write */
-       p = (uint8_t *)0xffffffe0;
+       /* Dummy write. */
+       p = (uint8_t *) 0xffffffe0;
        old = 0;
        *p = old;
        old = *p;
 
-       //  disable write
+       /* Disable write. */
        old = pci_read_config8(dev, 0x6d);
        new = old & 0xfe;
-       if (new != old) {
+       if (new != old)
                pci_write_config8(dev, 0x6d, new);
-
-       }
-
 }
-#if 0
+
 static void enable_hpet(struct device *dev)
 {
        unsigned long hpet_address;
 
-       pci_write_config32(dev,0x44, 0xfed00001);
-       hpet_address=pci_read_config32(dev,0x44)& 0xfffffffe;
-       printk_debug("enabling HPET @0x%x\n", hpet_address);
+       pci_write_config32(dev, 0x44, 0xfed00001);
+       hpet_address = pci_read_config32(dev, 0x44) & 0xfffffffe;
+       printk(BIOS_DEBUG, "Enabling HPET @0x%lx\n", hpet_address);
 }
-#endif
+
+unsigned pm_base=0;
 
 static void lpc_init(device_t dev)
 {
-       uint8_t byte;
-       uint8_t byte_old;
-       int on;
-       int nmi_option;
+       uint8_t byte, byte_old;
+       int on, nmi_option;
 
        lpc_common_init(dev);
 
+       pm_base = pci_read_config32(dev, 0x60) & 0xff00;
+       printk(BIOS_INFO, "%s: pm_base = %x \n", __func__, pm_base);
+
 #if CK804_CHIP_REV==1
-       if(dev->bus->secondary!=1) return;
+       if (dev->bus->secondary != 1)
+               return;
 #endif
 
 #if 0
-       /* posted memory write enable */
+       /* Posted memory write enable */
        byte = pci_read_config8(dev, 0x46);
-       pci_write_config8(dev, 0x46, byte | (1<<0));
-
+       pci_write_config8(dev, 0x46, byte | (1 << 0));
 #endif
-       /* power after power fail */
 
-       on = MAINBOARD_POWER_ON_AFTER_POWER_FAIL;
+       /* power after power fail */
+       on = CONFIG_MAINBOARD_POWER_ON_AFTER_POWER_FAIL;
        get_option(&on, "power_on_after_fail");
        byte = pci_read_config8(dev, PREVIOUS_POWER_STATE);
        byte &= ~0x40;
-       if (!on) {
+       if (!on)
                byte |= 0x40;
-       }
        pci_write_config8(dev, PREVIOUS_POWER_STATE, byte);
-       printk_info("set power %s after power fail\n", on?"on":"off");
+       printk(BIOS_INFO, "set power %s after power fail\n", on ? "on" : "off");
 
-       /* Throttle the CPU speed down for testing */
+       /* Throttle the CPU speed down for testing. */
        on = SLOW_CPU_OFF;
        get_option(&on, "slow_cpu");
-       if(on) {
+       if (on) {
                uint16_t pm10_bar;
                uint32_t dword;
-               pm10_bar = (pci_read_config16(dev, 0x60)&0xff00);
-               outl(((on<<1)+0x10)  ,(pm10_bar + 0x10));
+               pm10_bar = (pci_read_config16(dev, 0x60) & 0xff00);
+               outl(((on << 1) + 0x10), (pm10_bar + 0x10));
                dword = inl(pm10_bar + 0x10);
-               on = 8-on;
-               printk_debug("Throttling CPU %2d.%1.1d percent.\n",
-                               (on*12)+(on>>1),(on&1)*5);
+               on = 8 - on;
+               printk(BIOS_DEBUG, "Throttling CPU %2d.%1.1d percent.\n",
+                            (on * 12) + (on >> 1), (on & 1) * 5);
        }
-
 #if 0
 // default is enabled
-       /* Enable Port 92 fast reset */
+       /* Enable Port 92 fast reset. */
        byte = pci_read_config8(dev, 0xe8);
        byte |= ~(1 << 3);
        pci_write_config8(dev, 0xe8, byte);
 #endif
 
-       /* Enable Error reporting */
-       /* Set up sync flood detected */
+       /* Enable Error reporting. */
+       /* Set up sync flood detected. */
        byte = pci_read_config8(dev, 0x47);
        byte |= (1 << 1);
        pci_write_config8(dev, 0x47, byte);
 
-       /* Set up NMI on errors */
-       byte = inb(0x70); // RTC70
+       /* Set up NMI on errors. */
+       byte = inb(0x70);               /* RTC70 */
        byte_old = byte;
        nmi_option = NMI_OFF;
        get_option(&nmi_option, "nmi");
        if (nmi_option) {
-               byte &= ~(1 << 7); /* set NMI */
+               byte &= ~(1 << 7); /* Set NMI. */
        } else {
-               byte |= ( 1 << 7); // Can not mask NMI from PCI-E and NMI_NOW
-       }
-       if( byte != byte_old) {
-               outb(0x70, byte);
+               byte |= (1 << 7); /* Can't mask NMI from PCI-E and NMI_NOW. */
        }
+       if (byte != byte_old)
+               outb(byte, 0x70);
 
-       /* Initialize the real time clock */
+       /* Initialize the real time clock (RTC). */
        rtc_init(0);
 
-       /* Initialize isa dma */
+       /* Initialize ISA DMA. */
        isa_dma_init();
 
-       /* Initialize the High Precision Event Timers */
-//     enable_hpet(dev);
+       /* Initialize the High Precision Event Timers (HPET). */
+       enable_hpet(dev);
 
        rom_dummy_write(dev);
-
 }
 
 static void ck804_lpc_read_resources(device_t dev)
@@ -271,88 +188,94 @@ static void ck804_lpc_read_resources(device_t dev)
        struct resource *res;
        unsigned long index;
 
-       /* Get the normal pci resources of this device */
-       pci_dev_read_resources(dev); // We got one for APIC, or one more for TRAP
+       /* Get the normal PCI resources of this device. */
+       /* We got one for APIC, or one more for TRAP. */
+       pci_dev_read_resources(dev);
 
-       /* Get Resource for ACPI, SYSTEM_CONTROL, ANALOG_CONTROL */
-       for (index = 0x60; index <= 0x68; index+=4) { // We got another 3.
+       /* Get resource for ACPI, SYSTEM_CONTROL, ANALOG_CONTROL. */
+       for (index = 0x60; index <= 0x68; index += 4)   /* We got another 3. */
                pci_get_resource(dev, index);
-       }
        compact_resources(dev);
 
-       /* Add an extra subtractive resource for both memory and I/O */
+       /* Add an extra subtractive resource for both memory and I/O. */
        res = new_resource(dev, IOINDEX_SUBTRACTIVE(0, 0));
-       res->flags = IORESOURCE_IO | IORESOURCE_SUBTRACTIVE | IORESOURCE_ASSIGNED;
+       res->base = 0;
+       res->size = 0x1000;
+       res->flags = IORESOURCE_IO | IORESOURCE_SUBTRACTIVE |
+                    IORESOURCE_ASSIGNED | IORESOURCE_FIXED;
 
        res = new_resource(dev, IOINDEX_SUBTRACTIVE(1, 0));
-       res->flags = IORESOURCE_MEM | IORESOURCE_SUBTRACTIVE | IORESOURCE_ASSIGNED;
-
+       res->base = 0xff800000;
+       res->size = 0x00800000; /* 8 MB for flash */
+       res->flags = IORESOURCE_MEM | IORESOURCE_SUBTRACTIVE |
+                    IORESOURCE_ASSIGNED | IORESOURCE_FIXED;
+
+       res = new_resource(dev, 3); /* IOAPIC */
+       res->base = IO_APIC_ADDR;
+       res->size = 0x00001000;
+       res->flags = IORESOURCE_MEM | IORESOURCE_ASSIGNED | IORESOURCE_FIXED;
 }
 
 /**
- * @brief Enable resources for children devices
- *
- * @param dev the device whos children's resources are to be enabled
+ * Enable resources for children devices.
  *
- * This function is call by the global enable_resources() indirectly via the
+ * This function is called by the global enable_resources() indirectly via the
  * device_operation::enable_resources() method of devices.
  *
- * Indirect mutual recursion:
- *      enable_childrens_resources() -> enable_resources()
- *      enable_resources() -> device_operation::enable_resources()
- *      device_operation::enable_resources() -> enable_children_resources()
  */
 static void ck804_lpc_enable_childrens_resources(device_t dev)
 {
-       unsigned link;
+       struct bus *link;
        uint32_t reg, reg_var[4];
-       int i;
-       int var_num = 0;
+       int i, var_num = 0;
 
        reg = pci_read_config32(dev, 0xa0);
 
-       for (link = 0; link < dev->links; link++) {
+       for (link = dev->link_list; link; link = link->next) {
                device_t child;
-               for (child = dev->link[link].children; child; child = child->sibling) {
-                       enable_resources(child);
-                       if(child->have_resources && (child->path.type == DEVICE_PATH_PNP)) {
-                               for(i=0;i<child->resources;i++) {
-                                       struct resource *res;
-                                       unsigned long base, end; // don't need long long
-                                       res = &child->resource[i];
-                                       if(!(res->flags & IORESOURCE_IO)) continue;
+               for (child = link->children; child; child = child->sibling) {
+                       if (child->enabled && (child->path.type == DEVICE_PATH_PNP)) {
+                               struct resource *res;
+                               for (res = child->resource_list; res; res = res->next) {
+                                       unsigned long base, end;        // don't need long long
+                                       if (!(res->flags & IORESOURCE_IO))
+                                               continue;
                                        base = res->base;
                                        end = resource_end(res);
-                                       printk_debug("ck804 lpc decode:%s, base=0x%08x, end=0x%08x\r\n",dev_path(child),base, end);
-                                       switch(base) {
-                                       case 0x3f8: // COM1
-                                               reg |= (1<<0);  break;
-                                       case 0x2f8: // COM2
-                                               reg |= (1<<1);  break;
-                                       case 0x378: // Parallal 1
-                                               reg |= (1<<24); break;
-                                       case 0x3f0: // FD0
-                                               reg |= (1<<20); break;
-                                       case 0x220:  // Aduio 0
-                                               reg |= (1<<8);  break;
-                                       case 0x300:  // Midi 0
-                                               reg |= (1<<12); break;
+                                       printk(BIOS_DEBUG, "ck804 lpc decode:%s, base=0x%08lx, end=0x%08lx\n", dev_path(child), base, end);
+                                       switch (base) {
+                                       case 0x3f8:     // COM1
+                                               reg |= (1 << 0);
+                                               break;
+                                       case 0x2f8:     // COM2
+                                               reg |= (1 << 1);
+                                               break;
+                                       case 0x378:     // Parallel 1
+                                               reg |= (1 << 24);
+                                               break;
+                                       case 0x3f0:     // FD0
+                                               reg |= (1 << 20);
+                                               break;
+                                       case 0x220:     // Audio 0
+                                               reg |= (1 << 8);
+                                               break;
+                                       case 0x300:     // Midi 0
+                                               reg |= (1 << 12);
+                                               break;
                                        }
-                                       if( base == 0x290 || base >= 0x400) {
-                                               if(var_num>=4) continue; // only 4 var ; compact them ?
-                                               reg |= (1<<(28+var_num));
-                                               reg_var[var_num++] = (base & 0xffff)|((end & 0xffff)<<16);
+                                       if (base == 0x290 || base >= 0x400) {
+                                               if (var_num >= 4)
+                                                       continue;       // only 4 var ; compact them ?
+                                               reg |= (1 << (28 + var_num));
+                                               reg_var[var_num++] = (base & 0xffff) | ((end & 0xffff) << 16);
                                        }
                                }
                        }
                }
        }
        pci_write_config32(dev, 0xa0, reg);
-       for(i=0;i<var_num;i++) {
-               pci_write_config32(dev, 0xa8 + i*4, reg_var[i]);
-       }
-
-
+       for (i = 0; i < var_num; i++)
+               pci_write_config32(dev, 0xa8 + i * 4, reg_var[i]);
 }
 
 static void ck804_lpc_enable_resources(device_t dev)
@@ -364,22 +287,23 @@ static void ck804_lpc_enable_resources(device_t dev)
 static void lpci_set_subsystem(device_t dev, unsigned vendor, unsigned device)
 {
        pci_write_config32(dev, 0x40,
-               ((device & 0xffff) << 16) | (vendor & 0xffff));
+                          ((device & 0xffff) << 16) | (vendor & 0xffff));
 }
 
 static struct pci_operations lops_pci = {
        .set_subsystem = lpci_set_subsystem,
 };
 
-static struct device_operations lpc_ops  = {
+static struct device_operations lpc_ops = {
        .read_resources   = ck804_lpc_read_resources,
        .set_resources    = pci_dev_set_resources,
        .enable_resources = ck804_lpc_enable_resources,
        .init             = lpc_init,
        .scan_bus         = scan_static_bus,
-//     .enable           = ck804_enable,
+       // .enable        = ck804_enable,
        .ops_pci          = &lops_pci,
 };
+
 static const struct pci_driver lpc_driver __pci_driver = {
        .ops    = &lpc_ops,
        .vendor = PCI_VENDOR_ID_NVIDIA,
@@ -399,12 +323,12 @@ static const struct pci_driver lpc_driver_slave __pci_driver = {
        .device = PCI_DEVICE_ID_NVIDIA_CK804_SLAVE,
 };
 #else
-static struct device_operations lpc_slave_ops  = {
+static struct device_operations lpc_slave_ops = {
        .read_resources   = ck804_lpc_read_resources,
        .set_resources    = pci_dev_set_resources,
        .enable_resources = pci_dev_enable_resources,
        .init             = lpc_slave_init,
-//     .enable           = ck804_enable,
+       // .enable        = ck804_enable,
        .ops_pci          = &lops_pci,
 };