summaryrefslogtreecommitdiff
path: root/sys/arch/loongson
diff options
context:
space:
mode:
authorMiod Vallat <miod@cvs.openbsd.org>2009-12-25 22:11:09 +0000
committerMiod Vallat <miod@cvs.openbsd.org>2009-12-25 22:11:09 +0000
commitc8fc16bdb586df531725f3561028228475d80d4d (patch)
tree949932e0356f9d17eee51e1656de75ade4f91aed /sys/arch/loongson
parent7ac0ce90fe98c728d104ca49fd8b4745d3cf5a23 (diff)
By popular demand and peer pressure, check-in work in progress work to support
the Yeelong Lemote mips-based netbook. Kernel bits only for now, needs polishing; most of this work done during h2k9 last month, although the porting effort started earlier this year.
Diffstat (limited to 'sys/arch/loongson')
-rw-r--r--sys/arch/loongson/dev/glx.c761
1 files changed, 761 insertions, 0 deletions
diff --git a/sys/arch/loongson/dev/glx.c b/sys/arch/loongson/dev/glx.c
new file mode 100644
index 00000000000..ac56891fbce
--- /dev/null
+++ b/sys/arch/loongson/dev/glx.c
@@ -0,0 +1,761 @@
+/* $OpenBSD: glx.c,v 1.1 2009/12/25 22:11:08 miod Exp $ */
+
+/*
+ * Copyright (c) 2009 Miodrag Vallat.
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+/*
+ * AMD CS5536 PCI Mess
+ * XXX too many hardcoded numbers... need to expand glxreg.h
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/device.h>
+
+#include <machine/bus.h>
+
+#include <dev/pci/pcireg.h>
+#include <dev/pci/pcivar.h>
+#include <dev/pci/pcidevs.h>
+
+#include <dev/pci/pciidereg.h>
+#include <dev/pci/pciide_amd_reg.h>
+#include <dev/usb/ehcireg.h>
+#include <dev/usb/ohcireg.h>
+
+#include <loongson/dev/glxreg.h>
+#include <loongson/dev/glxvar.h>
+
+#include <loongson/dev/bonitovar.h>
+
+/*
+ * Since the purpose of this code is to present a different view of the
+ * PCI configuration space, it can not attach as a real device.
+ * (well it could, and then we'd have to attach a fake pci to it,
+ * and fake the configuration space accesses anyways - is it worth doing?)
+ *
+ * We just keep the `would-be softc' structure as global variables.
+ */
+
+static pci_chipset_tag_t glxbase_pc;
+static pcitag_t glxbase_tag;
+static int glxbase_dev;
+
+/* MSR access through PCI configuration space */
+#define PCI_MSR_CTRL 0x00f0
+#define PCI_MSR_ADDR 0x00f4
+#define PCI_MSR_LO32 0x00f8
+#define PCI_MSR_HI32 0x00fc
+
+int glx_pci_read_hook(void *, pci_chipset_tag_t, pcitag_t, int, pcireg_t *);
+int glx_pci_write_hook(void *, pci_chipset_tag_t, pcitag_t, int, pcireg_t);
+
+pcireg_t glx_get_status(void);
+pcireg_t glx_fn0_read(int);
+void glx_fn0_write(int, pcireg_t);
+pcireg_t glx_fn2_read(int);
+void glx_fn2_write(int, pcireg_t);
+pcireg_t glx_fn3_read(int);
+void glx_fn3_write(int, pcireg_t);
+pcireg_t glx_fn4_read(int);
+void glx_fn4_write(int, pcireg_t);
+pcireg_t glx_fn5_read(int);
+void glx_fn5_write(int, pcireg_t);
+pcireg_t glx_fn6_read(int);
+void glx_fn6_write(int, pcireg_t);
+pcireg_t glx_fn7_read(int);
+void glx_fn7_write(int, pcireg_t);
+
+void
+glx_init(pci_chipset_tag_t pc, pcitag_t tag, int dev)
+{
+ uint64_t msr;
+
+ glxbase_pc = pc;
+ glxbase_dev = dev;
+ glxbase_tag = tag;
+
+ /*
+ * Register PCI configuration hooks to make the various
+ * embedded devices visible as PCI subfunctions.
+ */
+
+ bonito_pci_hook(pc, NULL, glx_pci_read_hook, glx_pci_write_hook);
+
+ /*
+ * Perform some Geode intialization.
+ */
+ msr = rdmsr(DIVIL_BALL_OPTS); /* 0x71 */
+ wrmsr(DIVIL_BALL_OPTS, msr | 0x01);
+
+ /* route usb interrupt */
+ msr = rdmsr(PIC_YSEL_LOW);
+ msr &= ~(0xfUL << 8);
+ msr |= 11 << 8;
+ wrmsr(PIC_YSEL_LOW, msr);
+}
+
+uint64_t
+rdmsr(uint msr)
+{
+ uint64_t lo, hi;
+
+ if (glxbase_tag == 0)
+ panic("rdmsr invoked before glx initialization");
+
+ pci_conf_write(glxbase_pc, glxbase_tag, PCI_MSR_ADDR, msr);
+ lo = (uint32_t)pci_conf_read(glxbase_pc, glxbase_tag, PCI_MSR_LO32);
+ hi = (uint32_t)pci_conf_read(glxbase_pc, glxbase_tag, PCI_MSR_HI32);
+ return (hi << 32) | lo;
+}
+
+void
+wrmsr(uint msr, uint64_t value)
+{
+ if (glxbase_tag == 0)
+ panic("rdmsr invoked before glx initialization");
+
+ pci_conf_write(glxbase_pc, glxbase_tag, PCI_MSR_ADDR, msr);
+ pci_conf_write(glxbase_pc, glxbase_tag, PCI_MSR_LO32, (uint32_t)value);
+ pci_conf_write(glxbase_pc, glxbase_tag, PCI_MSR_HI32, value >> 32);
+}
+
+int
+glx_pci_read_hook(void *v, pci_chipset_tag_t pc, pcitag_t tag,
+ int offset, pcireg_t *data)
+{
+ int bus, dev, fn;
+
+ /*
+ * Do not get in the way of MSR programming
+ */
+ if (tag == glxbase_tag && offset >= PCI_MSR_CTRL)
+ return 0;
+
+ pci_decompose_tag(pc, tag, &bus, &dev, &fn);
+ if (bus != 0 || dev != glxbase_dev)
+ return 0;
+
+ *data = 0;
+
+ switch (fn) {
+ case 0: /* PCI-ISA bridge */
+ *data = glx_fn0_read(offset);
+ break;
+ case 1: /* Flash memory */
+ break;
+ case 2: /* IDE controller */
+ *data = glx_fn2_read(offset);
+ break;
+ case 3: /* AC97 codec */
+ break;
+ case 4: /* OHCI controller */
+ *data = glx_fn4_read(offset);
+ break;
+ case 5: /* EHCI controller */
+ *data = glx_fn5_read(offset);
+ break;
+ case 6: /* UDC */
+ break;
+ case 7: /* OTG */
+ break;
+ }
+
+ return 1;
+}
+
+int
+glx_pci_write_hook(void *v, pci_chipset_tag_t pc, pcitag_t tag,
+ int offset, pcireg_t data)
+{
+ int bus, dev, fn;
+
+ /*
+ * Do not get in the way of MSR programming
+ */
+ if (tag == glxbase_tag && offset >= PCI_MSR_CTRL)
+ return 0;
+
+ pci_decompose_tag(pc, tag, &bus, &dev, &fn);
+ if (bus != 0 || dev != glxbase_dev)
+ return 0;
+
+ switch (fn) {
+ case 0: /* PCI-ISA bridge */
+ glx_fn0_write(offset, data);
+ break;
+ case 1: /* Flash memory */
+ break;
+ case 2: /* IDE controller */
+ glx_fn2_write(offset, data);
+ break;
+ case 3: /* AC97 codec */
+ break;
+ case 4: /* OHCI controller */
+ glx_fn4_write(offset, data);
+ break;
+ case 5: /* EHCI controller */
+ glx_fn5_write(offset, data);
+ break;
+ case 6: /* USB UDC */
+ break;
+ case 7: /* USB OTG */
+ break;
+ }
+
+ return 1;
+}
+
+pcireg_t
+glx_get_status()
+{
+ uint64_t msr;
+ pcireg_t data;
+
+ data = 0;
+ msr = rdmsr(GLPCI_GLD_MSR_ERROR);
+ if (msr & (1UL << 5))
+ data |= PCI_COMMAND_PARITY_ENABLE;
+ data |= PCI_STATUS_66MHZ_SUPPORT |
+ PCI_STATUS_BACKTOBACK_SUPPORT | PCI_STATUS_DEVSEL_MEDIUM;
+ if (msr & (1UL << 21))
+ data |= PCI_STATUS_PARITY_DETECT;
+ if (msr & (1UL << 20))
+ data |= PCI_STATUS_TARGET_TARGET_ABORT;
+ if (msr & (1UL << 17))
+ data |= PCI_STATUS_MASTER_TARGET_ABORT;
+ if (msr & (1UL << 16))
+ data |= PCI_STATUS_MASTER_ABORT;
+
+ return data;
+}
+
+/*
+ * Function 0: PCI-ISA bridge
+ */
+
+static pcireg_t pcib_bar_sizes[(4 + PCI_MAPREG_END - PCI_MAPREG_START) / 4] = {
+ 0x008,
+ 0x100,
+ 0x040,
+ 0x020,
+ 0x080,
+ 0x020
+};
+
+static pcireg_t pcib_bar_values[(4 + PCI_MAPREG_END - PCI_MAPREG_START) / 4];
+
+static uint64_t pcib_bar_msr[(4 + PCI_MAPREG_END - PCI_MAPREG_START) / 4] = {
+ DIVIL_LBAR_SMB,
+ DIVIL_LBAR_GPIO,
+ DIVIL_LBAR_MFGPT,
+ DIVIL_LBAR_IRQ,
+ DIVIL_LBAR_PMS,
+ DIVIL_LBAR_ACPI
+};
+
+pcireg_t
+glx_fn0_read(int reg)
+{
+ uint64_t msr;
+ pcireg_t data;
+ int index;
+
+ switch (reg) {
+ case PCI_ID_REG:
+ case PCI_SUBSYS_ID_REG:
+ data = PCI_ID_CODE(PCI_VENDOR_AMD, PCI_PRODUCT_AMD_CS5536_PCIB);
+ break;
+ case PCI_COMMAND_STATUS_REG:
+ data = glx_get_status();
+ data |= PCI_COMMAND_MASTER_ENABLE;
+ msr = rdmsr(DIVIL_LBAR_SMB);
+ if (msr & (1UL << 32))
+ data |= PCI_COMMAND_IO_ENABLE;
+ break;
+ case PCI_CLASS_REG:
+ msr = rdmsr(GLCP_CHIP_REV_ID);
+ data = (PCI_CLASS_BRIDGE << PCI_CLASS_SHIFT) |
+ (PCI_SUBCLASS_BRIDGE_ISA << PCI_SUBCLASS_SHIFT) |
+ (msr & PCI_REVISION_MASK);
+ break;
+ case PCI_BHLC_REG:
+ msr = rdmsr(GLPCI_CTRL);
+ data = (0x80 << PCI_HDRTYPE_SHIFT) |
+ (((msr & 0xff00000000UL) >> 32) << PCI_LATTIMER_SHIFT) |
+ (0x08 << PCI_CACHELINE_SHIFT);
+ break;
+ case PCI_MAPREG_START + 0x00:
+ case PCI_MAPREG_START + 0x04:
+ case PCI_MAPREG_START + 0x08:
+ case PCI_MAPREG_START + 0x0c:
+ case PCI_MAPREG_START + 0x10:
+ case PCI_MAPREG_START + 0x14:
+ case PCI_MAPREG_START + 0x18:
+ index = (reg - PCI_MAPREG_START) / 4;
+ if (pcib_bar_msr[index] == 0)
+ data = 0;
+ else {
+ data = pcib_bar_values[index];
+ if (data == 0xffffffff)
+ data = PCI_MAPREG_IO_ADDR_MASK &
+ ~(pcib_bar_sizes[index] - 1);
+ else {
+ msr = rdmsr(pcib_bar_msr[index]);
+ data = msr & 0xfffff000;
+ }
+ if (data != 0)
+ data |= PCI_MAPREG_TYPE_IO;
+ }
+ break;
+ case PCI_INTERRUPT_REG:
+ data = (0x40 << PCI_MAX_LAT_SHIFT) |
+ (PCI_INTERRUPT_PIN_NONE << PCI_INTERRUPT_PIN_SHIFT);
+ break;
+ default:
+ data = 0;
+ break;
+ }
+
+ return data;
+}
+
+void
+glx_fn0_write(int reg, pcireg_t data)
+{
+ uint64_t msr;
+ int index;
+
+ switch (reg) {
+ case PCI_COMMAND_STATUS_REG:
+ for (index = 0; index < nitems(pcib_bar_msr); index++) {
+ if (pcib_bar_msr[index] == 0)
+ continue;
+ msr = rdmsr(pcib_bar_msr[index]);
+ if (data & PCI_COMMAND_IO_ENABLE)
+ msr |= 1UL << 32;
+ else
+ msr &= ~(1UL << 32);
+ wrmsr(pcib_bar_msr[index], msr);
+ }
+
+ msr = rdmsr(GLPCI_GLD_MSR_ERROR);
+ if (data & PCI_COMMAND_PARITY_ENABLE)
+ msr |= 1UL << 5;
+ else
+ msr &= ~(1UL << 5);
+ wrmsr(GLPCI_GLD_MSR_ERROR, msr);
+ break;
+ case PCI_BHLC_REG:
+ msr = rdmsr(GLPCI_CTRL);
+ msr &= 0xff00000000UL;
+ msr |= ((uint64_t)PCI_LATTIMER(data)) << 32;
+ break;
+ case PCI_MAPREG_START + 0x00:
+ case PCI_MAPREG_START + 0x04:
+ case PCI_MAPREG_START + 0x08:
+ case PCI_MAPREG_START + 0x0c:
+ case PCI_MAPREG_START + 0x10:
+ case PCI_MAPREG_START + 0x14:
+ case PCI_MAPREG_START + 0x18:
+ index = (reg - PCI_MAPREG_START) / 4;
+ if (data == 0xffffffff) {
+ pcib_bar_values[index] = data;
+ } else if (pcib_bar_msr[index] != 0) {
+ if ((data & PCI_MAPREG_TYPE_MASK) ==
+ PCI_MAPREG_TYPE_IO) {
+ data &= PCI_MAPREG_IO_ADDR_MASK;
+ data &= 0xfffff000;
+ wrmsr(pcib_bar_msr[index],
+ (0x0000f000UL << 32) | (1UL << 32) | data);
+ } else {
+ wrmsr(pcib_bar_msr[index], 0UL);
+ }
+ pcib_bar_values[index] = 0;
+ }
+ break;
+ }
+}
+
+/*
+ * Function 2: IDE Controller
+ */
+
+static pcireg_t pciide_bar_size = 0x10;
+static pcireg_t pciide_bar_value;
+
+pcireg_t
+glx_fn2_read(int reg)
+{
+ uint64_t msr;
+ pcireg_t data;
+
+ switch (reg) {
+ case PCI_ID_REG:
+ case PCI_SUBSYS_ID_REG:
+ data = PCI_ID_CODE(PCI_VENDOR_AMD, PCI_PRODUCT_AMD_CS5536_IDE);
+ break;
+ case PCI_COMMAND_STATUS_REG:
+ data = glx_get_status();
+ data |= PCI_COMMAND_IO_ENABLE;
+ msr = rdmsr(GLIU_PAE);
+ if ((msr & (0x3 << 4)) == 0x03)
+ data |= PCI_COMMAND_MASTER_ENABLE;
+ break;
+ case PCI_CLASS_REG:
+ msr = rdmsr(IDE_GLD_MSR_CAP);
+ data = (PCI_CLASS_MASS_STORAGE << PCI_CLASS_SHIFT) |
+ (PCI_SUBCLASS_MASS_STORAGE_IDE << PCI_SUBCLASS_SHIFT) |
+ (PCIIDE_INTERFACE_BUS_MASTER_DMA << PCI_INTERFACE_SHIFT) |
+ (msr & PCI_REVISION_MASK);
+ break;
+ case PCI_BHLC_REG:
+ msr = rdmsr(GLPCI_CTRL);
+ data = (0x00 << PCI_HDRTYPE_SHIFT) |
+ (((msr & 0xff00000000UL) >> 32) << PCI_LATTIMER_SHIFT) |
+ (0x08 << PCI_CACHELINE_SHIFT);
+ break;
+ case PCI_MAPREG_START + 0x10:
+ data = pciide_bar_value;
+ if (data == 0xffffffff)
+ data = PCI_MAPREG_IO_ADDR_MASK & ~(pciide_bar_size - 1);
+ else {
+ msr = rdmsr(IDE_IO_BAR);
+ data = msr & 0xfffffff0;
+ }
+ if (data != 0)
+ data |= PCI_MAPREG_TYPE_IO;
+ break;
+ case PCI_INTERRUPT_REG:
+ /* compat mode */
+ data = (0x40 << PCI_MAX_LAT_SHIFT) |
+ (PCI_INTERRUPT_PIN_NONE << PCI_INTERRUPT_PIN_SHIFT);
+ break;
+ /*
+ * The following registers are used by pciide(4)
+ */
+ case AMD756_CHANSTATUS_EN:
+ data = rdmsr(IDE_CFG);
+ break;
+ case AMD756_DATATIM:
+ data = rdmsr(IDE_DTC);
+ break;
+ case AMD756_UDMA:
+ data = rdmsr(IDE_ETC);
+ break;
+ default:
+ data = 0;
+ break;
+ }
+
+ return data;
+}
+
+void
+glx_fn2_write(int reg, pcireg_t data)
+{
+ uint64_t msr;
+
+ switch (reg) {
+ case PCI_COMMAND_STATUS_REG:
+ msr = rdmsr(GLIU_PAE);
+ if (data & PCI_COMMAND_MASTER_ENABLE)
+ msr |= 0x03 << 4;
+ else
+ msr &= ~(0x03 << 4);
+ wrmsr(GLIU_PAE, msr);
+ break;
+ case PCI_BHLC_REG:
+ msr = rdmsr(GLPCI_CTRL);
+ msr &= 0xff00000000UL;
+ msr |= ((uint64_t)PCI_LATTIMER(data)) << 32;
+ break;
+ case PCI_MAPREG_START + 0x10:
+ if (data == 0xffffffff) {
+ pciide_bar_value = data;
+ } else {
+ if ((data & PCI_MAPREG_TYPE_MASK) ==
+ PCI_MAPREG_TYPE_IO) {
+ data &= PCI_MAPREG_IO_ADDR_MASK;
+ msr = (uint32_t)data & 0xfffffff0;
+ wrmsr(IDE_IO_BAR, msr);
+ } else {
+ wrmsr(IDE_IO_BAR, 0);
+ }
+ pciide_bar_value = 0;
+ }
+ break;
+ /*
+ * The following registers are used by pciide(4)
+ */
+ case AMD756_CHANSTATUS_EN:
+ wrmsr(IDE_CFG, (uint32_t)data);
+ break;
+ case AMD756_DATATIM:
+ wrmsr(IDE_DTC, (uint32_t)data);
+ break;
+ case AMD756_UDMA:
+ wrmsr(IDE_ETC, (uint32_t)data);
+ break;
+ }
+}
+
+/*
+ * Function 4: OHCI Controller
+ */
+
+static pcireg_t ohci_bar_size = 0x1000;
+static pcireg_t ohci_bar_value;
+
+pcireg_t
+glx_fn4_read(int reg)
+{
+ uint64_t msr;
+ pcireg_t data;
+
+ switch (reg) {
+ case PCI_ID_REG:
+ case PCI_SUBSYS_ID_REG:
+ data = PCI_ID_CODE(PCI_VENDOR_AMD, PCI_PRODUCT_AMD_CS5536_OHCI);
+ break;
+ case PCI_COMMAND_STATUS_REG:
+ data = glx_get_status();
+ msr = rdmsr(USB_MSR_OHCB);
+ if (msr & (1UL << 34))
+ data |= PCI_COMMAND_MASTER_ENABLE;
+ if (msr & (1UL << 33))
+ data |= PCI_COMMAND_MEM_ENABLE;
+ break;
+ case PCI_CLASS_REG:
+ msr = rdmsr(USB_GLD_MSR_CAP);
+ data = (PCI_CLASS_SERIALBUS << PCI_CLASS_SHIFT) |
+ (PCI_SUBCLASS_SERIALBUS_USB << PCI_SUBCLASS_SHIFT) |
+ (PCI_INTERFACE_OHCI << PCI_INTERFACE_SHIFT) |
+ (msr & PCI_REVISION_MASK);
+ break;
+ case PCI_BHLC_REG:
+ msr = rdmsr(GLPCI_CTRL);
+ data = (0x00 << PCI_HDRTYPE_SHIFT) |
+ (((msr & 0xff00000000UL) >> 32) << PCI_LATTIMER_SHIFT) |
+ (0x08 << PCI_CACHELINE_SHIFT);
+ break;
+ case PCI_MAPREG_START + 0x00:
+ data = ohci_bar_value;
+ if (data == 0xffffffff)
+ data = PCI_MAPREG_MEM_ADDR_MASK & ~(ohci_bar_size - 1);
+ else {
+ msr = rdmsr(USB_MSR_OHCB);
+ data = msr & 0xffffff00;
+ }
+ if (data != 0)
+ data |= PCI_MAPREG_TYPE_MEM | PCI_MAPREG_MEM_TYPE_32BIT;
+ break;
+ case PCI_CAPLISTPTR_REG:
+ data = 0x40;
+ break;
+ case PCI_INTERRUPT_REG:
+ data = (0x40 << PCI_MAX_LAT_SHIFT) |
+ (PCI_INTERRUPT_PIN_A << PCI_INTERRUPT_PIN_SHIFT);
+ break;
+ case 0x40: /* USB capability pointer */
+ data = 0;
+ break;
+ default:
+ data = 0;
+ break;
+ }
+
+ return data;
+}
+
+void
+glx_fn4_write(int reg, pcireg_t data)
+{
+ uint64_t msr;
+
+ switch (reg) {
+ case PCI_COMMAND_STATUS_REG:
+ msr = rdmsr(USB_MSR_OHCB);
+ if (data & PCI_COMMAND_MASTER_ENABLE)
+ msr |= 1UL << 34;
+ else
+ msr &= ~(1UL << 34);
+ if (data & PCI_COMMAND_MEM_ENABLE)
+ msr |= 1UL << 33;
+ else
+ msr &= ~(1UL << 33);
+ wrmsr(USB_MSR_OHCB, msr);
+ break;
+ case PCI_BHLC_REG:
+ msr = rdmsr(GLPCI_CTRL);
+ msr &= 0xff00000000UL;
+ msr |= ((uint64_t)PCI_LATTIMER(data)) << 32;
+ break;
+ case PCI_MAPREG_START + 0x00:
+ if (data == 0xffffffff) {
+ ohci_bar_value = data;
+ } else {
+ if ((data & PCI_MAPREG_TYPE_MASK) ==
+ PCI_MAPREG_TYPE_MEM) {
+ data &= PCI_MAPREG_MEM_ADDR_MASK;
+ msr = rdmsr(GLIU_P2D_BM3);
+ msr &= 0x0fffff0000000000UL;
+ msr |= 2UL << 61; /* USB */
+ msr |= (((uint64_t)data) >> 12) << 20;
+ msr |= 0x000fffff;
+ wrmsr(GLIU_P2D_BM3, msr);
+
+ msr = rdmsr(USB_MSR_OHCB);
+ msr &= ~0xffffff00UL;
+ msr |= data;
+ } else {
+ msr = rdmsr(USB_MSR_OHCB);
+ msr &= ~0xffffff00UL;
+ }
+ wrmsr(USB_MSR_OHCB, msr);
+ ohci_bar_value = 0;
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+/*
+ * Function 5: EHCI Controller
+ */
+
+static pcireg_t ehci_bar_size = 0x1000;
+static pcireg_t ehci_bar_value;
+
+pcireg_t
+glx_fn5_read(int reg)
+{
+ uint64_t msr;
+ pcireg_t data;
+
+ switch (reg) {
+ case PCI_ID_REG:
+ case PCI_SUBSYS_ID_REG:
+ data = PCI_ID_CODE(PCI_VENDOR_AMD, PCI_PRODUCT_AMD_CS5536_EHCI);
+ break;
+ case PCI_COMMAND_STATUS_REG:
+ data = glx_get_status();
+ msr = rdmsr(USB_MSR_EHCB);
+ if (msr & (1UL << 34))
+ data |= PCI_COMMAND_MASTER_ENABLE;
+ if (msr & (1UL << 33))
+ data |= PCI_COMMAND_MEM_ENABLE;
+ break;
+ case PCI_CLASS_REG:
+ msr = rdmsr(USB_GLD_MSR_CAP);
+ data = (PCI_CLASS_SERIALBUS << PCI_CLASS_SHIFT) |
+ (PCI_SUBCLASS_SERIALBUS_USB << PCI_SUBCLASS_SHIFT) |
+ (PCI_INTERFACE_EHCI << PCI_INTERFACE_SHIFT) |
+ (msr & PCI_REVISION_MASK);
+ break;
+ case PCI_BHLC_REG:
+ msr = rdmsr(GLPCI_CTRL);
+ data = (0x00 << PCI_HDRTYPE_SHIFT) |
+ (((msr & 0xff00000000UL) >> 32) << PCI_LATTIMER_SHIFT) |
+ (0x08 << PCI_CACHELINE_SHIFT);
+ break;
+ case PCI_MAPREG_START + 0x00:
+ data = ehci_bar_value;
+ if (data == 0xffffffff)
+ data = PCI_MAPREG_MEM_ADDR_MASK & ~(ehci_bar_size - 1);
+ else {
+ msr = rdmsr(USB_MSR_EHCB);
+ data = msr & 0xffffff00;
+ }
+ if (data != 0)
+ data |= PCI_MAPREG_TYPE_MEM | PCI_MAPREG_MEM_TYPE_32BIT;
+ break;
+ case PCI_CAPLISTPTR_REG:
+ data = 0x40;
+ break;
+ case PCI_INTERRUPT_REG:
+ data = (0x40 << PCI_MAX_LAT_SHIFT) |
+ (PCI_INTERRUPT_PIN_A << PCI_INTERRUPT_PIN_SHIFT);
+ break;
+ case 0x40: /* USB capability pointer */
+ data = 0;
+ break;
+ case PCI_USBREV:
+ msr = rdmsr(USB_MSR_EHCB);
+ data = PCI_USBREV_2_0;
+ data |= ((msr >> 40) & 0x3f) << 8; /* PCI_EHCI_FLADJ */
+ break;
+ default:
+ data = 0;
+ break;
+ }
+
+ return data;
+}
+
+void
+glx_fn5_write(int reg, pcireg_t data)
+{
+ uint64_t msr;
+
+ switch (reg) {
+ case PCI_COMMAND_STATUS_REG:
+ msr = rdmsr(USB_MSR_EHCB);
+ if (data & PCI_COMMAND_MASTER_ENABLE)
+ msr |= 1UL << 34;
+ else
+ msr &= ~(1UL << 34);
+ if (data & PCI_COMMAND_MEM_ENABLE)
+ msr |= 1UL << 33;
+ else
+ msr &= ~(1UL << 33);
+ wrmsr(USB_MSR_EHCB, msr);
+ break;
+ case PCI_BHLC_REG:
+ msr = rdmsr(GLPCI_CTRL);
+ msr &= 0xff00000000UL;
+ msr |= ((uint64_t)PCI_LATTIMER(data)) << 32;
+ break;
+ case PCI_MAPREG_START + 0x00:
+ if (data == 0xffffffff) {
+ ehci_bar_value = data;
+ } else {
+ if ((data & PCI_MAPREG_TYPE_MASK) ==
+ PCI_MAPREG_TYPE_MEM) {
+ data &= PCI_MAPREG_MEM_ADDR_MASK;
+ msr = rdmsr(GLIU_P2D_BM4);
+ msr &= 0x0fffff0000000000UL;
+ msr |= 2UL << 61; /* USB */
+ msr |= (((uint64_t)data) >> 12) << 20;
+ msr |= 0x000fffff;
+ wrmsr(GLIU_P2D_BM4, msr);
+
+ msr = rdmsr(USB_MSR_EHCB);
+ msr &= ~0xffffff00UL;
+ msr |= data;
+ } else {
+ msr = rdmsr(USB_MSR_EHCB);
+ msr &= ~0xffffff00UL;
+ }
+ wrmsr(USB_MSR_EHCB, msr);
+ ehci_bar_value = 0;
+ }
+ break;
+ default:
+ break;
+ }
+}