Angel Pons | c74dae9 | 2020-04-02 23:48:16 +0200 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0-only */ |
Uwe Hermann | b80dbf0 | 2007-04-22 19:08:13 +0000 | [diff] [blame] | 2 | |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 3 | #include <console/console.h> |
Marshall Dawson | 6f978cf | 2017-04-19 18:31:07 -0600 | [diff] [blame] | 4 | #include <commonlib/endian.h> |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 5 | #include <device/device.h> |
| 6 | #include <device/pci.h> |
| 7 | #include <device/pci_ids.h> |
| 8 | #include <device/pci_ops.h> |
Raul E Rangel | a736f48 | 2021-07-16 14:03:21 -0600 | [diff] [blame] | 9 | #include <stdio.h> |
Stefan Reinauer | d98cf5b | 2008-08-01 11:25:41 +0000 | [diff] [blame] | 10 | #include <string.h> |
Peter Stuge | 483b7bb | 2009-04-14 07:40:01 +0000 | [diff] [blame] | 11 | #include <cbfs.h> |
Patrick Rudolph | 00c0cd2 | 2017-06-06 19:30:55 +0200 | [diff] [blame] | 12 | #include <cbmem.h> |
Furquan Shaikh | 76cedd2 | 2020-05-02 10:24:23 -0700 | [diff] [blame] | 13 | #include <acpi/acpigen.h> |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 14 | |
Kyösti Mälkki | 1bdd321 | 2014-12-26 13:29:09 +0200 | [diff] [blame] | 15 | /* Rmodules don't like weak symbols. */ |
Martin Roth | dafcc7a | 2020-02-05 16:46:30 -0700 | [diff] [blame] | 16 | void __weak map_oprom_vendev_rev(u32 *vendev, u8 *rev) { return; } |
Aaron Durbin | 6403167 | 2018-04-21 14:45:32 -0600 | [diff] [blame] | 17 | u32 __weak map_oprom_vendev(u32 vendev) { return vendev; } |
Kyösti Mälkki | 1bdd321 | 2014-12-26 13:29:09 +0200 | [diff] [blame] | 18 | |
Raul E Rangel | a3811fe | 2021-11-01 13:40:14 -0600 | [diff] [blame] | 19 | void vga_oprom_preload(void) |
| 20 | { |
| 21 | /* The CONFIG_VGA_BIOS_ID symbol is only defined when VGA_BIOS is selected */ |
| 22 | #if CONFIG(VGA_BIOS) |
| 23 | const char name[] = "pci" CONFIG_VGA_BIOS_ID ".rom"; |
| 24 | |
| 25 | if (!CONFIG(CBFS_PRELOAD)) |
| 26 | return; |
| 27 | |
| 28 | printk(BIOS_DEBUG, "Preloading VGA ROM %s\n", name); |
| 29 | |
| 30 | cbfs_preload(name); |
| 31 | #endif |
| 32 | } |
| 33 | |
Raul E Rangel | a736f48 | 2021-07-16 14:03:21 -0600 | [diff] [blame] | 34 | static void *cbfs_boot_map_optionrom(uint16_t vendor, uint16_t device) |
| 35 | { |
| 36 | char name[17] = "pciXXXX,XXXX.rom"; |
| 37 | |
| 38 | snprintf(name, sizeof(name), "pci%04hx,%04hx.rom", vendor, device); |
| 39 | |
| 40 | return cbfs_map(name, NULL); |
| 41 | } |
| 42 | |
| 43 | static void *cbfs_boot_map_optionrom_revision(uint16_t vendor, uint16_t device, uint8_t rev) |
| 44 | { |
| 45 | char name[20] = "pciXXXX,XXXX,XX.rom"; |
| 46 | |
| 47 | snprintf(name, sizeof(name), "pci%04hx,%04hx,%02hhx.rom", vendor, device, rev); |
| 48 | |
| 49 | return cbfs_map(name, NULL); |
| 50 | } |
| 51 | |
Furquan Shaikh | 0f007d8 | 2020-04-24 06:41:18 -0700 | [diff] [blame] | 52 | struct rom_header *pci_rom_probe(const struct device *dev) |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 53 | { |
Martin Roth | a616a4b | 2020-01-21 09:28:40 -0700 | [diff] [blame] | 54 | struct rom_header *rom_header = NULL; |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 55 | struct pci_data *rom_data; |
Martin Roth | dafcc7a | 2020-02-05 16:46:30 -0700 | [diff] [blame] | 56 | u8 rev = pci_read_config8(dev, PCI_REVISION_ID); |
| 57 | u8 mapped_rev = rev; |
| 58 | u32 vendev = (dev->vendor << 16) | dev->device; |
| 59 | u32 mapped_vendev = vendev; |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 60 | |
Martin Roth | dafcc7a | 2020-02-05 16:46:30 -0700 | [diff] [blame] | 61 | /* If the ROM is in flash, then don't check the PCI device for it. */ |
Martin Roth | a616a4b | 2020-01-21 09:28:40 -0700 | [diff] [blame] | 62 | if (CONFIG(CHECK_REV_IN_OPROM_NAME)) { |
Martin Roth | dafcc7a | 2020-02-05 16:46:30 -0700 | [diff] [blame] | 63 | map_oprom_vendev_rev(&mapped_vendev, &mapped_rev); |
Felix Held | 42f0396 | 2023-03-08 15:38:01 +0100 | [diff] [blame] | 64 | rom_header = cbfs_boot_map_optionrom_revision(mapped_vendev >> 16, |
| 65 | mapped_vendev & 0xffff, |
| 66 | mapped_rev); |
Martin Roth | dafcc7a | 2020-02-05 16:46:30 -0700 | [diff] [blame] | 67 | } else { |
Martin Roth | dafcc7a | 2020-02-05 16:46:30 -0700 | [diff] [blame] | 68 | mapped_vendev = map_oprom_vendev(vendev); |
Felix Held | 42f0396 | 2023-03-08 15:38:01 +0100 | [diff] [blame] | 69 | rom_header = cbfs_boot_map_optionrom(mapped_vendev >> 16, |
| 70 | mapped_vendev & 0xffff); |
Stefan Reinauer | c0a6c6b | 2012-01-23 14:17:52 -0800 | [diff] [blame] | 71 | } |
| 72 | |
Myles Watson | d27c08c | 2009-11-06 23:42:26 +0000 | [diff] [blame] | 73 | if (rom_header) { |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 74 | printk(BIOS_DEBUG, "In CBFS, ROM address for %s = %p\n", |
| 75 | dev_path(dev), rom_header); |
Raul E Rangel | ad5307e | 2021-07-16 15:06:56 -0600 | [diff] [blame] | 76 | } else if (CONFIG(ON_DEVICE_ROM_LOAD)) { |
Stefan Reinauer | 83fa169 | 2015-06-18 22:01:07 -0700 | [diff] [blame] | 77 | uintptr_t rom_address; |
Myles Watson | d27c08c | 2009-11-06 23:42:26 +0000 | [diff] [blame] | 78 | |
arch import user (historical) | 2305364 | 2005-07-06 16:49:59 +0000 | [diff] [blame] | 79 | rom_address = pci_read_config32(dev, PCI_ROM_ADDRESS); |
Myles Watson | d27c08c | 2009-11-06 23:42:26 +0000 | [diff] [blame] | 80 | |
| 81 | if (rom_address == 0x00000000 || rom_address == 0xffffffff) { |
Raul E Rangel | 69cb69f | 2021-07-16 15:04:18 -0600 | [diff] [blame] | 82 | if (CONFIG(CPU_QEMU_X86) && (dev->class >> 8) == PCI_CLASS_DISPLAY_VGA) |
Myles Watson | ae3e998 | 2009-11-17 15:20:22 +0000 | [diff] [blame] | 83 | rom_address = 0xc0000; |
| 84 | else |
Myles Watson | ae3e998 | 2009-11-17 15:20:22 +0000 | [diff] [blame] | 85 | return NULL; |
Myles Watson | d27c08c | 2009-11-06 23:42:26 +0000 | [diff] [blame] | 86 | } else { |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 87 | /* Enable expansion ROM address decoding. */ |
Myles Watson | d27c08c | 2009-11-06 23:42:26 +0000 | [diff] [blame] | 88 | pci_write_config32(dev, PCI_ROM_ADDRESS, |
| 89 | rom_address|PCI_ROM_ADDRESS_ENABLE); |
| 90 | } |
| 91 | |
Kyösti Mälkki | 8c9be43 | 2019-06-29 22:34:07 +0300 | [diff] [blame] | 92 | rom_address &= PCI_ROM_ADDRESS_MASK; |
| 93 | |
Stefan Reinauer | afaa257 | 2011-10-06 16:47:51 -0700 | [diff] [blame] | 94 | printk(BIOS_DEBUG, "Option ROM address for %s = %lx\n", |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 95 | dev_path(dev), (unsigned long)rom_address); |
Myles Watson | d27c08c | 2009-11-06 23:42:26 +0000 | [diff] [blame] | 96 | rom_header = (struct rom_header *)rom_address; |
Raul E Rangel | ad5307e | 2021-07-16 15:06:56 -0600 | [diff] [blame] | 97 | } else { |
| 98 | printk(BIOS_DEBUG, "PCI Option ROM loading disabled for %s\n", |
| 99 | dev_path(dev)); |
| 100 | return NULL; |
arch import user (historical) | 2305364 | 2005-07-06 16:49:59 +0000 | [diff] [blame] | 101 | } |
Yinghai Lu | 9e4faef | 2005-01-14 22:04:49 +0000 | [diff] [blame] | 102 | |
Angel Pons | d19cc11 | 2021-07-04 11:41:31 +0200 | [diff] [blame] | 103 | printk(BIOS_SPEW, |
| 104 | "PCI expansion ROM, signature 0x%04x, INIT size 0x%04x, data ptr 0x%04x\n", |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 105 | le32_to_cpu(rom_header->signature), |
| 106 | rom_header->size * 512, le32_to_cpu(rom_header->data)); |
| 107 | |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 108 | if (le32_to_cpu(rom_header->signature) != PCI_ROM_HDR) { |
Angel Pons | d19cc11 | 2021-07-04 11:41:31 +0200 | [diff] [blame] | 109 | printk(BIOS_ERR, "Incorrect expansion ROM header signature %04x\n", |
| 110 | le32_to_cpu(rom_header->signature)); |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 111 | return NULL; |
| 112 | } |
| 113 | |
Myles Watson | d27c08c | 2009-11-06 23:42:26 +0000 | [diff] [blame] | 114 | rom_data = (((void *)rom_header) + le32_to_cpu(rom_header->data)); |
| 115 | |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 116 | printk(BIOS_SPEW, "PCI ROM image, vendor ID %04x, device ID %04x,\n", |
| 117 | rom_data->vendor, rom_data->device); |
Stefan Reinauer | c0a6c6b | 2012-01-23 14:17:52 -0800 | [diff] [blame] | 118 | /* If the device id is mapped, a mismatch is expected */ |
| 119 | if ((dev->vendor != rom_data->vendor |
| 120 | || dev->device != rom_data->device) |
| 121 | && (vendev == mapped_vendev)) { |
Angel Pons | d19cc11 | 2021-07-04 11:41:31 +0200 | [diff] [blame] | 122 | printk(BIOS_ERR, "ID mismatch: vendor ID %04x, device ID %04x\n", |
| 123 | dev->vendor, dev->device); |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 124 | return NULL; |
| 125 | } |
| 126 | |
Angel Pons | d19cc11 | 2021-07-04 11:41:31 +0200 | [diff] [blame] | 127 | printk(BIOS_SPEW, "PCI ROM image, Class Code %04x%02x, Code Type %02x\n", |
| 128 | rom_data->class_hi, rom_data->class_lo, |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 129 | rom_data->type); |
| 130 | |
Yinghai Lu | 54ab311 | 2005-01-18 03:10:46 +0000 | [diff] [blame] | 131 | if (dev->class != ((rom_data->class_hi << 8) | rom_data->class_lo)) { |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 132 | printk(BIOS_DEBUG, "Class Code mismatch ROM %08x, dev %08x\n", |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 133 | (rom_data->class_hi << 8) | rom_data->class_lo, |
| 134 | dev->class); |
| 135 | // return NULL; |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 136 | } |
| 137 | |
| 138 | return rom_header; |
| 139 | } |
| 140 | |
Stefan Reinauer | 7ce8c54 | 2005-12-02 21:52:30 +0000 | [diff] [blame] | 141 | static void *pci_ram_image_start = (void *)PCI_RAM_IMAGE_START; |
Yinghai Lu | 9e4faef | 2005-01-14 22:04:49 +0000 | [diff] [blame] | 142 | |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 143 | struct rom_header *pci_rom_load(struct device *dev, |
| 144 | struct rom_header *rom_header) |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 145 | { |
| 146 | struct pci_data * rom_data; |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 147 | unsigned int rom_size; |
arch import user (historical) | 2305364 | 2005-07-06 16:49:59 +0000 | [diff] [blame] | 148 | unsigned int image_size=0; |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 149 | |
arch import user (historical) | 2305364 | 2005-07-06 16:49:59 +0000 | [diff] [blame] | 150 | do { |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 151 | /* Get next image. */ |
Elyes Haouas | d369c66 | 2022-11-18 15:06:21 +0100 | [diff] [blame] | 152 | rom_header = (struct rom_header *)((void *)rom_header |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 153 | + image_size); |
| 154 | |
Elyes Haouas | d369c66 | 2022-11-18 15:06:21 +0100 | [diff] [blame] | 155 | rom_data = (struct pci_data *)((void *)rom_header |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 156 | + le32_to_cpu(rom_header->data)); |
| 157 | |
| 158 | image_size = le32_to_cpu(rom_data->ilen) * 512; |
| 159 | } while ((rom_data->type != 0) && (rom_data->indicator != 0)); // make sure we got x86 version |
arch import user (historical) | 2305364 | 2005-07-06 16:49:59 +0000 | [diff] [blame] | 160 | |
Stefan Reinauer | 4d59eaa | 2009-04-22 16:32:18 +0000 | [diff] [blame] | 161 | if (rom_data->type != 0) |
| 162 | return NULL; |
arch import user (historical) | 2305364 | 2005-07-06 16:49:59 +0000 | [diff] [blame] | 163 | |
| 164 | rom_size = rom_header->size * 512; |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 165 | |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 166 | /* |
| 167 | * We check to see if the device thinks it is a VGA device not |
| 168 | * whether the ROM image is for a VGA device because some |
| 169 | * devices have a mismatch between the hardware and the ROM. |
| 170 | */ |
Elyes HAOUAS | 0f8b8d9 | 2019-01-03 10:23:28 +0100 | [diff] [blame] | 171 | if ((dev->class >> 8) == PCI_CLASS_DISPLAY_VGA) { |
Elyes HAOUAS | e348066 | 2018-05-06 20:32:23 +0200 | [diff] [blame] | 172 | extern struct device *vga_pri; /* Primary VGA device (device.c). */ |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 173 | if (dev != vga_pri) return NULL; /* Only one VGA supported. */ |
Ronald G. Minnich | a88db7b | 2009-06-09 14:44:37 +0000 | [diff] [blame] | 174 | if ((void *)PCI_VGA_RAM_IMAGE_START != rom_header) { |
Angel Pons | d19cc11 | 2021-07-04 11:41:31 +0200 | [diff] [blame] | 175 | printk(BIOS_DEBUG, |
| 176 | "Copying VGA ROM Image from %p to 0x%x, 0x%x bytes\n", |
| 177 | rom_header, PCI_VGA_RAM_IMAGE_START, rom_size); |
Uwe Hermann | e487047 | 2010-11-04 23:23:47 +0000 | [diff] [blame] | 178 | memcpy((void *)PCI_VGA_RAM_IMAGE_START, rom_header, |
| 179 | rom_size); |
Ronald G. Minnich | a88db7b | 2009-06-09 14:44:37 +0000 | [diff] [blame] | 180 | } |
Elyes Haouas | d369c66 | 2022-11-18 15:06:21 +0100 | [diff] [blame] | 181 | return (struct rom_header *)(PCI_VGA_RAM_IMAGE_START); |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 182 | } |
Stefan Reinauer | 4d59eaa | 2009-04-22 16:32:18 +0000 | [diff] [blame] | 183 | |
Angel Pons | d19cc11 | 2021-07-04 11:41:31 +0200 | [diff] [blame] | 184 | printk(BIOS_DEBUG, "Copying non-VGA ROM image from %p to %p, 0x%x bytes\n", |
| 185 | rom_header, pci_ram_image_start, rom_size); |
Stefan Reinauer | 4d59eaa | 2009-04-22 16:32:18 +0000 | [diff] [blame] | 186 | |
| 187 | memcpy(pci_ram_image_start, rom_header, rom_size); |
| 188 | pci_ram_image_start += rom_size; |
Elyes Haouas | d369c66 | 2022-11-18 15:06:21 +0100 | [diff] [blame] | 189 | return (struct rom_header *)(pci_ram_image_start-rom_size); |
Li-Ta Lo | 883b879 | 2005-01-10 23:16:22 +0000 | [diff] [blame] | 190 | } |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 191 | |
| 192 | /* ACPI */ |
Julius Werner | cd49cce | 2019-03-05 16:53:33 -0800 | [diff] [blame] | 193 | #if CONFIG(HAVE_ACPI_TABLES) |
Marshall Dawson | 6f978cf | 2017-04-19 18:31:07 -0600 | [diff] [blame] | 194 | |
| 195 | /* VBIOS may be modified after oprom init so use the copy if present. */ |
Furquan Shaikh | 0f007d8 | 2020-04-24 06:41:18 -0700 | [diff] [blame] | 196 | static struct rom_header *check_initialized(const struct device *dev) |
Marshall Dawson | 6f978cf | 2017-04-19 18:31:07 -0600 | [diff] [blame] | 197 | { |
| 198 | struct rom_header *run_rom; |
| 199 | struct pci_data *rom_data; |
| 200 | |
Nikolai Vyssotski | b649d6a | 2021-03-11 19:15:03 -0600 | [diff] [blame] | 201 | if (!CONFIG(VGA_ROM_RUN) && !CONFIG(RUN_FSP_GOP)) |
Marshall Dawson | 6f978cf | 2017-04-19 18:31:07 -0600 | [diff] [blame] | 202 | return NULL; |
| 203 | |
| 204 | run_rom = (struct rom_header *)(uintptr_t)PCI_VGA_RAM_IMAGE_START; |
| 205 | if (read_le16(&run_rom->signature) != PCI_ROM_HDR) |
| 206 | return NULL; |
| 207 | |
| 208 | rom_data = (struct pci_data *)((u8 *)run_rom |
Jacob Garber | 7cfe68d | 2019-07-16 13:14:09 -0600 | [diff] [blame] | 209 | + read_le16(&run_rom->data)); |
Marshall Dawson | 6f978cf | 2017-04-19 18:31:07 -0600 | [diff] [blame] | 210 | |
| 211 | if (read_le32(&rom_data->signature) == PCI_DATA_HDR |
| 212 | && read_le16(&rom_data->device) == dev->device |
| 213 | && read_le16(&rom_data->vendor) == dev->vendor) |
| 214 | return run_rom; |
| 215 | else |
| 216 | return NULL; |
| 217 | } |
| 218 | |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 219 | static unsigned long |
Matt DeVillier | 0cd2a50 | 2023-09-04 08:32:39 -0500 | [diff] [blame] | 220 | ati_rom_acpi_fill_vfct(const struct device *device, acpi_vfct_t *vfct_struct, |
Elyes HAOUAS | e348066 | 2018-05-06 20:32:23 +0200 | [diff] [blame] | 221 | unsigned long current) |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 222 | { |
Himanshu Sahdev | afd0505 | 2019-10-21 11:23:20 +0530 | [diff] [blame] | 223 | acpi_vfct_image_hdr_t *header = &vfct_struct->image_hdr; |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 224 | struct rom_header *rom; |
| 225 | |
Marshall Dawson | 6f978cf | 2017-04-19 18:31:07 -0600 | [diff] [blame] | 226 | rom = check_initialized(device); |
| 227 | if (!rom) |
| 228 | rom = pci_rom_probe(device); |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 229 | if (!rom) { |
Elyes HAOUAS | 0c1d660 | 2021-01-16 17:29:29 +0100 | [diff] [blame] | 230 | printk(BIOS_ERR, "%s failed\n", __func__); |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 231 | return current; |
| 232 | } |
Arthur Heymans | 7fcd4d5 | 2023-08-24 15:12:19 +0200 | [diff] [blame] | 233 | if (device->upstream->segment_group) { |
Felix Held | 3b5b66d | 2024-01-11 22:26:18 +0100 | [diff] [blame] | 234 | printk(BIOS_ERR, "VFCT only supports GPU in first PCI segment group.\n"); |
| 235 | return current; |
| 236 | } |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 237 | |
Marshall Dawson | 6f978cf | 2017-04-19 18:31:07 -0600 | [diff] [blame] | 238 | printk(BIOS_DEBUG, " Copying %sVBIOS image from %p\n", |
| 239 | rom == (struct rom_header *) |
| 240 | (uintptr_t)PCI_VGA_RAM_IMAGE_START ? |
| 241 | "initialized " : "", |
| 242 | rom); |
| 243 | |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 244 | header->DeviceID = device->device; |
| 245 | header->VendorID = device->vendor; |
Arthur Heymans | 7fcd4d5 | 2023-08-24 15:12:19 +0200 | [diff] [blame] | 246 | header->PCIBus = device->upstream->secondary; |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 247 | header->PCIFunction = PCI_FUNC(device->path.pci.devfn); |
| 248 | header->PCIDevice = PCI_SLOT(device->path.pci.devfn); |
| 249 | header->ImageLength = rom->size * 512; |
Matt DeVillier | 7c04d0e | 2023-09-03 12:51:58 -0500 | [diff] [blame] | 250 | memcpy((void *)header->VbiosContent, rom, header->ImageLength); |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 251 | |
Kyösti Mälkki | fb49379 | 2019-06-30 06:29:52 +0300 | [diff] [blame] | 252 | vfct_struct->VBIOSImageOffset = (size_t)header - (size_t)vfct_struct; |
| 253 | |
Matt DeVillier | 7c04d0e | 2023-09-03 12:51:58 -0500 | [diff] [blame] | 254 | /* Calculate and set checksum for VBIOS data if FSP GOP driver used, |
| 255 | Since GOP driver modifies ATOMBIOS tables at end of VBIOS */ |
| 256 | if (CONFIG(RUN_FSP_GOP)) { |
| 257 | /* Clear existing checksum before recalculating */ |
| 258 | header->VbiosContent[VFCT_VBIOS_CHECKSUM_OFFSET] = 0; |
| 259 | header->VbiosContent[VFCT_VBIOS_CHECKSUM_OFFSET] = |
| 260 | acpi_checksum(header->VbiosContent, header->ImageLength); |
| 261 | } |
| 262 | |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 263 | current += header->ImageLength; |
| 264 | return current; |
| 265 | } |
| 266 | |
| 267 | unsigned long |
Furquan Shaikh | 0f007d8 | 2020-04-24 06:41:18 -0700 | [diff] [blame] | 268 | pci_rom_write_acpi_tables(const struct device *device, unsigned long current, |
Elyes HAOUAS | e348066 | 2018-05-06 20:32:23 +0200 | [diff] [blame] | 269 | struct acpi_rsdp *rsdp) |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 270 | { |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 271 | /* Only handle VGA devices */ |
| 272 | if ((device->class >> 8) != PCI_CLASS_DISPLAY_VGA) |
| 273 | return current; |
| 274 | |
| 275 | /* Only handle enabled devices */ |
| 276 | if (!device->enabled) |
| 277 | return current; |
| 278 | |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 279 | /* AMD/ATI uses VFCT */ |
Felix Singer | 43b7f41 | 2022-03-07 04:34:52 +0100 | [diff] [blame] | 280 | if (device->vendor == PCI_VID_ATI) { |
Himanshu Sahdev | 7f1da07 | 2019-10-21 15:27:19 +0530 | [diff] [blame] | 281 | acpi_vfct_t *vfct; |
Kyösti Mälkki | fb49379 | 2019-06-30 06:29:52 +0300 | [diff] [blame] | 282 | |
Felix Held | d5a11ed | 2019-06-20 14:35:44 +0200 | [diff] [blame] | 283 | current = ALIGN_UP(current, 8); |
Himanshu Sahdev | 7f1da07 | 2019-10-21 15:27:19 +0530 | [diff] [blame] | 284 | vfct = (acpi_vfct_t *)current; |
Matt DeVillier | 0cd2a50 | 2023-09-04 08:32:39 -0500 | [diff] [blame] | 285 | acpi_create_vfct(device, vfct, ati_rom_acpi_fill_vfct); |
Kyösti Mälkki | fb49379 | 2019-06-30 06:29:52 +0300 | [diff] [blame] | 286 | if (vfct->header.length) { |
| 287 | printk(BIOS_DEBUG, "ACPI: * VFCT at %lx\n", current); |
| 288 | current += vfct->header.length; |
| 289 | acpi_add_table(rsdp, vfct); |
| 290 | } |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 291 | } |
| 292 | |
| 293 | return current; |
| 294 | } |
Patrick Rudolph | 00c0cd2 | 2017-06-06 19:30:55 +0200 | [diff] [blame] | 295 | |
Furquan Shaikh | 7536a39 | 2020-04-24 21:59:21 -0700 | [diff] [blame] | 296 | void pci_rom_ssdt(const struct device *device) |
Patrick Rudolph | 00c0cd2 | 2017-06-06 19:30:55 +0200 | [diff] [blame] | 297 | { |
| 298 | static size_t ngfx; |
| 299 | |
Benjamin Doron | 90341c1 | 2020-08-04 06:24:03 +0000 | [diff] [blame] | 300 | /* Only handle display devices */ |
| 301 | if ((device->class >> 16) != PCI_BASE_CLASS_DISPLAY) |
Patrick Rudolph | 00c0cd2 | 2017-06-06 19:30:55 +0200 | [diff] [blame] | 302 | return; |
| 303 | |
| 304 | /* Only handle enabled devices */ |
| 305 | if (!device->enabled) |
| 306 | return; |
| 307 | |
| 308 | /* Probe for option rom */ |
| 309 | const struct rom_header *rom = pci_rom_probe(device); |
| 310 | if (!rom || !rom->size) { |
| 311 | printk(BIOS_WARNING, "%s: Missing PCI Option ROM\n", |
| 312 | dev_path(device)); |
| 313 | return; |
| 314 | } |
| 315 | |
| 316 | const char *scope = acpi_device_path(device); |
| 317 | if (!scope) { |
| 318 | printk(BIOS_ERR, "%s: Missing ACPI scope\n", dev_path(device)); |
| 319 | return; |
| 320 | } |
| 321 | |
| 322 | /* Supports up to four devices. */ |
| 323 | if ((CBMEM_ID_ROM0 + ngfx) > CBMEM_ID_ROM3) { |
| 324 | printk(BIOS_ERR, "%s: Out of CBMEM IDs.\n", dev_path(device)); |
| 325 | return; |
| 326 | } |
| 327 | |
| 328 | /* Prepare memory */ |
| 329 | const size_t cbrom_length = rom->size * 512; |
| 330 | if (!cbrom_length) { |
| 331 | printk(BIOS_ERR, "%s: ROM has zero length!\n", |
| 332 | dev_path(device)); |
| 333 | return; |
| 334 | } |
| 335 | |
| 336 | void *cbrom = cbmem_add(CBMEM_ID_ROM0 + ngfx, cbrom_length); |
| 337 | if (!cbrom) { |
| 338 | printk(BIOS_ERR, "%s: Failed to allocate CBMEM.\n", |
| 339 | dev_path(device)); |
| 340 | return; |
| 341 | } |
| 342 | /* Increment CBMEM id for next device */ |
| 343 | ngfx++; |
| 344 | |
| 345 | memcpy(cbrom, rom, cbrom_length); |
| 346 | |
| 347 | /* write _ROM method */ |
| 348 | acpigen_write_scope(scope); |
| 349 | acpigen_write_rom(cbrom, cbrom_length); |
| 350 | acpigen_pop_len(); /* pop scope */ |
| 351 | } |
Patrick Rudolph | a5c2ac6 | 2016-03-31 20:04:23 +0200 | [diff] [blame] | 352 | #endif |