2010-03-14 18:01:08 +01:00
|
|
|
/*
|
|
|
|
* This file is part of the coreboot project.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2004 Ron G. Minnich
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
2004-08-24 18:20:46 +02:00
|
|
|
#include <console/console.h>
|
|
|
|
#include <device/device.h>
|
|
|
|
#include <device/pci.h>
|
|
|
|
#include <device/pci_ids.h>
|
2010-02-27 02:50:21 +01:00
|
|
|
#include "i82801dx.h"
|
2004-08-24 18:20:46 +02:00
|
|
|
|
2010-02-27 02:50:21 +01:00
|
|
|
void i82801dx_enable(device_t dev)
|
2004-08-24 18:20:46 +02:00
|
|
|
{
|
2005-09-21 15:49:44 +02:00
|
|
|
unsigned int index = 0;
|
|
|
|
uint8_t bHasDisableBit = 0;
|
|
|
|
uint16_t cur_disable_mask, new_disable_mask;
|
|
|
|
|
2010-03-14 18:01:08 +01:00
|
|
|
// all 82801dbm devices are in bus 0
|
|
|
|
unsigned int devfn = PCI_DEVFN(0x1f, 0); // lpc
|
|
|
|
device_t lpc_dev = dev_find_slot(0, devfn); // 0
|
2005-09-21 15:49:44 +02:00
|
|
|
if (!lpc_dev)
|
2004-08-24 18:20:46 +02:00
|
|
|
return;
|
2005-09-21 15:49:44 +02:00
|
|
|
|
|
|
|
// Calculate disable bit position for specified device:function
|
|
|
|
// NOTE: For ICH-4, only the following devices can be disabled:
|
2010-03-14 18:01:08 +01:00
|
|
|
// D31: F0, F1, F3, F5, F6,
|
|
|
|
// D29: F0, F1, F2, F7
|
2005-09-21 15:49:44 +02:00
|
|
|
|
2010-03-14 18:01:08 +01:00
|
|
|
if (PCI_SLOT(dev->path.pci.devfn) == 31) {
|
|
|
|
index = PCI_FUNC(dev->path.pci.devfn);
|
2005-09-21 15:49:44 +02:00
|
|
|
|
|
|
|
switch (index) {
|
2010-03-14 18:01:08 +01:00
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
case 3:
|
|
|
|
case 5:
|
|
|
|
case 6:
|
|
|
|
bHasDisableBit = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
2005-09-21 15:49:44 +02:00
|
|
|
};
|
2010-03-14 18:01:08 +01:00
|
|
|
|
2005-09-21 15:49:44 +02:00
|
|
|
if (index == 0)
|
2010-03-14 18:01:08 +01:00
|
|
|
index = 14; // D31:F0 bit is an exception
|
2005-09-21 15:49:44 +02:00
|
|
|
|
2010-03-14 18:01:08 +01:00
|
|
|
} else if (PCI_SLOT(dev->path.pci.devfn) == 29) {
|
|
|
|
index = 8 + PCI_FUNC(dev->path.pci.devfn);
|
2005-09-21 15:49:44 +02:00
|
|
|
|
2010-03-14 18:01:08 +01:00
|
|
|
if ((PCI_FUNC(dev->path.pci.devfn) < 3)
|
|
|
|
|| (PCI_FUNC(dev->path.pci.devfn) == 7))
|
2005-09-21 15:49:44 +02:00
|
|
|
bHasDisableBit = 1;
|
2010-03-14 18:01:08 +01:00
|
|
|
}
|
2005-09-21 15:49:44 +02:00
|
|
|
|
|
|
|
if (bHasDisableBit) {
|
|
|
|
cur_disable_mask = pci_read_config16(lpc_dev, FUNC_DIS);
|
2010-03-14 18:01:08 +01:00
|
|
|
new_disable_mask = cur_disable_mask & ~(1 << index); // enable it
|
2005-09-21 15:49:44 +02:00
|
|
|
if (!dev->enabled) {
|
2010-03-14 18:01:08 +01:00
|
|
|
new_disable_mask |= (1 << index); // disable it
|
2005-09-21 15:49:44 +02:00
|
|
|
}
|
|
|
|
if (new_disable_mask != cur_disable_mask) {
|
|
|
|
pci_write_config16(lpc_dev, FUNC_DIS, new_disable_mask);
|
2004-08-24 18:20:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-27 02:50:21 +01:00
|
|
|
struct chip_operations southbridge_intel_i82801dx_ops = {
|
|
|
|
CHIP_NAME("Intel ICH4/ICH4-M (82801Dx) Series Southbridge")
|
2010-03-14 18:01:08 +01:00
|
|
|
.enable_dev = i82801dx_enable,
|
2004-08-24 18:20:46 +02:00
|
|
|
};
|