Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0-or-later */ |
| 2 | |
| 3 | #include <assert.h> |
| 4 | #include <console/console.h> |
| 5 | #include <post.h> |
| 6 | #include <device/pci.h> |
| 7 | #include <soc/chip_common.h> |
| 8 | #include <soc/soc_util.h> |
| 9 | #include <soc/util.h> |
| 10 | #include <stdlib.h> |
| 11 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 12 | static const STACK_RES *domain_to_stack_res(const struct device *dev) |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 13 | { |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 14 | assert(dev->path.type == DEVICE_PATH_DOMAIN); |
Patrick Rudolph | 8c99ebc | 2024-01-19 17:28:47 +0100 | [diff] [blame] | 15 | const union xeon_domain_path dn = { |
| 16 | .domain_path = dev->path.domain.domain |
| 17 | }; |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 18 | |
| 19 | const IIO_UDS *hob = get_iio_uds(); |
| 20 | assert(hob != NULL); |
| 21 | |
Patrick Rudolph | 8c99ebc | 2024-01-19 17:28:47 +0100 | [diff] [blame] | 22 | return &hob->PlatformData.IIO_resource[dn.socket].StackRes[dn.stack]; |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 23 | } |
| 24 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 25 | void iio_pci_domain_read_resources(struct device *dev) |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 26 | { |
| 27 | struct resource *res; |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 28 | const STACK_RES *sr = domain_to_stack_res(dev); |
| 29 | |
| 30 | if (!sr) |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 31 | return; |
| 32 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 33 | int index = 0; |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 34 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 35 | if (dev->path.domain.domain == 0) { |
| 36 | /* The 0 - 0xfff IO range is not reported by the HOB but still gets decoded */ |
| 37 | res = new_resource(dev, index++); |
| 38 | res->base = 0; |
| 39 | res->size = 0x1000; |
| 40 | res->limit = 0xfff; |
| 41 | res->flags = IORESOURCE_IO | IORESOURCE_SUBTRACTIVE | IORESOURCE_ASSIGNED; |
| 42 | } |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 43 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 44 | if (sr->PciResourceIoBase < sr->PciResourceIoLimit) { |
| 45 | res = new_resource(dev, index++); |
| 46 | res->base = sr->PciResourceIoBase; |
| 47 | res->limit = sr->PciResourceIoLimit; |
| 48 | res->size = res->limit - res->base + 1; |
| 49 | res->flags = IORESOURCE_IO | IORESOURCE_ASSIGNED; |
| 50 | } |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 51 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 52 | if (sr->PciResourceMem32Base < sr->PciResourceMem32Limit) { |
| 53 | res = new_resource(dev, index++); |
| 54 | res->base = sr->PciResourceMem32Base; |
| 55 | res->limit = sr->PciResourceMem32Limit; |
| 56 | res->size = res->limit - res->base + 1; |
| 57 | res->flags = IORESOURCE_MEM | IORESOURCE_ASSIGNED; |
| 58 | } |
| 59 | |
| 60 | if (sr->PciResourceMem64Base < sr->PciResourceMem64Limit) { |
| 61 | res = new_resource(dev, index++); |
| 62 | res->base = sr->PciResourceMem64Base; |
| 63 | res->limit = sr->PciResourceMem64Limit; |
| 64 | res->size = res->limit - res->base + 1; |
| 65 | res->flags = IORESOURCE_MEM | IORESOURCE_ASSIGNED; |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 66 | } |
| 67 | } |
| 68 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 69 | void iio_pci_domain_scan_bus(struct device *dev) |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 70 | { |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 71 | const STACK_RES *sr = domain_to_stack_res(dev); |
| 72 | if (!sr) |
| 73 | return; |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 74 | |
Arthur Heymans | 3e99ba0 | 2024-01-25 22:26:07 +0100 | [diff] [blame^] | 75 | struct bus *bus = alloc_bus(dev); |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 76 | bus->secondary = sr->BusBase; |
| 77 | bus->subordinate = sr->BusBase; |
| 78 | bus->max_subordinate = sr->BusLimit; |
| 79 | |
| 80 | printk(BIOS_SPEW, "Scanning IIO stack %d: busses %x-%x\n", dev->path.domain.domain, |
| 81 | dev->link_list->secondary, dev->link_list->max_subordinate); |
| 82 | pci_host_bridge_scan_bus(dev); |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 83 | } |
| 84 | |
Jonathan Zhang | a63ea89 | 2023-01-25 11:16:58 -0800 | [diff] [blame] | 85 | /* |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 86 | * Used by IIO stacks for PCIe bridges. Those contain 1 PCI host bridges, |
| 87 | * all the bus numbers on the IIO stack can be used for this bridge |
Jonathan Zhang | a63ea89 | 2023-01-25 11:16:58 -0800 | [diff] [blame] | 88 | */ |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 89 | static struct device_operations iio_pcie_domain_ops = { |
| 90 | .read_resources = iio_pci_domain_read_resources, |
| 91 | .set_resources = pci_domain_set_resources, |
| 92 | .scan_bus = iio_pci_domain_scan_bus, |
| 93 | }; |
Arthur Heymans | 165893b | 2020-11-06 12:15:41 +0100 | [diff] [blame] | 94 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 95 | /* Attach IIO stack as domains */ |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 96 | void attach_iio_stacks(struct device *dev) |
| 97 | { |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 98 | const IIO_UDS *hob = get_iio_uds(); |
Patrick Rudolph | 8c99ebc | 2024-01-19 17:28:47 +0100 | [diff] [blame] | 99 | union xeon_domain_path dn = { .domain_path = 0 }; |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 100 | if (!hob) |
| 101 | return; |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 102 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 103 | for (int s = 0; s < hob->PlatformData.numofIIO; ++s) { |
| 104 | for (int x = 0; x < MAX_LOGIC_IIO_STACK; ++x) { |
| 105 | if (s == 0 && x == 0) |
| 106 | continue; |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 107 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 108 | const STACK_RES *ri = &hob->PlatformData.IIO_resource[s].StackRes[x]; |
Arthur Heymans | 470f1d3 | 2023-08-31 18:19:09 +0200 | [diff] [blame] | 109 | if (ri->BusBase > ri->BusLimit) |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 110 | continue; |
Jonathan Zhang | 532e8c0 | 2023-01-25 11:28:49 -0800 | [diff] [blame] | 111 | |
Patrick Rudolph | 8c99ebc | 2024-01-19 17:28:47 +0100 | [diff] [blame] | 112 | /* Prepare domain path */ |
| 113 | dn.socket = s; |
| 114 | dn.stack = x; |
| 115 | dn.bus = ri->BusBase; |
| 116 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 117 | if (!is_pcie_iio_stack_res(ri)) { |
| 118 | if (CONFIG(HAVE_IOAT_DOMAINS)) |
Patrick Rudolph | 8c99ebc | 2024-01-19 17:28:47 +0100 | [diff] [blame] | 119 | soc_create_ioat_domains(dn, dev->bus, ri); |
Jonathan Zhang | 532e8c0 | 2023-01-25 11:28:49 -0800 | [diff] [blame] | 120 | continue; |
| 121 | } |
| 122 | |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 123 | struct device_path path; |
| 124 | path.type = DEVICE_PATH_DOMAIN; |
Patrick Rudolph | 8c99ebc | 2024-01-19 17:28:47 +0100 | [diff] [blame] | 125 | path.domain.domain = dn.domain_path; |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 126 | struct device *iio_domain = alloc_dev(dev->bus, &path); |
| 127 | if (iio_domain == NULL) |
Jonathan Zhang | 532e8c0 | 2023-01-25 11:28:49 -0800 | [diff] [blame] | 128 | die("%s: out of memory.\n", __func__); |
Arthur Heymans | 550f55e | 2022-08-24 14:44:26 +0200 | [diff] [blame] | 129 | iio_domain->ops = &iio_pcie_domain_ops; |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 130 | } |
| 131 | } |
Marc Jones | 1f50084 | 2020-10-15 14:32:51 -0600 | [diff] [blame] | 132 | } |