| |
| |
| |
| |
| |
| |
| |
| #include <linux/kernel.h> |
| #include <linux/export.h> |
| #include <linux/pci.h> |
| #include <linux/slab.h> |
| |
| #include "pci.h" |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| int pci_enable_rom(struct pci_dev *pdev) |
| { |
| <------>struct resource *res = &pdev->resource[PCI_ROM_RESOURCE]; |
| <------>struct pci_bus_region region; |
| <------>u32 rom_addr; |
| |
| <------>if (!res->flags) |
| <------><------>return -1; |
| |
| <------> |
| <------>if (res->flags & IORESOURCE_ROM_SHADOW) |
| <------><------>return 0; |
| |
| <------> |
| <------> * Ideally pci_update_resource() would update the ROM BAR address, |
| <------> * and we would only set the enable bit here. But apparently some |
| <------> * devices have buggy ROM BARs that read as zero when disabled. |
| <------> */ |
| <------>pcibios_resource_to_bus(pdev->bus, ®ion, res); |
| <------>pci_read_config_dword(pdev, pdev->rom_base_reg, &rom_addr); |
| <------>rom_addr &= ~PCI_ROM_ADDRESS_MASK; |
| <------>rom_addr |= region.start | PCI_ROM_ADDRESS_ENABLE; |
| <------>pci_write_config_dword(pdev, pdev->rom_base_reg, rom_addr); |
| <------>return 0; |
| } |
| EXPORT_SYMBOL_GPL(pci_enable_rom); |
| |
| |
| |
| |
| |
| |
| |
| |
| void pci_disable_rom(struct pci_dev *pdev) |
| { |
| <------>struct resource *res = &pdev->resource[PCI_ROM_RESOURCE]; |
| <------>u32 rom_addr; |
| |
| <------>if (res->flags & IORESOURCE_ROM_SHADOW) |
| <------><------>return; |
| |
| <------>pci_read_config_dword(pdev, pdev->rom_base_reg, &rom_addr); |
| <------>rom_addr &= ~PCI_ROM_ADDRESS_ENABLE; |
| <------>pci_write_config_dword(pdev, pdev->rom_base_reg, rom_addr); |
| } |
| EXPORT_SYMBOL_GPL(pci_disable_rom); |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static size_t pci_get_rom_size(struct pci_dev *pdev, void __iomem *rom, |
| <------><------><------> size_t size) |
| { |
| <------>void __iomem *image; |
| <------>int last_image; |
| <------>unsigned length; |
| |
| <------>image = rom; |
| <------>do { |
| <------><------>void __iomem *pds; |
| <------><------> |
| <------><------>if (readw(image) != 0xAA55) { |
| <------><------><------>pci_info(pdev, "Invalid PCI ROM header signature: expecting 0xaa55, got %#06x\n", |
| <------><------><------><------> readw(image)); |
| <------><------><------>break; |
| <------><------>} |
| <------><------> |
| <------><------>pds = image + readw(image + 24); |
| <------><------>if (readl(pds) != 0x52494350) { |
| <------><------><------>pci_info(pdev, "Invalid PCI ROM data signature: expecting 0x52494350, got %#010x\n", |
| <------><------><------><------> readl(pds)); |
| <------><------><------>break; |
| <------><------>} |
| <------><------>last_image = readb(pds + 21) & 0x80; |
| <------><------>length = readw(pds + 16); |
| <------><------>image += length * 512; |
| <------><------> |
| <------><------>if (image >= rom + size) |
| <------><------><------>break; |
| <------><------>if (!last_image) { |
| <------><------><------>if (readw(image) != 0xAA55) { |
| <------><------><------><------>pci_info(pdev, "No more image in the PCI ROM\n"); |
| <------><------><------><------>break; |
| <------><------><------>} |
| <------><------>} |
| <------>} while (length && !last_image); |
| |
| <------> |
| <------> |
| <------>return min((size_t)(image - rom), size); |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| void __iomem *pci_map_rom(struct pci_dev *pdev, size_t *size) |
| { |
| <------>struct resource *res = &pdev->resource[PCI_ROM_RESOURCE]; |
| <------>loff_t start; |
| <------>void __iomem *rom; |
| |
| <------> |
| <------>if (res->parent == NULL && pci_assign_resource(pdev, PCI_ROM_RESOURCE)) |
| <------><------>return NULL; |
| |
| <------>start = pci_resource_start(pdev, PCI_ROM_RESOURCE); |
| <------>*size = pci_resource_len(pdev, PCI_ROM_RESOURCE); |
| <------>if (*size == 0) |
| <------><------>return NULL; |
| |
| <------> |
| <------>if (pci_enable_rom(pdev)) |
| <------><------>return NULL; |
| |
| <------>rom = ioremap(start, *size); |
| <------>if (!rom) |
| <------><------>goto err_ioremap; |
| |
| <------> |
| <------> * Try to find the true size of the ROM since sometimes the PCI window |
| <------> * size is much larger than the actual size of the ROM. |
| <------> * True size is important if the ROM is going to be copied. |
| <------> */ |
| <------>*size = pci_get_rom_size(pdev, rom, *size); |
| <------>if (!*size) |
| <------><------>goto invalid_rom; |
| |
| <------>return rom; |
| |
| invalid_rom: |
| <------>iounmap(rom); |
| err_ioremap: |
| <------> |
| <------>if (!(res->flags & IORESOURCE_ROM_ENABLE)) |
| <------><------>pci_disable_rom(pdev); |
| <------>return NULL; |
| } |
| EXPORT_SYMBOL(pci_map_rom); |
| |
| |
| |
| |
| |
| |
| |
| |
| void pci_unmap_rom(struct pci_dev *pdev, void __iomem *rom) |
| { |
| <------>struct resource *res = &pdev->resource[PCI_ROM_RESOURCE]; |
| |
| <------>iounmap(rom); |
| |
| <------> |
| <------>if (!(res->flags & IORESOURCE_ROM_ENABLE)) |
| <------><------>pci_disable_rom(pdev); |
| } |
| EXPORT_SYMBOL(pci_unmap_rom); |
| |