Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0-only */ |
Patrick Rudolph | e357ac3 | 2024-02-23 09:47:24 +0100 | [diff] [blame] | 2 | #include <acpi/acpigen_pci.h> |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 3 | #include <stdbool.h> |
| 4 | |
| 5 | #include <console/console.h> |
| 6 | #include <device/device.h> |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 7 | |
| 8 | #include <defs_iio.h> |
| 9 | #include <hob_iiouds.h> |
Patrick Rudolph | 40e0748 | 2024-02-23 09:23:41 +0100 | [diff] [blame] | 10 | #include <intelblocks/acpi.h> |
Shuo Liu | 2eb9d5e | 2024-06-19 06:48:45 +0800 | [diff] [blame^] | 11 | #include <intelblocks/vtd.h> |
Patrick Rudolph | 40e0748 | 2024-02-23 09:23:41 +0100 | [diff] [blame] | 12 | #include <soc/acpi.h> |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 13 | #include <IioPcieConfigUpd.h> |
| 14 | |
| 15 | #include <soc/chip_common.h> |
| 16 | |
| 17 | /* |
| 18 | * Used for IIO stacks for accelerators and other functionality (IOAT). |
| 19 | * Those have only integrated PCI endpoints (no bridges) behind the host bridge. |
| 20 | */ |
| 21 | |
| 22 | static struct device_operations ioat_domain_ops = { |
| 23 | .read_resources = noop_read_resources, |
| 24 | .set_resources = pci_domain_set_resources, |
| 25 | .scan_bus = pci_host_bridge_scan_bus, |
Patrick Rudolph | 40e0748 | 2024-02-23 09:23:41 +0100 | [diff] [blame] | 26 | #if CONFIG(HAVE_ACPI_TABLES) |
| 27 | .acpi_name = soc_acpi_name, |
Shuo Liu | 255f927 | 2023-03-29 20:14:11 +0800 | [diff] [blame] | 28 | .write_acpi_tables = northbridge_write_acpi_tables, |
Patrick Rudolph | e357ac3 | 2024-02-23 09:47:24 +0100 | [diff] [blame] | 29 | .acpi_fill_ssdt = pci_domain_fill_ssdt, |
Patrick Rudolph | 40e0748 | 2024-02-23 09:23:41 +0100 | [diff] [blame] | 30 | #endif |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 31 | }; |
| 32 | |
Shuo Liu | 2eb9d5e | 2024-06-19 06:48:45 +0800 | [diff] [blame^] | 33 | static struct device *const create_ioat_domain(const union xeon_domain_path dp, struct bus *const upstream, |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 34 | const unsigned int bus_base, const unsigned int bus_limit, |
| 35 | const resource_t mem32_base, const resource_t mem32_limit, |
Patrick Rudolph | 40e0748 | 2024-02-23 09:23:41 +0100 | [diff] [blame] | 36 | const resource_t mem64_base, const resource_t mem64_limit, |
Patrick Rudolph | 8061957 | 2024-03-12 18:32:35 +0100 | [diff] [blame] | 37 | const char *prefix, const size_t pci_segment_group) |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 38 | { |
Patrick Rudolph | 8c99ebc | 2024-01-19 17:28:47 +0100 | [diff] [blame] | 39 | union xeon_domain_path new_path = { |
| 40 | .domain_path = dp.domain_path |
| 41 | }; |
| 42 | new_path.bus = bus_base; |
| 43 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 44 | struct device_path path = { |
| 45 | .type = DEVICE_PATH_DOMAIN, |
| 46 | .domain = { |
Patrick Rudolph | 8c99ebc | 2024-01-19 17:28:47 +0100 | [diff] [blame] | 47 | .domain = new_path.domain_path, |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 48 | }, |
| 49 | }; |
Shuo Liu | 255f927 | 2023-03-29 20:14:11 +0800 | [diff] [blame] | 50 | struct device *const domain = alloc_find_dev(upstream, &path); |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 51 | if (!domain) |
| 52 | die("%s: out of memory.\n", __func__); |
| 53 | |
| 54 | domain->ops = &ioat_domain_ops; |
Patrick Rudolph | 40e0748 | 2024-02-23 09:23:41 +0100 | [diff] [blame] | 55 | iio_domain_set_acpi_name(domain, prefix); |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 56 | |
Arthur Heymans | 3e99ba0 | 2024-01-25 22:26:07 +0100 | [diff] [blame] | 57 | struct bus *const bus = alloc_bus(domain); |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 58 | bus->secondary = bus_base; |
| 59 | bus->subordinate = bus->secondary; |
| 60 | bus->max_subordinate = bus_limit; |
Patrick Rudolph | 8061957 | 2024-03-12 18:32:35 +0100 | [diff] [blame] | 61 | bus->segment_group = pci_segment_group; |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 62 | |
| 63 | unsigned int index = 0; |
| 64 | |
Shuo Liu | 6c708d8 | 2024-04-29 18:16:30 +0800 | [diff] [blame] | 65 | if (mem32_base <= mem32_limit) |
| 66 | domain_mem_window_from_to(domain, index++, mem32_base, mem32_limit + 1); |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 67 | |
Shuo Liu | 6c708d8 | 2024-04-29 18:16:30 +0800 | [diff] [blame] | 68 | if (mem64_base <= mem64_limit) |
| 69 | domain_mem_window_from_to(domain, index++, mem64_base, mem64_limit + 1); |
Shuo Liu | 2eb9d5e | 2024-06-19 06:48:45 +0800 | [diff] [blame^] | 70 | |
| 71 | return domain; |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 72 | } |
| 73 | |
Shuo Liu | ec58beb | 2024-03-11 07:14:07 +0800 | [diff] [blame] | 74 | void create_ioat_domains(const union xeon_domain_path path, |
| 75 | struct bus *const bus, |
| 76 | const STACK_RES *const sr, |
| 77 | const size_t pci_segment_group) |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 78 | { |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 79 | if (sr->BusLimit < sr->BusBase + HQM_BUS_OFFSET + HQM_RESERVED_BUS) { |
| 80 | printk(BIOS_WARNING, |
| 81 | "Ignoring IOAT domain with limited bus range.\n"); |
| 82 | return; |
| 83 | } |
| 84 | |
| 85 | if (sr->PciResourceMem64Limit - sr->PciResourceMem64Base + 1 |
| 86 | < 2 * CPM_MMIO_SIZE + 2 * HQM_MMIO_SIZE) { |
| 87 | printk(BIOS_WARNING, |
| 88 | "Ignoring IOAT domain with limited 64-bit MMIO window.\n"); |
| 89 | return; |
| 90 | } |
| 91 | |
| 92 | /* The FSP HOB doesn't provide accurate information about the |
| 93 | resource allocation. Hence use pre-defined offsets. Based |
Shuo Liu | 08f1f05 | 2024-01-20 02:52:17 +0800 | [diff] [blame] | 94 | on ACPI code in create_dsdt_ioat_resource(), soc_acpi.c: */ |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 95 | resource_t mem64_base, mem64_limit, bus_base, bus_limit; |
| 96 | |
| 97 | /* CPM0 */ |
| 98 | mem64_base = sr->PciResourceMem64Base; |
| 99 | mem64_limit = mem64_base + CPM_MMIO_SIZE - 1; |
| 100 | bus_base = sr->BusBase + CPM_BUS_OFFSET; |
| 101 | bus_limit = bus_base + CPM_RESERVED_BUS; |
Patrick Rudolph | 7d83441 | 2024-02-29 18:08:09 +0100 | [diff] [blame] | 102 | create_ioat_domain(path, bus, bus_base, bus_limit, -1, 0, mem64_base, mem64_limit, |
Patrick Rudolph | 8061957 | 2024-03-12 18:32:35 +0100 | [diff] [blame] | 103 | DOMAIN_TYPE_CPM0, pci_segment_group); |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 104 | |
| 105 | /* HQM0 */ |
| 106 | mem64_base = mem64_limit + 1; |
| 107 | mem64_limit = mem64_base + HQM_MMIO_SIZE - 1; |
| 108 | bus_base = sr->BusBase + HQM_BUS_OFFSET; |
| 109 | bus_limit = bus_base + HQM_RESERVED_BUS; |
Patrick Rudolph | 7d83441 | 2024-02-29 18:08:09 +0100 | [diff] [blame] | 110 | create_ioat_domain(path, bus, bus_base, bus_limit, -1, 0, mem64_base, mem64_limit, |
Patrick Rudolph | 8061957 | 2024-03-12 18:32:35 +0100 | [diff] [blame] | 111 | DOMAIN_TYPE_HQM0, pci_segment_group); |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 112 | |
| 113 | /* CPM1 (optional) */ |
| 114 | mem64_base = mem64_limit + 1; |
| 115 | mem64_limit = mem64_base + CPM_MMIO_SIZE - 1; |
| 116 | bus_base = sr->BusBase + CPM1_BUS_OFFSET; |
| 117 | bus_limit = bus_base + CPM_RESERVED_BUS; |
| 118 | if (bus_limit <= sr->BusLimit) |
Patrick Rudolph | 7d83441 | 2024-02-29 18:08:09 +0100 | [diff] [blame] | 119 | create_ioat_domain(path, bus, bus_base, bus_limit, -1, 0, mem64_base, mem64_limit, |
Patrick Rudolph | 8061957 | 2024-03-12 18:32:35 +0100 | [diff] [blame] | 120 | DOMAIN_TYPE_CPM1, pci_segment_group); |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 121 | |
| 122 | /* HQM1 (optional) */ |
| 123 | mem64_base = mem64_limit + 1; |
| 124 | mem64_limit = mem64_base + HQM_MMIO_SIZE - 1; |
| 125 | bus_base = sr->BusBase + HQM1_BUS_OFFSET; |
| 126 | bus_limit = bus_base + HQM_RESERVED_BUS; |
| 127 | if (bus_limit <= sr->BusLimit) |
Patrick Rudolph | 7d83441 | 2024-02-29 18:08:09 +0100 | [diff] [blame] | 128 | create_ioat_domain(path, bus, bus_base, bus_limit, -1, 0, mem64_base, mem64_limit, |
Patrick Rudolph | 8061957 | 2024-03-12 18:32:35 +0100 | [diff] [blame] | 129 | DOMAIN_TYPE_HQM1, pci_segment_group); |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 130 | |
| 131 | /* DINO */ |
| 132 | mem64_base = mem64_limit + 1; |
| 133 | mem64_limit = sr->PciResourceMem64Limit; |
| 134 | bus_base = sr->BusBase; |
| 135 | bus_limit = bus_base; |
Shuo Liu | 2eb9d5e | 2024-06-19 06:48:45 +0800 | [diff] [blame^] | 136 | struct device *const dev = create_ioat_domain(path, bus, bus_base, bus_limit, |
| 137 | sr->PciResourceMem32Base, sr->PciResourceMem32Limit, |
| 138 | mem64_base, mem64_limit, DOMAIN_TYPE_DINO, pci_segment_group); |
| 139 | |
| 140 | /* Declare domain reserved MMIO */ |
| 141 | uint64_t reserved_mmio = sr->VtdBarAddress + vtd_probe_bar_size(pcidev_on_root(0, 0)); |
| 142 | if ((reserved_mmio >= sr->PciResourceMem32Base) && |
| 143 | (reserved_mmio <= sr->PciResourceMem32Limit)) { |
| 144 | int index = 0; |
| 145 | for (struct resource *res = dev->resource_list; res; res = res->next) |
| 146 | index++; |
| 147 | mmio_range(dev, index, reserved_mmio, |
| 148 | sr->PciResourceMem32Limit - reserved_mmio + 1); |
| 149 | } |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 150 | } |