2020-07-09 00:47:19 +02:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
|
|
|
|
|
|
#include <acpi/acpigen.h>
|
2020-12-15 21:31:56 +01:00
|
|
|
#include <arch/ioapic.h>
|
2020-12-15 00:55:09 +01:00
|
|
|
#include <assert.h>
|
2020-12-16 18:08:41 +01:00
|
|
|
#include <amdblocks/amd_pci_util.h>
|
2020-07-09 00:47:19 +02:00
|
|
|
#include <device/device.h>
|
|
|
|
#include <device/pci.h>
|
2020-07-29 09:44:25 +02:00
|
|
|
#include <device/pciexp.h>
|
2020-07-09 00:47:19 +02:00
|
|
|
#include <device/pci_ids.h>
|
2020-12-16 18:08:41 +01:00
|
|
|
#include <soc/pci.h>
|
2020-07-09 00:47:19 +02:00
|
|
|
#include <soc/pci_devs.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2020-12-15 00:55:09 +01:00
|
|
|
/**
|
|
|
|
* Each PCI bridge has its INTx lines routed to one of the 8 GNB IOAPIC PCI
|
|
|
|
* groups. Each group has 4 interrupts. The INTx lines can be swizzled before
|
|
|
|
* being routed to the IOAPIC. If the IOAPIC redirection entry is masked, the
|
|
|
|
* interrupt is reduced modulo 8 onto INT[A-H] and forwarded to the FCH IOAPIC.
|
|
|
|
**/
|
|
|
|
struct pci_routing {
|
|
|
|
unsigned int devfn;
|
|
|
|
unsigned int group;
|
2021-01-22 17:53:18 +01:00
|
|
|
uint8_t pin[4];
|
2020-12-15 00:55:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/* See AMD PPR 55570 - IOAPIC Initialization for the table that AGESA sets up */
|
|
|
|
static const struct pci_routing pci_routing_table[] = {
|
2021-01-22 17:53:18 +01:00
|
|
|
{PCIE_GPP_0_DEVFN, 0, {PIRQ_A, PIRQ_B, PIRQ_C, PIRQ_D} },
|
|
|
|
{PCIE_GPP_1_DEVFN, 1, {PIRQ_A, PIRQ_B, PIRQ_C, PIRQ_D} },
|
|
|
|
{PCIE_GPP_2_DEVFN, 2, {PIRQ_A, PIRQ_B, PIRQ_C, PIRQ_D} },
|
|
|
|
{PCIE_GPP_3_DEVFN, 3, {PIRQ_A, PIRQ_B, PIRQ_C, PIRQ_D} },
|
|
|
|
{PCIE_GPP_4_DEVFN, 4, {PIRQ_A, PIRQ_B, PIRQ_C, PIRQ_D} },
|
|
|
|
{PCIE_GPP_5_DEVFN, 5, {PIRQ_A, PIRQ_B, PIRQ_C, PIRQ_D} },
|
|
|
|
{PCIE_GPP_6_DEVFN, 6, {PIRQ_A, PIRQ_B, PIRQ_C, PIRQ_D} },
|
|
|
|
{PCIE_GPP_A_DEVFN, 7, {PIRQ_A, PIRQ_B, PIRQ_C, PIRQ_D} },
|
|
|
|
{PCIE_GPP_B_DEVFN, 7, {PIRQ_C, PIRQ_D, PIRQ_A, PIRQ_B} },
|
2020-12-15 00:55:09 +01:00
|
|
|
};
|
|
|
|
|
2020-12-16 18:08:41 +01:00
|
|
|
/*
|
|
|
|
* This data structure is populated from the raw data above. It is used
|
|
|
|
* by amd/common/block/pci/amd_pci_util to write the PCI_INT_LINE register
|
|
|
|
* to each PCI device.
|
|
|
|
*/
|
|
|
|
static struct pirq_struct pirq_data[] = {
|
|
|
|
{ PCIE_GPP_0_DEVFN },
|
|
|
|
{ PCIE_GPP_1_DEVFN },
|
|
|
|
{ PCIE_GPP_2_DEVFN },
|
|
|
|
{ PCIE_GPP_3_DEVFN },
|
|
|
|
{ PCIE_GPP_4_DEVFN },
|
|
|
|
{ PCIE_GPP_5_DEVFN },
|
|
|
|
{ PCIE_GPP_6_DEVFN },
|
|
|
|
{ PCIE_GPP_A_DEVFN },
|
|
|
|
{ PCIE_GPP_B_DEVFN },
|
|
|
|
};
|
|
|
|
|
|
|
|
_Static_assert(ARRAY_SIZE(pci_routing_table) == ARRAY_SIZE(pirq_data),
|
|
|
|
"PCI and PIRQ tables must be the same size");
|
|
|
|
|
2020-12-15 00:55:09 +01:00
|
|
|
static const struct pci_routing *get_pci_routing(unsigned int devfn)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < ARRAY_SIZE(pci_routing_table); ++i) {
|
|
|
|
if (devfn == pci_routing_table[i].devfn)
|
|
|
|
return &pci_routing_table[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned int calculate_irq(const struct pci_routing *pci_routing, unsigned int i)
|
|
|
|
{
|
|
|
|
unsigned int irq_index;
|
|
|
|
irq_index = pci_routing->group * 4;
|
2021-01-22 17:53:18 +01:00
|
|
|
irq_index += pci_routing->pin[i];
|
2020-12-15 00:55:09 +01:00
|
|
|
|
|
|
|
return irq_index;
|
|
|
|
}
|
|
|
|
|
2020-12-16 18:08:41 +01:00
|
|
|
void populate_pirq_data(void)
|
|
|
|
{
|
|
|
|
const struct pci_routing *pci_routing;
|
|
|
|
struct pirq_struct *pirq;
|
|
|
|
unsigned int irq_index;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < ARRAY_SIZE(pirq_data); ++i) {
|
|
|
|
pirq = &pirq_data[i];
|
|
|
|
pci_routing = get_pci_routing(pirq->devfn);
|
|
|
|
if (!pci_routing)
|
|
|
|
die("%s: devfn %u not found\n", __func__, pirq->devfn);
|
|
|
|
|
|
|
|
for (size_t j = 0; j < 4; ++j) {
|
|
|
|
irq_index = calculate_irq(pci_routing, j);
|
|
|
|
|
|
|
|
pirq->PIN[j] = irq_index % 8;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pirq_data_ptr = pirq_data;
|
|
|
|
pirq_data_size = ARRAY_SIZE(pirq_data);
|
|
|
|
}
|
|
|
|
|
2020-07-09 00:47:19 +02:00
|
|
|
static const char *pcie_gpp_acpi_name(const struct device *dev)
|
|
|
|
{
|
|
|
|
if (dev->path.type != DEVICE_PATH_PCI)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
switch (dev->path.pci.devfn) {
|
|
|
|
case PCIE_GPP_0_DEVFN:
|
|
|
|
return "PBR0";
|
|
|
|
case PCIE_GPP_1_DEVFN:
|
|
|
|
return "PBR1";
|
|
|
|
case PCIE_GPP_2_DEVFN:
|
|
|
|
return "PBR2";
|
|
|
|
case PCIE_GPP_3_DEVFN:
|
|
|
|
return "PBR3";
|
|
|
|
case PCIE_GPP_4_DEVFN:
|
|
|
|
return "PBR4";
|
|
|
|
case PCIE_GPP_5_DEVFN:
|
|
|
|
return "PBR5";
|
|
|
|
case PCIE_GPP_6_DEVFN:
|
|
|
|
return "PBR6";
|
|
|
|
case PCIE_GPP_A_DEVFN:
|
|
|
|
return "PBRA";
|
|
|
|
case PCIE_GPP_B_DEVFN:
|
|
|
|
return "PBRB";
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-12-15 00:55:09 +01:00
|
|
|
static void acpigen_write_PRT(const struct device *dev)
|
|
|
|
{
|
|
|
|
char link_template[] = "\\_SB.INTX";
|
|
|
|
unsigned int irq_index;
|
|
|
|
|
|
|
|
const struct pci_routing *pci_routing = get_pci_routing(dev->path.pci.devfn);
|
|
|
|
if (!pci_routing) {
|
|
|
|
printk(BIOS_ERR, "PCI routing table not found for %s\n", dev_path(dev));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
acpigen_write_method("_PRT", 0);
|
2020-12-15 21:31:56 +01:00
|
|
|
|
|
|
|
/* If (PMOD) */
|
|
|
|
acpigen_write_if();
|
|
|
|
acpigen_emit_namestring("PMOD");
|
|
|
|
|
|
|
|
/* Return (Package{...}) */
|
2020-12-15 00:55:09 +01:00
|
|
|
acpigen_emit_byte(RETURN_OP);
|
|
|
|
|
2020-12-15 21:31:56 +01:00
|
|
|
acpigen_write_package(4); /* Package - APIC Routing */
|
|
|
|
for (unsigned int i = 0; i < 4; ++i) {
|
|
|
|
irq_index = calculate_irq(pci_routing, i);
|
|
|
|
|
|
|
|
acpigen_write_package(4);
|
2021-01-22 17:44:03 +01:00
|
|
|
/* There is only one device attached to the bridge */
|
2020-12-15 21:31:56 +01:00
|
|
|
acpigen_write_dword(0x0000FFFF);
|
|
|
|
acpigen_write_byte(i);
|
|
|
|
acpigen_write_byte(0); /* Source: GSI */
|
|
|
|
/* GNB IO-APIC is located after the FCH IO-APIC */
|
|
|
|
acpigen_write_dword(IO_APIC_INTERRUPTS + irq_index);
|
|
|
|
acpigen_pop_len();
|
|
|
|
}
|
|
|
|
acpigen_pop_len(); /* Package - APIC Routing */
|
|
|
|
acpigen_pop_len(); /* End If */
|
|
|
|
|
|
|
|
/* Else */
|
|
|
|
acpigen_write_else();
|
|
|
|
|
|
|
|
/* Return (Package{...}) */
|
|
|
|
acpigen_emit_byte(RETURN_OP);
|
|
|
|
|
|
|
|
acpigen_write_package(4); /* Package - PIC Routing */
|
2020-12-15 00:55:09 +01:00
|
|
|
for (unsigned int i = 0; i < 4; ++i) {
|
|
|
|
irq_index = calculate_irq(pci_routing, i);
|
|
|
|
|
|
|
|
link_template[8] = 'A' + (irq_index % 8);
|
|
|
|
|
|
|
|
acpigen_write_package(4);
|
2021-01-22 17:44:03 +01:00
|
|
|
/* There is only one device attached to the bridge */
|
2020-12-15 00:55:09 +01:00
|
|
|
acpigen_write_dword(0x0000FFFF);
|
|
|
|
acpigen_write_byte(i);
|
|
|
|
acpigen_emit_namestring(link_template);
|
|
|
|
acpigen_write_dword(0);
|
|
|
|
acpigen_pop_len();
|
|
|
|
}
|
2020-12-15 21:31:56 +01:00
|
|
|
acpigen_pop_len(); /* Package - PIC Routing */
|
|
|
|
|
|
|
|
acpigen_pop_len(); /* End Else */
|
2020-12-15 00:55:09 +01:00
|
|
|
|
|
|
|
acpigen_pop_len(); /* Method */
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This function writes a PCI device with _ADR, _STA, and _PRT objects:
|
|
|
|
* Example:
|
|
|
|
* Scope (\_SB.PCI0)
|
|
|
|
* {
|
|
|
|
* Device (PBRA)
|
|
|
|
* {
|
|
|
|
* Name (_ADR, 0x0000000000080001) // _ADR: Address
|
|
|
|
* Method (_STA, 0, NotSerialized) // _STA: Status
|
|
|
|
* {
|
|
|
|
* Return (0x0F)
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* Method (_PRT, 0, NotSerialized) // _PRT: PCI Routing Table
|
|
|
|
* {
|
2020-12-15 21:31:56 +01:00
|
|
|
* If (PMOD)
|
2020-12-15 00:55:09 +01:00
|
|
|
* {
|
2020-12-15 21:31:56 +01:00
|
|
|
* Return (Package (0x04)
|
2020-12-15 00:55:09 +01:00
|
|
|
* {
|
2020-12-15 21:31:56 +01:00
|
|
|
* Package (0x04)
|
|
|
|
* {
|
|
|
|
* 0x0000FFFF,
|
|
|
|
* 0x00,
|
|
|
|
* 0x00,
|
|
|
|
* 0x00000034
|
|
|
|
* },
|
2020-12-15 00:55:09 +01:00
|
|
|
*
|
2020-12-15 21:31:56 +01:00
|
|
|
* Package (0x04)
|
|
|
|
* {
|
|
|
|
* 0x0000FFFF,
|
|
|
|
* 0x01,
|
|
|
|
* 0x00,
|
|
|
|
* 0x00000035
|
|
|
|
* },
|
2020-12-15 00:55:09 +01:00
|
|
|
*
|
2020-12-15 21:31:56 +01:00
|
|
|
* Package (0x04)
|
|
|
|
* {
|
|
|
|
* 0x0000FFFF,
|
|
|
|
* 0x02,
|
|
|
|
* 0x00,
|
|
|
|
* 0x00000036
|
|
|
|
* },
|
2020-12-15 00:55:09 +01:00
|
|
|
*
|
2020-12-15 21:31:56 +01:00
|
|
|
* Package (0x04)
|
|
|
|
* {
|
|
|
|
* 0x0000FFFF,
|
|
|
|
* 0x03,
|
|
|
|
* 0x00,
|
|
|
|
* 0x00000037
|
|
|
|
* }
|
|
|
|
* })
|
|
|
|
* }
|
|
|
|
* Else
|
|
|
|
* {
|
|
|
|
* Return (Package (0x04)
|
2020-12-15 00:55:09 +01:00
|
|
|
* {
|
2020-12-15 21:31:56 +01:00
|
|
|
* Package (0x04)
|
|
|
|
* {
|
|
|
|
* 0x0000FFFF,
|
|
|
|
* 0x00,
|
|
|
|
* \_SB.INTE,
|
|
|
|
* 0x00000000
|
|
|
|
* },
|
|
|
|
*
|
|
|
|
* Package (0x04)
|
|
|
|
* {
|
|
|
|
* 0x0000FFFF,
|
|
|
|
* 0x01,
|
|
|
|
* \_SB.INTF,
|
|
|
|
* 0x00000000
|
|
|
|
* },
|
|
|
|
*
|
|
|
|
* Package (0x04)
|
|
|
|
* {
|
|
|
|
* 0x0000FFFF,
|
|
|
|
* 0x02,
|
|
|
|
* \_SB.INTG,
|
|
|
|
* 0x00000000
|
|
|
|
* },
|
|
|
|
*
|
|
|
|
* Package (0x04)
|
|
|
|
* {
|
|
|
|
* 0x0000FFFF,
|
|
|
|
* 0x03,
|
|
|
|
* \_SB.INTH,
|
|
|
|
* 0x00000000
|
|
|
|
* }
|
|
|
|
* })
|
|
|
|
* }
|
2020-12-15 00:55:09 +01:00
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
*/
|
|
|
|
static void acpi_device_write_gpp_pci_dev(const struct device *dev)
|
|
|
|
{
|
|
|
|
const char *scope = acpi_device_scope(dev);
|
|
|
|
const char *name = acpi_device_name(dev);
|
|
|
|
|
|
|
|
assert(dev->path.type == DEVICE_PATH_PCI);
|
|
|
|
assert(name);
|
|
|
|
assert(scope);
|
|
|
|
|
|
|
|
acpigen_write_scope(scope);
|
|
|
|
acpigen_write_device(name);
|
|
|
|
|
|
|
|
acpigen_write_ADR_pci_device(dev);
|
|
|
|
acpigen_write_STA(acpi_device_status(dev));
|
|
|
|
|
|
|
|
acpigen_write_PRT(dev);
|
|
|
|
|
|
|
|
acpigen_pop_len(); /* Device */
|
|
|
|
acpigen_pop_len(); /* Scope */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-07-29 09:44:25 +02:00
|
|
|
static struct device_operations internal_pcie_gpp_ops = {
|
2020-07-09 00:47:19 +02:00
|
|
|
.read_resources = pci_bus_read_resources,
|
|
|
|
.set_resources = pci_dev_set_resources,
|
|
|
|
.enable_resources = pci_bus_enable_resources,
|
|
|
|
.scan_bus = pci_scan_bridge,
|
|
|
|
.reset_bus = pci_bus_reset,
|
|
|
|
.acpi_name = pcie_gpp_acpi_name,
|
2020-12-15 00:55:09 +01:00
|
|
|
.acpi_fill_ssdt = acpi_device_write_gpp_pci_dev,
|
2020-07-09 00:47:19 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static const unsigned short pci_device_ids[] = {
|
2020-11-17 16:39:36 +01:00
|
|
|
PCI_DEVICE_ID_AMD_FAM17H_MODEL18H_PCIE_GPP_BUSA,
|
|
|
|
PCI_DEVICE_ID_AMD_FAM17H_MODEL18H_PCIE_GPP_BUSB,
|
2020-07-09 00:47:19 +02:00
|
|
|
0
|
|
|
|
};
|
|
|
|
|
2020-07-29 09:44:25 +02:00
|
|
|
static const struct pci_driver internal_pcie_gpp_driver __pci_driver = {
|
|
|
|
.ops = &internal_pcie_gpp_ops,
|
2020-07-09 00:47:19 +02:00
|
|
|
.vendor = PCI_VENDOR_ID_AMD,
|
|
|
|
.devices = pci_device_ids,
|
|
|
|
};
|
2020-07-29 09:44:25 +02:00
|
|
|
|
|
|
|
static struct device_operations external_pcie_gpp_ops = {
|
|
|
|
.read_resources = pci_bus_read_resources,
|
|
|
|
.set_resources = pci_dev_set_resources,
|
|
|
|
.enable_resources = pci_bus_enable_resources,
|
|
|
|
.scan_bus = pciexp_scan_bridge,
|
|
|
|
.reset_bus = pci_bus_reset,
|
|
|
|
.acpi_name = pcie_gpp_acpi_name,
|
2020-12-15 00:55:09 +01:00
|
|
|
.acpi_fill_ssdt = acpi_device_write_gpp_pci_dev,
|
2020-07-29 09:44:25 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static const struct pci_driver external_pcie_gpp_driver __pci_driver = {
|
|
|
|
.ops = &external_pcie_gpp_ops,
|
|
|
|
.vendor = PCI_VENDOR_ID_AMD,
|
2020-11-17 16:39:36 +01:00
|
|
|
.device = PCI_DEVICE_ID_AMD_FAM17H_MODEL18H_PCIE_GPP,
|
2020-07-29 09:44:25 +02:00
|
|
|
};
|