Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 1 | /* |
Stefan Reinauer | 7e61e45 | 2008-01-18 10:35:56 +0000 | [diff] [blame] | 2 | * This file is part of the coreboot project. |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 3 | * |
| 4 | * Copyright (C) 2003 Linux Networx |
| 5 | * Copyright (C) 2003 SuSE Linux AG |
| 6 | * Copyright (C) 2004 Tyan Computer |
| 7 | * Written by Yinghai Lu <yhlu@tyan.com> for Tyan Computer. |
| 8 | * Copyright (C) 2006,2007 AMD |
| 9 | * Written by Yinghai Lu <yinghai.lu@amd.com> for AMD. |
| 10 | * Copyright (C) 2007 Silicon Integrated Systems Corp. (SiS) |
| 11 | * Written by Morgan Tsai <my_tsai@sis.com> for SiS. |
| 12 | * |
| 13 | * This program is free software; you can redistribute it and/or modify |
| 14 | * it under the terms of the GNU General Public License as published by |
| 15 | * the Free Software Foundation; either version 2 of the License, or |
| 16 | * (at your option) any later version. |
| 17 | * |
| 18 | * This program is distributed in the hope that it will be useful, |
| 19 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 20 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 21 | * GNU General Public License for more details. |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 22 | */ |
| 23 | |
| 24 | #include <console/console.h> |
| 25 | #include <device/device.h> |
| 26 | #include <device/pci.h> |
| 27 | #include <device/pnp.h> |
| 28 | #include <device/pci_ids.h> |
| 29 | #include <device/pci_ops.h> |
| 30 | #include <pc80/mc146818rtc.h> |
| 31 | #include <pc80/isa-dma.h> |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 32 | #include <arch/io.h> |
Stefan Reinauer | 0401bd8 | 2010-01-16 18:31:34 +0000 | [diff] [blame] | 33 | #include <arch/ioapic.h> |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 34 | #include <cpu/x86/lapic.h> |
Carl-Daniel Hailfinger | 2ee6779 | 2008-10-01 12:52:52 +0000 | [diff] [blame] | 35 | #include <stdlib.h> |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 36 | #include "sis966.h" |
| 37 | #include <pc80/keyboard.h> |
| 38 | |
| 39 | #define NMI_OFF 0 |
| 40 | |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 41 | // 0x7a or e3 |
| 42 | #define PREVIOUS_POWER_STATE 0x7A |
| 43 | |
| 44 | #define MAINBOARD_POWER_OFF 0 |
| 45 | #define MAINBOARD_POWER_ON 1 |
| 46 | #define SLOW_CPU_OFF 0 |
| 47 | #define SLOW_CPU__ON 1 |
| 48 | |
Stefan Reinauer | 0867062 | 2009-06-30 15:17:49 +0000 | [diff] [blame] | 49 | #ifndef CONFIG_MAINBOARD_POWER_ON_AFTER_POWER_FAIL |
| 50 | #define CONFIG_MAINBOARD_POWER_ON_AFTER_POWER_FAIL MAINBOARD_POWER_ON |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 51 | #endif |
| 52 | |
Stefan Reinauer | eea66b7 | 2010-04-07 15:32:52 +0000 | [diff] [blame] | 53 | #undef SLAVE_INIT |
| 54 | |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 55 | static void lpc_common_init(device_t dev) |
| 56 | { |
| 57 | uint8_t byte; |
Kevin Paul Herbert | bde6d30 | 2014-12-24 18:43:20 -0800 | [diff] [blame] | 58 | void *ioapic_base; |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 59 | |
| 60 | /* IO APIC initialization */ |
| 61 | byte = pci_read_config8(dev, 0x74); |
| 62 | byte |= (1<<0); // enable APIC |
| 63 | pci_write_config8(dev, 0x74, byte); |
Kevin Paul Herbert | bde6d30 | 2014-12-24 18:43:20 -0800 | [diff] [blame] | 64 | ioapic_base = (void *)pci_read_config32(dev, PCI_BASE_ADDRESS_1); // 0x14 |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 65 | |
Stefan Reinauer | 0401bd8 | 2010-01-16 18:31:34 +0000 | [diff] [blame] | 66 | setup_ioapic(ioapic_base, 0); // Don't rename IO APIC ID |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 67 | } |
| 68 | |
Stefan Reinauer | eea66b7 | 2010-04-07 15:32:52 +0000 | [diff] [blame] | 69 | #ifdef SLAVE_INIT |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 70 | static void lpc_slave_init(device_t dev) |
| 71 | { |
| 72 | lpc_common_init(dev); |
| 73 | } |
Stefan Reinauer | eea66b7 | 2010-04-07 15:32:52 +0000 | [diff] [blame] | 74 | #endif |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 75 | |
| 76 | static void lpc_usb_legacy_init(device_t dev) |
| 77 | { |
| 78 | uint16_t acpi_base; |
| 79 | |
| 80 | acpi_base = (pci_read_config8(dev,0x75) << 8); |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 81 | |
| 82 | outb(inb(acpi_base + 0xbb) |0x80, acpi_base + 0xbb); |
| 83 | outb(inb(acpi_base + 0xba) |0x80, acpi_base + 0xba); |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 84 | } |
| 85 | |
| 86 | static void lpc_init(device_t dev) |
| 87 | { |
Edward O'Callaghan | def00be | 2014-04-30 05:01:52 +1000 | [diff] [blame] | 88 | uint8_t byte; |
| 89 | uint8_t byte_old; |
| 90 | int on; |
| 91 | int nmi_option; |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 92 | |
Edward O'Callaghan | def00be | 2014-04-30 05:01:52 +1000 | [diff] [blame] | 93 | printk(BIOS_DEBUG, "LPC_INIT -------->\n"); |
| 94 | pc_keyboard_init(); |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 95 | |
Edward O'Callaghan | def00be | 2014-04-30 05:01:52 +1000 | [diff] [blame] | 96 | lpc_usb_legacy_init(dev); |
| 97 | lpc_common_init(dev); |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 98 | |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 99 | /* power after power fail */ |
| 100 | |
Morgan Tsai | 218c265 | 2007-11-02 16:09:58 +0000 | [diff] [blame] | 101 | |
Stefan Reinauer | 0867062 | 2009-06-30 15:17:49 +0000 | [diff] [blame] | 102 | on = CONFIG_MAINBOARD_POWER_ON_AFTER_POWER_FAIL; |
Luc Verhaegen | a9c5ea0 | 2009-06-03 14:19:33 +0000 | [diff] [blame] | 103 | get_option(&on, "power_on_after_fail"); |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 104 | byte = pci_read_config8(dev, PREVIOUS_POWER_STATE); |
| 105 | byte &= ~0x40; |
| 106 | if (!on) { |
| 107 | byte |= 0x40; |
| 108 | } |
| 109 | pci_write_config8(dev, PREVIOUS_POWER_STATE, byte); |
Stefan Reinauer | c02b4fc | 2010-03-22 11:42:32 +0000 | [diff] [blame] | 110 | printk(BIOS_INFO, "set power %s after power fail\n", on?"on":"off"); |
Morgan Tsai | 218c265 | 2007-11-02 16:09:58 +0000 | [diff] [blame] | 111 | |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 112 | /* Throttle the CPU speed down for testing */ |
| 113 | on = SLOW_CPU_OFF; |
Luc Verhaegen | a9c5ea0 | 2009-06-03 14:19:33 +0000 | [diff] [blame] | 114 | get_option(&on, "slow_cpu"); |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 115 | if(on) { |
| 116 | uint16_t pm10_bar; |
| 117 | uint32_t dword; |
| 118 | pm10_bar = (pci_read_config16(dev, 0x60)&0xff00); |
| 119 | outl(((on<<1)+0x10) ,(pm10_bar + 0x10)); |
| 120 | dword = inl(pm10_bar + 0x10); |
| 121 | on = 8-on; |
Stefan Reinauer | c02b4fc | 2010-03-22 11:42:32 +0000 | [diff] [blame] | 122 | printk(BIOS_DEBUG, "Throttling CPU %2d.%1.1d percent.\n", |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 123 | (on*12)+(on>>1),(on&1)*5); |
| 124 | } |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 125 | |
Morgan Tsai | 218c265 | 2007-11-02 16:09:58 +0000 | [diff] [blame] | 126 | /* Enable Error reporting */ |
| 127 | /* Set up sync flood detected */ |
| 128 | byte = pci_read_config8(dev, 0x47); |
| 129 | byte |= (1 << 1); |
| 130 | pci_write_config8(dev, 0x47, byte); |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 131 | |
Morgan Tsai | 218c265 | 2007-11-02 16:09:58 +0000 | [diff] [blame] | 132 | /* Set up NMI on errors */ |
| 133 | byte = inb(0x70); // RTC70 |
| 134 | byte_old = byte; |
| 135 | nmi_option = NMI_OFF; |
Luc Verhaegen | a9c5ea0 | 2009-06-03 14:19:33 +0000 | [diff] [blame] | 136 | get_option(&nmi_option, "nmi"); |
Morgan Tsai | 218c265 | 2007-11-02 16:09:58 +0000 | [diff] [blame] | 137 | if (nmi_option) { |
| 138 | byte &= ~(1 << 7); /* set NMI */ |
| 139 | } else { |
| 140 | byte |= ( 1 << 7); // Can not mask NMI from PCI-E and NMI_NOW |
| 141 | } |
| 142 | if( byte != byte_old) { |
Ed Swierk | e42e142 | 2009-07-10 15:05:35 +0000 | [diff] [blame] | 143 | outb(byte, 0x70); |
Morgan Tsai | 218c265 | 2007-11-02 16:09:58 +0000 | [diff] [blame] | 144 | } |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 145 | |
Morgan Tsai | 218c265 | 2007-11-02 16:09:58 +0000 | [diff] [blame] | 146 | /* Initialize the real time clock */ |
Gabe Black | b3f08c6 | 2014-04-30 17:12:25 -0700 | [diff] [blame] | 147 | cmos_init(0); |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 148 | |
Morgan Tsai | 218c265 | 2007-11-02 16:09:58 +0000 | [diff] [blame] | 149 | /* Initialize isa dma */ |
| 150 | isa_dma_init(); |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 151 | |
Stefan Reinauer | c02b4fc | 2010-03-22 11:42:32 +0000 | [diff] [blame] | 152 | printk(BIOS_DEBUG, "LPC_INIT <--------\n"); |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 153 | } |
| 154 | |
| 155 | static void sis966_lpc_read_resources(device_t dev) |
| 156 | { |
| 157 | struct resource *res; |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 158 | |
| 159 | /* Get the normal pci resources of this device */ |
| 160 | pci_dev_read_resources(dev); // We got one for APIC, or one more for TRAP |
| 161 | |
Myles Watson | 29cc9ed | 2009-07-02 18:56:24 +0000 | [diff] [blame] | 162 | /* Add an extra subtractive resource for both memory and I/O. */ |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 163 | res = new_resource(dev, IOINDEX_SUBTRACTIVE(0, 0)); |
Myles Watson | 29cc9ed | 2009-07-02 18:56:24 +0000 | [diff] [blame] | 164 | res->base = 0; |
| 165 | res->size = 0x1000; |
| 166 | res->flags = IORESOURCE_IO | IORESOURCE_SUBTRACTIVE | |
| 167 | IORESOURCE_ASSIGNED | IORESOURCE_FIXED; |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 168 | |
| 169 | res = new_resource(dev, IOINDEX_SUBTRACTIVE(1, 0)); |
Myles Watson | 29cc9ed | 2009-07-02 18:56:24 +0000 | [diff] [blame] | 170 | res->base = 0xff800000; |
| 171 | res->size = 0x00800000; /* 8 MB for flash */ |
| 172 | res->flags = IORESOURCE_MEM | IORESOURCE_SUBTRACTIVE | |
| 173 | IORESOURCE_ASSIGNED | IORESOURCE_FIXED; |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 174 | |
Myles Watson | 29cc9ed | 2009-07-02 18:56:24 +0000 | [diff] [blame] | 175 | res = new_resource(dev, 3); /* IOAPIC */ |
Uwe Hermann | 74d1a6e | 2010-10-12 17:34:08 +0000 | [diff] [blame] | 176 | res->base = IO_APIC_ADDR; |
Myles Watson | 29cc9ed | 2009-07-02 18:56:24 +0000 | [diff] [blame] | 177 | res->size = 0x00001000; |
| 178 | res->flags = IORESOURCE_MEM | IORESOURCE_ASSIGNED | IORESOURCE_FIXED; |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 179 | } |
| 180 | |
| 181 | /** |
Uwe Hermann | b69cb5a | 2010-10-26 22:46:43 +0000 | [diff] [blame] | 182 | * Enable resources for children devices. |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 183 | * |
Martin Roth | 226db05 | 2014-12-09 13:51:49 -0700 | [diff] [blame] | 184 | * @param dev The device whose children's resources are to be enabled. |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 185 | */ |
| 186 | static void sis966_lpc_enable_childrens_resources(device_t dev) |
| 187 | { |
Myles Watson | 894a347 | 2010-06-09 22:41:35 +0000 | [diff] [blame] | 188 | struct bus *link; |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 189 | uint32_t reg, reg_var[4]; |
| 190 | int i; |
| 191 | int var_num = 0; |
| 192 | |
| 193 | reg = pci_read_config32(dev, 0xa0); |
| 194 | |
Myles Watson | 894a347 | 2010-06-09 22:41:35 +0000 | [diff] [blame] | 195 | for (link = dev->link_list; link; link = link->next) { |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 196 | device_t child; |
Myles Watson | 894a347 | 2010-06-09 22:41:35 +0000 | [diff] [blame] | 197 | for (child = link->children; child; child = child->sibling) { |
Myles Watson | 29cc9ed | 2009-07-02 18:56:24 +0000 | [diff] [blame] | 198 | if(child->enabled && (child->path.type == DEVICE_PATH_PNP)) { |
Myles Watson | c25cc11 | 2010-05-21 14:33:48 +0000 | [diff] [blame] | 199 | struct resource *res; |
| 200 | for(res = child->resource_list; res; res = res->next) { |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 201 | unsigned long base, end; // don't need long long |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 202 | if(!(res->flags & IORESOURCE_IO)) continue; |
| 203 | base = res->base; |
| 204 | end = resource_end(res); |
Myles Watson | 08e0fb8 | 2010-03-22 16:33:25 +0000 | [diff] [blame] | 205 | printk(BIOS_DEBUG, "sis966 lpc decode:%s, base=0x%08lx, end=0x%08lx\n",dev_path(child),base, end); |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 206 | switch(base) { |
| 207 | case 0x3f8: // COM1 |
| 208 | reg |= (1<<0); break; |
| 209 | case 0x2f8: // COM2 |
| 210 | reg |= (1<<1); break; |
Martin Roth | 226db05 | 2014-12-09 13:51:49 -0700 | [diff] [blame] | 211 | case 0x378: // Parallel 1 |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 212 | reg |= (1<<24); break; |
| 213 | case 0x3f0: // FD0 |
| 214 | reg |= (1<<20); break; |
Martin Roth | 226db05 | 2014-12-09 13:51:49 -0700 | [diff] [blame] | 215 | case 0x220: // Audio 0 |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 216 | reg |= (1<<8); break; |
| 217 | case 0x300: // Midi 0 |
| 218 | reg |= (1<<12); break; |
| 219 | } |
| 220 | if( (base == 0x290) || (base >= 0x400)) { |
| 221 | if(var_num>=4) continue; // only 4 var ; compact them ? |
| 222 | reg |= (1<<(28+var_num)); |
| 223 | reg_var[var_num++] = (base & 0xffff)|((end & 0xffff)<<16); |
| 224 | } |
| 225 | } |
| 226 | } |
| 227 | } |
| 228 | } |
| 229 | pci_write_config32(dev, 0xa0, reg); |
| 230 | for(i=0;i<var_num;i++) { |
| 231 | pci_write_config32(dev, 0xa8 + i*4, reg_var[i]); |
| 232 | } |
| 233 | |
| 234 | |
| 235 | } |
| 236 | |
| 237 | static void sis966_lpc_enable_resources(device_t dev) |
| 238 | { |
| 239 | pci_dev_enable_resources(dev); |
| 240 | sis966_lpc_enable_childrens_resources(dev); |
| 241 | } |
| 242 | |
| 243 | static void lpci_set_subsystem(device_t dev, unsigned vendor, unsigned device) |
| 244 | { |
| 245 | pci_write_config32(dev, 0x40, |
| 246 | ((device & 0xffff) << 16) | (vendor & 0xffff)); |
| 247 | } |
| 248 | |
| 249 | static struct pci_operations lops_pci = { |
| 250 | .set_subsystem = lpci_set_subsystem, |
| 251 | }; |
| 252 | |
| 253 | static struct device_operations lpc_ops = { |
| 254 | .read_resources = sis966_lpc_read_resources, |
| 255 | .set_resources = pci_dev_set_resources, |
| 256 | .enable_resources = sis966_lpc_enable_resources, |
| 257 | .init = lpc_init, |
Kyösti Mälkki | d0e212c | 2015-02-26 20:47:47 +0200 | [diff] [blame] | 258 | .scan_bus = scan_lpc_bus, |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 259 | // .enable = sis966_enable, |
| 260 | .ops_pci = &lops_pci, |
| 261 | }; |
Stefan Reinauer | eea66b7 | 2010-04-07 15:32:52 +0000 | [diff] [blame] | 262 | |
Stefan Reinauer | 83b52e7 | 2007-10-30 02:17:49 +0000 | [diff] [blame] | 263 | static const struct pci_driver lpc_driver __pci_driver = { |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 264 | .ops = &lpc_ops, |
| 265 | .vendor = PCI_VENDOR_ID_SIS, |
| 266 | .device = PCI_DEVICE_ID_SIS_SIS966_LPC, |
| 267 | }; |
| 268 | |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 269 | #ifdef SLAVE_INIT // No device? |
Morgan Tsai | 1602dd5 | 2007-10-29 21:00:14 +0000 | [diff] [blame] | 270 | static struct device_operations lpc_slave_ops = { |
| 271 | .read_resources = sis966_lpc_read_resources, |
| 272 | .set_resources = pci_dev_set_resources, |
| 273 | .enable_resources = pci_dev_enable_resources, |
| 274 | .init = lpc_slave_init, |
| 275 | // .enable = sis966_enable, |
| 276 | .ops_pci = &lops_pci, |
| 277 | }; |
Stefan Reinauer | eea66b7 | 2010-04-07 15:32:52 +0000 | [diff] [blame] | 278 | #endif |