Patch from tqh, modified by myself: enable PCIe snooping through vendor specific registers (ATI and nVidia).

git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@28838 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Jérôme Duval 2009-01-01 14:58:10 +00:00
parent e4737a9260
commit 321e633fa4
2 changed files with 38 additions and 8 deletions

View File

@ -48,6 +48,17 @@ static const struct {
};
static inline void
update_pci_register(hda_controller* controller, uint8 reg, uint8 mask, uint8 value)
{
uint8 tmp = (gPci->read_pci_config)(controller->pci_info.bus,
controller->pci_info.device, controller->pci_info.function, reg, 1);
(gPci->write_pci_config)(controller->pci_info.bus,
controller->pci_info.device, controller->pci_info.function,
reg, 1, (tmp & mask) | value);
}
static inline rirb_t&
current_rirb(hda_controller *controller)
{
@ -700,8 +711,7 @@ hda_hw_init(hda_controller* controller)
{
uint16 capabilities, stateStatus, cmd;
status_t status;
uint8 tcsel;
/* Map MMIO registers */
controller->regs_area = map_physical_memory("hda_hw_regs",
(void*)controller->pci_info.u.h0.base_registers[0],
@ -729,11 +739,22 @@ hda_hw_init(hda_controller* controller)
goto no_irq;
/* TCSEL is reset to TC0 (clear 0-2 bits) */
tcsel = (gPci->read_pci_config)(controller->pci_info.bus,
controller->pci_info.device, controller->pci_info.function, PCI_HDA_TCSEL, 1);
(gPci->write_pci_config)(controller->pci_info.bus,
controller->pci_info.device, controller->pci_info.function,
PCI_HDA_TCSEL, 1, tcsel & 0xf8);
update_pci_register(controller, PCI_HDA_TCSEL, PCI_HDA_TCSEL_MASK, 0);
/* Enable snooping for ATI and Nvidia, right now for all their hda-devices,
but only based on guessing. */
switch (controller->pci_info.vendor_id) {
/* NVIDIA */
case 0x10de:
update_pci_register(controller, NVIDIA_HDA_TRANSREG,
NVIDIA_HDA_TRANSREG_MASK, NVIDIA_HDA_ENABLE_COHBITS);
break;
/* ATI */
case 0x1002:
update_pci_register(controller, ATI_HDA_MISC_CNTR2,
ATI_HDA_MISC_CNTR2_MASK, ATI_HDA_ENABLE_SNOOP);
break;
}
capabilities = controller->Read16(HDAC_GLOBAL_CAP);
controller->num_input_streams = GLOBAL_CAP_INPUT_STREAMS(capabilities);
@ -762,7 +783,7 @@ hda_hw_init(hda_controller* controller)
dprintf("hda: init_corb_rirb_pos failed\n");
goto corb_rirb_failed;
}
controller->Write16(HDAC_WAKE_ENABLE, 0x7fff);
/* Enable controller interrupts */

View File

@ -133,6 +133,15 @@
/* PCI space register definitions */
#define PCI_HDA_TCSEL 0x44
#define PCI_HDA_TCSEL_MASK 0xf8
#define ATI_HDA_MISC_CNTR2 0x42
#define ATI_HDA_MISC_CNTR2_MASK 0xf8
#define ATI_HDA_ENABLE_SNOOP 0x02
#define NVIDIA_HDA_TRANSREG 0x4e
#define NVIDIA_HDA_TRANSREG_MASK 0xf0
#define NVIDIA_HDA_ENABLE_COHBITS 0x0f
typedef uint32 corb_t;
typedef struct {