提交 b72c4094 编写于 作者: L Linus Torvalds

Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/jbarnes/pci-2.6

* 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/jbarnes/pci-2.6:
  x86/PCI: truncate _CRS windows with _LEN > _MAX - _MIN + 1
  x86/PCI: for host bridge address space collisions, show conflicting resource
  frv/PCI: remove redundant warnings
  x86/PCI: remove redundant warnings
  PCI: don't say we claimed a resource if we failed
  PCI quirk: Disable MSI on VIA K8T890 systems
  PCI quirk: RS780/RS880: work around missing MSI initialization
  PCI quirk: only apply CX700 PCI bus parking quirk if external VT6212L is present
  PCI: complain about devices that seem to be broken
  PCI: print resources consistently with %pR
  PCI: make disabled window printk style match the enabled ones
  PCI: break out primary/secondary/subordinate for readability
  PCI: for address space collisions, show conflicting resource
  resources: add interfaces that return conflict information
  PCI: cleanup error return for pcix get and set mmrbc functions
  PCI: fix access of PCI_X_CMD by pcix get and set mmrbc functions
  PCI: kill off pci_register_set_vga_state() symbol export.
  PCI: fix return value from pcix_get_max_mmrbc()
...@@ -94,8 +94,7 @@ static void __init pcibios_allocate_bus_resources(struct list_head *bus_list) ...@@ -94,8 +94,7 @@ static void __init pcibios_allocate_bus_resources(struct list_head *bus_list)
r = &dev->resource[idx]; r = &dev->resource[idx];
if (!r->start) if (!r->start)
continue; continue;
if (pci_claim_resource(dev, idx) < 0) pci_claim_resource(dev, idx);
printk(KERN_ERR "PCI: Cannot allocate resource region %d of bridge %s\n", idx, pci_name(dev));
} }
} }
pcibios_allocate_bus_resources(&bus->children); pcibios_allocate_bus_resources(&bus->children);
...@@ -125,7 +124,6 @@ static void __init pcibios_allocate_resources(int pass) ...@@ -125,7 +124,6 @@ static void __init pcibios_allocate_resources(int pass)
DBG("PCI: Resource %08lx-%08lx (f=%lx, d=%d, p=%d)\n", DBG("PCI: Resource %08lx-%08lx (f=%lx, d=%d, p=%d)\n",
r->start, r->end, r->flags, disabled, pass); r->start, r->end, r->flags, disabled, pass);
if (pci_claim_resource(dev, idx) < 0) { if (pci_claim_resource(dev, idx) < 0) {
printk(KERN_ERR "PCI: Cannot allocate resource region %d of device %s\n", idx, pci_name(dev));
/* We'll assign a new address later */ /* We'll assign a new address later */
r->end -= r->start; r->end -= r->start;
r->start = 0; r->start = 0;
......
...@@ -122,8 +122,8 @@ setup_resource(struct acpi_resource *acpi_res, void *data) ...@@ -122,8 +122,8 @@ setup_resource(struct acpi_resource *acpi_res, void *data)
struct acpi_resource_address64 addr; struct acpi_resource_address64 addr;
acpi_status status; acpi_status status;
unsigned long flags; unsigned long flags;
struct resource *root; struct resource *root, *conflict;
u64 start, end; u64 start, end, max_len;
status = resource_to_addr(acpi_res, &addr); status = resource_to_addr(acpi_res, &addr);
if (!ACPI_SUCCESS(status)) if (!ACPI_SUCCESS(status))
...@@ -140,6 +140,17 @@ setup_resource(struct acpi_resource *acpi_res, void *data) ...@@ -140,6 +140,17 @@ setup_resource(struct acpi_resource *acpi_res, void *data)
} else } else
return AE_OK; return AE_OK;
max_len = addr.maximum - addr.minimum + 1;
if (addr.address_length > max_len) {
dev_printk(KERN_DEBUG, &info->bridge->dev,
"host bridge window length %#llx doesn't fit in "
"%#llx-%#llx, trimming\n",
(unsigned long long) addr.address_length,
(unsigned long long) addr.minimum,
(unsigned long long) addr.maximum);
addr.address_length = max_len;
}
start = addr.minimum + addr.translation_offset; start = addr.minimum + addr.translation_offset;
end = start + addr.address_length - 1; end = start + addr.address_length - 1;
...@@ -157,9 +168,12 @@ setup_resource(struct acpi_resource *acpi_res, void *data) ...@@ -157,9 +168,12 @@ setup_resource(struct acpi_resource *acpi_res, void *data)
return AE_OK; return AE_OK;
} }
if (insert_resource(root, res)) { conflict = insert_resource_conflict(root, res);
if (conflict) {
dev_err(&info->bridge->dev, dev_err(&info->bridge->dev,
"can't allocate host bridge window %pR\n", res); "address space collision: host bridge window %pR "
"conflicts with %s %pR\n",
res, conflict->name, conflict);
} else { } else {
pci_bus_add_resource(info->bus, res, 0); pci_bus_add_resource(info->bus, res, 0);
info->res_num++; info->res_num++;
......
...@@ -127,9 +127,6 @@ static void __init pcibios_allocate_bus_resources(struct list_head *bus_list) ...@@ -127,9 +127,6 @@ static void __init pcibios_allocate_bus_resources(struct list_head *bus_list)
continue; continue;
if (!r->start || if (!r->start ||
pci_claim_resource(dev, idx) < 0) { pci_claim_resource(dev, idx) < 0) {
dev_info(&dev->dev,
"can't reserve window %pR\n",
r);
/* /*
* Something is wrong with the region. * Something is wrong with the region.
* Invalidate the resource to prevent * Invalidate the resource to prevent
...@@ -181,8 +178,6 @@ static void __init pcibios_allocate_resources(int pass) ...@@ -181,8 +178,6 @@ static void __init pcibios_allocate_resources(int pass)
"BAR %d: reserving %pr (d=%d, p=%d)\n", "BAR %d: reserving %pr (d=%d, p=%d)\n",
idx, r, disabled, pass); idx, r, disabled, pass);
if (pci_claim_resource(dev, idx) < 0) { if (pci_claim_resource(dev, idx) < 0) {
dev_info(&dev->dev,
"can't reserve %pR\n", r);
/* We'll assign a new address later */ /* We'll assign a new address later */
r->end -= r->start; r->end -= r->start;
r->start = 0; r->start = 0;
......
...@@ -116,13 +116,7 @@ int radeon_irq_kms_init(struct radeon_device *rdev) ...@@ -116,13 +116,7 @@ int radeon_irq_kms_init(struct radeon_device *rdev)
} }
/* enable msi */ /* enable msi */
rdev->msi_enabled = 0; rdev->msi_enabled = 0;
/* MSIs don't seem to work on my rs780; if (rdev->family >= CHIP_RV380) {
* not sure about rs880 or other rs780s.
* Needs more investigation.
*/
if ((rdev->family >= CHIP_RV380) &&
(rdev->family != CHIP_RS780) &&
(rdev->family != CHIP_RS880)) {
int ret = pci_enable_msi(rdev->pdev); int ret = pci_enable_msi(rdev->pdev);
if (!ret) { if (!ret) {
rdev->msi_enabled = 1; rdev->msi_enabled = 1;
......
...@@ -832,9 +832,8 @@ static inline void dbg_ctrl(struct controller *ctrl) ...@@ -832,9 +832,8 @@ static inline void dbg_ctrl(struct controller *ctrl)
for (i = 0; i < DEVICE_COUNT_RESOURCE; i++) { for (i = 0; i < DEVICE_COUNT_RESOURCE; i++) {
if (!pci_resource_len(pdev, i)) if (!pci_resource_len(pdev, i))
continue; continue;
ctrl_info(ctrl, " PCI resource [%d] : 0x%llx@0x%llx\n", ctrl_info(ctrl, " PCI resource [%d] : %pR\n",
i, (unsigned long long)pci_resource_len(pdev, i), i, &pdev->resource[i]);
(unsigned long long)pci_resource_start(pdev, i));
} }
ctrl_info(ctrl, "Slot Capabilities : 0x%08x\n", ctrl->slot_cap); ctrl_info(ctrl, "Slot Capabilities : 0x%08x\n", ctrl->slot_cap);
ctrl_info(ctrl, " Physical Slot Number : %d\n", PSN(ctrl)); ctrl_info(ctrl, " Physical Slot Number : %d\n", PSN(ctrl));
......
...@@ -31,9 +31,9 @@ static int ioapic_probe(struct pci_dev *dev, const struct pci_device_id *ent) ...@@ -31,9 +31,9 @@ static int ioapic_probe(struct pci_dev *dev, const struct pci_device_id *ent)
acpi_status status; acpi_status status;
unsigned long long gsb; unsigned long long gsb;
struct ioapic *ioapic; struct ioapic *ioapic;
u64 addr;
int ret; int ret;
char *type; char *type;
struct resource *res;
handle = DEVICE_ACPI_HANDLE(&dev->dev); handle = DEVICE_ACPI_HANDLE(&dev->dev);
if (!handle) if (!handle)
...@@ -69,13 +69,12 @@ static int ioapic_probe(struct pci_dev *dev, const struct pci_device_id *ent) ...@@ -69,13 +69,12 @@ static int ioapic_probe(struct pci_dev *dev, const struct pci_device_id *ent)
if (pci_request_region(dev, 0, type)) if (pci_request_region(dev, 0, type))
goto exit_disable; goto exit_disable;
addr = pci_resource_start(dev, 0); res = &dev->resource[0];
if (acpi_register_ioapic(ioapic->handle, addr, ioapic->gsi_base)) if (acpi_register_ioapic(ioapic->handle, res->start, ioapic->gsi_base))
goto exit_release; goto exit_release;
pci_set_drvdata(dev, ioapic); pci_set_drvdata(dev, ioapic);
dev_info(&dev->dev, "%s at %#llx, GSI %u\n", type, addr, dev_info(&dev->dev, "%s at %pR, GSI %u\n", type, res, ioapic->gsi_base);
ioapic->gsi_base);
return 0; return 0;
exit_release: exit_release:
......
...@@ -2576,18 +2576,17 @@ EXPORT_SYMBOL_GPL(pci_reset_function); ...@@ -2576,18 +2576,17 @@ EXPORT_SYMBOL_GPL(pci_reset_function);
*/ */
int pcix_get_max_mmrbc(struct pci_dev *dev) int pcix_get_max_mmrbc(struct pci_dev *dev)
{ {
int err, cap; int cap;
u32 stat; u32 stat;
cap = pci_find_capability(dev, PCI_CAP_ID_PCIX); cap = pci_find_capability(dev, PCI_CAP_ID_PCIX);
if (!cap) if (!cap)
return -EINVAL; return -EINVAL;
err = pci_read_config_dword(dev, cap + PCI_X_STATUS, &stat); if (pci_read_config_dword(dev, cap + PCI_X_STATUS, &stat))
if (err)
return -EINVAL; return -EINVAL;
return (stat & PCI_X_STATUS_MAX_READ) >> 12; return 512 << ((stat & PCI_X_STATUS_MAX_READ) >> 21);
} }
EXPORT_SYMBOL(pcix_get_max_mmrbc); EXPORT_SYMBOL(pcix_get_max_mmrbc);
...@@ -2600,18 +2599,17 @@ EXPORT_SYMBOL(pcix_get_max_mmrbc); ...@@ -2600,18 +2599,17 @@ EXPORT_SYMBOL(pcix_get_max_mmrbc);
*/ */
int pcix_get_mmrbc(struct pci_dev *dev) int pcix_get_mmrbc(struct pci_dev *dev)
{ {
int ret, cap; int cap;
u32 cmd; u16 cmd;
cap = pci_find_capability(dev, PCI_CAP_ID_PCIX); cap = pci_find_capability(dev, PCI_CAP_ID_PCIX);
if (!cap) if (!cap)
return -EINVAL; return -EINVAL;
ret = pci_read_config_dword(dev, cap + PCI_X_CMD, &cmd); if (pci_read_config_word(dev, cap + PCI_X_CMD, &cmd))
if (!ret) return -EINVAL;
ret = 512 << ((cmd & PCI_X_CMD_MAX_READ) >> 2);
return ret; return 512 << ((cmd & PCI_X_CMD_MAX_READ) >> 2);
} }
EXPORT_SYMBOL(pcix_get_mmrbc); EXPORT_SYMBOL(pcix_get_mmrbc);
...@@ -2626,28 +2624,27 @@ EXPORT_SYMBOL(pcix_get_mmrbc); ...@@ -2626,28 +2624,27 @@ EXPORT_SYMBOL(pcix_get_mmrbc);
*/ */
int pcix_set_mmrbc(struct pci_dev *dev, int mmrbc) int pcix_set_mmrbc(struct pci_dev *dev, int mmrbc)
{ {
int cap, err = -EINVAL; int cap;
u32 stat, cmd, v, o; u32 stat, v, o;
u16 cmd;
if (mmrbc < 512 || mmrbc > 4096 || !is_power_of_2(mmrbc)) if (mmrbc < 512 || mmrbc > 4096 || !is_power_of_2(mmrbc))
goto out; return -EINVAL;
v = ffs(mmrbc) - 10; v = ffs(mmrbc) - 10;
cap = pci_find_capability(dev, PCI_CAP_ID_PCIX); cap = pci_find_capability(dev, PCI_CAP_ID_PCIX);
if (!cap) if (!cap)
goto out; return -EINVAL;
err = pci_read_config_dword(dev, cap + PCI_X_STATUS, &stat); if (pci_read_config_dword(dev, cap + PCI_X_STATUS, &stat))
if (err) return -EINVAL;
goto out;
if (v > (stat & PCI_X_STATUS_MAX_READ) >> 21) if (v > (stat & PCI_X_STATUS_MAX_READ) >> 21)
return -E2BIG; return -E2BIG;
err = pci_read_config_dword(dev, cap + PCI_X_CMD, &cmd); if (pci_read_config_word(dev, cap + PCI_X_CMD, &cmd))
if (err) return -EINVAL;
goto out;
o = (cmd & PCI_X_CMD_MAX_READ) >> 2; o = (cmd & PCI_X_CMD_MAX_READ) >> 2;
if (o != v) { if (o != v) {
...@@ -2657,10 +2654,10 @@ int pcix_set_mmrbc(struct pci_dev *dev, int mmrbc) ...@@ -2657,10 +2654,10 @@ int pcix_set_mmrbc(struct pci_dev *dev, int mmrbc)
cmd &= ~PCI_X_CMD_MAX_READ; cmd &= ~PCI_X_CMD_MAX_READ;
cmd |= v << 2; cmd |= v << 2;
err = pci_write_config_dword(dev, cap + PCI_X_CMD, cmd); if (pci_write_config_word(dev, cap + PCI_X_CMD, cmd))
return -EIO;
} }
out: return 0;
return err;
} }
EXPORT_SYMBOL(pcix_set_mmrbc); EXPORT_SYMBOL(pcix_set_mmrbc);
...@@ -3023,7 +3020,6 @@ EXPORT_SYMBOL(pcim_pin_device); ...@@ -3023,7 +3020,6 @@ EXPORT_SYMBOL(pcim_pin_device);
EXPORT_SYMBOL(pci_disable_device); EXPORT_SYMBOL(pci_disable_device);
EXPORT_SYMBOL(pci_find_capability); EXPORT_SYMBOL(pci_find_capability);
EXPORT_SYMBOL(pci_bus_find_capability); EXPORT_SYMBOL(pci_bus_find_capability);
EXPORT_SYMBOL(pci_register_set_vga_state);
EXPORT_SYMBOL(pci_release_regions); EXPORT_SYMBOL(pci_release_regions);
EXPORT_SYMBOL(pci_request_regions); EXPORT_SYMBOL(pci_request_regions);
EXPORT_SYMBOL(pci_request_regions_exclusive); EXPORT_SYMBOL(pci_request_regions_exclusive);
......
...@@ -174,14 +174,19 @@ int __pci_read_base(struct pci_dev *dev, enum pci_bar_type type, ...@@ -174,14 +174,19 @@ int __pci_read_base(struct pci_dev *dev, enum pci_bar_type type,
pci_read_config_dword(dev, pos, &sz); pci_read_config_dword(dev, pos, &sz);
pci_write_config_dword(dev, pos, l); pci_write_config_dword(dev, pos, l);
if (!sz)
goto fail; /* BAR not implemented */
/* /*
* All bits set in sz means the device isn't working properly. * All bits set in sz means the device isn't working properly.
* If the BAR isn't implemented, all bits must be 0. If it's a * If it's a memory BAR or a ROM, bit 0 must be clear; if it's
* memory BAR or a ROM, bit 0 must be clear; if it's an io BAR, bit * an io BAR, bit 1 must be clear.
* 1 must be clear.
*/ */
if (!sz || sz == 0xffffffff) if (sz == 0xffffffff) {
dev_err(&dev->dev, "reg %x: invalid size %#x; broken device?\n",
pos, sz);
goto fail; goto fail;
}
/* /*
* I don't know how l can have all bits set. Copied from old code. * I don't know how l can have all bits set. Copied from old code.
...@@ -244,13 +249,17 @@ int __pci_read_base(struct pci_dev *dev, enum pci_bar_type type, ...@@ -244,13 +249,17 @@ int __pci_read_base(struct pci_dev *dev, enum pci_bar_type type,
pos, res); pos, res);
} }
} else { } else {
sz = pci_size(l, sz, mask); u32 size = pci_size(l, sz, mask);
if (!sz) if (!size) {
dev_err(&dev->dev, "reg %x: invalid size "
"(l %#x sz %#x mask %#x); broken device?",
pos, l, sz, mask);
goto fail; goto fail;
}
res->start = l; res->start = l;
res->end = l + sz; res->end = l + size;
dev_printk(KERN_DEBUG, &dev->dev, "reg %x: %pR\n", pos, res); dev_printk(KERN_DEBUG, &dev->dev, "reg %x: %pR\n", pos, res);
} }
...@@ -312,7 +321,7 @@ static void __devinit pci_read_bridge_io(struct pci_bus *child) ...@@ -312,7 +321,7 @@ static void __devinit pci_read_bridge_io(struct pci_bus *child)
dev_printk(KERN_DEBUG, &dev->dev, " bridge window %pR\n", res); dev_printk(KERN_DEBUG, &dev->dev, " bridge window %pR\n", res);
} else { } else {
dev_printk(KERN_DEBUG, &dev->dev, dev_printk(KERN_DEBUG, &dev->dev,
" bridge window [io %04lx - %04lx] reg reading\n", " bridge window [io %#06lx-%#06lx] (disabled)\n",
base, limit); base, limit);
} }
} }
...@@ -336,7 +345,7 @@ static void __devinit pci_read_bridge_mmio(struct pci_bus *child) ...@@ -336,7 +345,7 @@ static void __devinit pci_read_bridge_mmio(struct pci_bus *child)
dev_printk(KERN_DEBUG, &dev->dev, " bridge window %pR\n", res); dev_printk(KERN_DEBUG, &dev->dev, " bridge window %pR\n", res);
} else { } else {
dev_printk(KERN_DEBUG, &dev->dev, dev_printk(KERN_DEBUG, &dev->dev,
" bridge window [mem 0x%08lx - 0x%08lx] reg reading\n", " bridge window [mem %#010lx-%#010lx] (disabled)\n",
base, limit + 0xfffff); base, limit + 0xfffff);
} }
} }
...@@ -387,7 +396,7 @@ static void __devinit pci_read_bridge_mmio_pref(struct pci_bus *child) ...@@ -387,7 +396,7 @@ static void __devinit pci_read_bridge_mmio_pref(struct pci_bus *child)
dev_printk(KERN_DEBUG, &dev->dev, " bridge window %pR\n", res); dev_printk(KERN_DEBUG, &dev->dev, " bridge window %pR\n", res);
} else { } else {
dev_printk(KERN_DEBUG, &dev->dev, dev_printk(KERN_DEBUG, &dev->dev,
" bridge window [mem 0x%08lx - %08lx pref] reg reading\n", " bridge window [mem %#010lx-%#010lx pref] (disabled)\n",
base, limit + 0xfffff); base, limit + 0xfffff);
} }
} }
...@@ -673,16 +682,20 @@ int __devinit pci_scan_bridge(struct pci_bus *bus, struct pci_dev *dev, int max, ...@@ -673,16 +682,20 @@ int __devinit pci_scan_bridge(struct pci_bus *bus, struct pci_dev *dev, int max,
int is_cardbus = (dev->hdr_type == PCI_HEADER_TYPE_CARDBUS); int is_cardbus = (dev->hdr_type == PCI_HEADER_TYPE_CARDBUS);
u32 buses, i, j = 0; u32 buses, i, j = 0;
u16 bctl; u16 bctl;
u8 primary, secondary, subordinate;
int broken = 0; int broken = 0;
pci_read_config_dword(dev, PCI_PRIMARY_BUS, &buses); pci_read_config_dword(dev, PCI_PRIMARY_BUS, &buses);
primary = buses & 0xFF;
secondary = (buses >> 8) & 0xFF;
subordinate = (buses >> 16) & 0xFF;
dev_dbg(&dev->dev, "scanning behind bridge, config %06x, pass %d\n", dev_dbg(&dev->dev, "scanning [bus %02x-%02x] behind bridge, pass %d\n",
buses & 0xffffff, pass); secondary, subordinate, pass);
/* Check if setup is sensible at all */ /* Check if setup is sensible at all */
if (!pass && if (!pass &&
((buses & 0xff) != bus->number || ((buses >> 8) & 0xff) <= bus->number)) { (primary != bus->number || secondary <= bus->number)) {
dev_dbg(&dev->dev, "bus configuration invalid, reconfiguring\n"); dev_dbg(&dev->dev, "bus configuration invalid, reconfiguring\n");
broken = 1; broken = 1;
} }
...@@ -693,15 +706,15 @@ int __devinit pci_scan_bridge(struct pci_bus *bus, struct pci_dev *dev, int max, ...@@ -693,15 +706,15 @@ int __devinit pci_scan_bridge(struct pci_bus *bus, struct pci_dev *dev, int max,
pci_write_config_word(dev, PCI_BRIDGE_CONTROL, pci_write_config_word(dev, PCI_BRIDGE_CONTROL,
bctl & ~PCI_BRIDGE_CTL_MASTER_ABORT); bctl & ~PCI_BRIDGE_CTL_MASTER_ABORT);
if ((buses & 0xffff00) && !pcibios_assign_all_busses() && !is_cardbus && !broken) { if ((secondary || subordinate) && !pcibios_assign_all_busses() &&
unsigned int cmax, busnr; !is_cardbus && !broken) {
unsigned int cmax;
/* /*
* Bus already configured by firmware, process it in the first * Bus already configured by firmware, process it in the first
* pass and just note the configuration. * pass and just note the configuration.
*/ */
if (pass) if (pass)
goto out; goto out;
busnr = (buses >> 8) & 0xFF;
/* /*
* If we already got to this bus through a different bridge, * If we already got to this bus through a different bridge,
...@@ -710,13 +723,13 @@ int __devinit pci_scan_bridge(struct pci_bus *bus, struct pci_dev *dev, int max, ...@@ -710,13 +723,13 @@ int __devinit pci_scan_bridge(struct pci_bus *bus, struct pci_dev *dev, int max,
* However, we continue to descend down the hierarchy and * However, we continue to descend down the hierarchy and
* scan remaining child buses. * scan remaining child buses.
*/ */
child = pci_find_bus(pci_domain_nr(bus), busnr); child = pci_find_bus(pci_domain_nr(bus), secondary);
if (!child) { if (!child) {
child = pci_add_new_bus(bus, dev, busnr); child = pci_add_new_bus(bus, dev, secondary);
if (!child) if (!child)
goto out; goto out;
child->primary = buses & 0xFF; child->primary = primary;
child->subordinate = (buses >> 16) & 0xFF; child->subordinate = subordinate;
child->bridge_ctl = bctl; child->bridge_ctl = bctl;
} }
......
...@@ -368,8 +368,9 @@ static void __devinit quirk_io_region(struct pci_dev *dev, unsigned region, ...@@ -368,8 +368,9 @@ static void __devinit quirk_io_region(struct pci_dev *dev, unsigned region,
bus_region.end = res->end; bus_region.end = res->end;
pcibios_bus_to_resource(dev, res, &bus_region); pcibios_bus_to_resource(dev, res, &bus_region);
pci_claim_resource(dev, nr); if (pci_claim_resource(dev, nr) == 0)
dev_info(&dev->dev, "quirk: %pR claimed by %s\n", res, name); dev_info(&dev->dev, "quirk: %pR claimed by %s\n",
res, name);
} }
} }
...@@ -1977,11 +1978,25 @@ static void __devinit quirk_via_cx700_pci_parking_caching(struct pci_dev *dev) ...@@ -1977,11 +1978,25 @@ static void __devinit quirk_via_cx700_pci_parking_caching(struct pci_dev *dev)
/* /*
* Disable PCI Bus Parking and PCI Master read caching on CX700 * Disable PCI Bus Parking and PCI Master read caching on CX700
* which causes unspecified timing errors with a VT6212L on the PCI * which causes unspecified timing errors with a VT6212L on the PCI
* bus leading to USB2.0 packet loss. The defaults are that these * bus leading to USB2.0 packet loss.
* features are turned off but some BIOSes turn them on. *
* This quirk is only enabled if a second (on the external PCI bus)
* VT6212L is found -- the CX700 core itself also contains a USB
* host controller with the same PCI ID as the VT6212L.
*/ */
/* Count VT6212L instances */
struct pci_dev *p = pci_get_device(PCI_VENDOR_ID_VIA,
PCI_DEVICE_ID_VIA_8235_USB_2, NULL);
uint8_t b; uint8_t b;
/* p should contain the first (internal) VT6212L -- see if we have
an external one by searching again */
p = pci_get_device(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_8235_USB_2, p);
if (!p)
return;
pci_dev_put(p);
if (pci_read_config_byte(dev, 0x76, &b) == 0) { if (pci_read_config_byte(dev, 0x76, &b) == 0) {
if (b & 0x40) { if (b & 0x40) {
/* Turn off PCI Bus Parking */ /* Turn off PCI Bus Parking */
...@@ -2008,7 +2023,7 @@ static void __devinit quirk_via_cx700_pci_parking_caching(struct pci_dev *dev) ...@@ -2008,7 +2023,7 @@ static void __devinit quirk_via_cx700_pci_parking_caching(struct pci_dev *dev)
} }
} }
} }
DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_VIA, 0x324e, quirk_via_cx700_pci_parking_caching); DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_VIA, 0x324e, quirk_via_cx700_pci_parking_caching);
/* /*
* For Broadcom 5706, 5708, 5709 rev. A nics, any read beyond the * For Broadcom 5706, 5708, 5709 rev. A nics, any read beyond the
...@@ -2108,6 +2123,7 @@ static void __devinit quirk_disable_msi(struct pci_dev *dev) ...@@ -2108,6 +2123,7 @@ static void __devinit quirk_disable_msi(struct pci_dev *dev)
} }
} }
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_8131_BRIDGE, quirk_disable_msi); DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_8131_BRIDGE, quirk_disable_msi);
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_VIA, 0xa238, quirk_disable_msi);
/* Go through the list of Hypertransport capabilities and /* Go through the list of Hypertransport capabilities and
* return 1 if a HT MSI capability is found and enabled */ * return 1 if a HT MSI capability is found and enabled */
...@@ -2479,6 +2495,39 @@ DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATI, 0x4374, ...@@ -2479,6 +2495,39 @@ DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATI, 0x4374,
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATI, 0x4375, DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATI, 0x4375,
quirk_msi_intx_disable_bug); quirk_msi_intx_disable_bug);
/*
* MSI does not work with the AMD RS780/RS880 internal graphics and HDMI audio
* devices unless the BIOS has initialized the nb_cntl.strap_msi_enable bit.
*/
static void __init rs780_int_gfx_disable_msi(struct pci_dev *int_gfx_bridge)
{
u32 nb_cntl;
if (!int_gfx_bridge->subordinate)
return;
pci_bus_write_config_dword(int_gfx_bridge->bus, PCI_DEVFN(0, 0),
0x60, 0);
pci_bus_read_config_dword(int_gfx_bridge->bus, PCI_DEVFN(0, 0),
0x64, &nb_cntl);
if (!(nb_cntl & BIT(10))) {
dev_warn(&int_gfx_bridge->dev,
FW_WARN "RS780: MSI for internal graphics disabled\n");
int_gfx_bridge->subordinate->bus_flags |= PCI_BUS_FLAGS_NO_MSI;
}
}
#define PCI_DEVICE_ID_AMD_RS780_P2P_INT_GFX 0x9602
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_AMD,
PCI_DEVICE_ID_AMD_RS780_P2P_INT_GFX,
rs780_int_gfx_disable_msi);
/* wrong vendor ID on M4A785TD motherboard: */
DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ASUSTEK,
PCI_DEVICE_ID_AMD_RS780_P2P_INT_GFX,
rs780_int_gfx_disable_msi);
#endif /* CONFIG_PCI_MSI */ #endif /* CONFIG_PCI_MSI */
#ifdef CONFIG_PCI_IOV #ifdef CONFIG_PCI_IOV
......
...@@ -93,8 +93,7 @@ void pci_update_resource(struct pci_dev *dev, int resno) ...@@ -93,8 +93,7 @@ void pci_update_resource(struct pci_dev *dev, int resno)
int pci_claim_resource(struct pci_dev *dev, int resource) int pci_claim_resource(struct pci_dev *dev, int resource)
{ {
struct resource *res = &dev->resource[resource]; struct resource *res = &dev->resource[resource];
struct resource *root; struct resource *root, *conflict;
int err;
root = pci_find_parent_resource(dev, res); root = pci_find_parent_resource(dev, res);
if (!root) { if (!root) {
...@@ -103,12 +102,15 @@ int pci_claim_resource(struct pci_dev *dev, int resource) ...@@ -103,12 +102,15 @@ int pci_claim_resource(struct pci_dev *dev, int resource)
return -EINVAL; return -EINVAL;
} }
err = request_resource(root, res); conflict = request_resource_conflict(root, res);
if (err) if (conflict) {
dev_err(&dev->dev, dev_err(&dev->dev,
"address space collision: %pR already in use\n", res); "address space collision: %pR conflicts with %s %pR\n",
res, conflict->name, conflict);
return -EBUSY;
}
return err; return 0;
} }
EXPORT_SYMBOL(pci_claim_resource); EXPORT_SYMBOL(pci_claim_resource);
......
...@@ -874,10 +874,8 @@ static int nonstatic_autoadd_resources(struct pcmcia_socket *s) ...@@ -874,10 +874,8 @@ static int nonstatic_autoadd_resources(struct pcmcia_socket *s)
if (res == &ioport_resource) if (res == &ioport_resource)
continue; continue;
dev_printk(KERN_INFO, &s->cb_dev->dev, dev_printk(KERN_INFO, &s->cb_dev->dev,
"pcmcia: parent PCI bridge I/O " "pcmcia: parent PCI bridge window: %pR\n",
"window: 0x%llx - 0x%llx\n", res);
(unsigned long long)res->start,
(unsigned long long)res->end);
if (!adjust_io(s, ADD_MANAGED_RESOURCE, res->start, res->end)) if (!adjust_io(s, ADD_MANAGED_RESOURCE, res->start, res->end))
done |= IORESOURCE_IO; done |= IORESOURCE_IO;
...@@ -887,10 +885,8 @@ static int nonstatic_autoadd_resources(struct pcmcia_socket *s) ...@@ -887,10 +885,8 @@ static int nonstatic_autoadd_resources(struct pcmcia_socket *s)
if (res == &iomem_resource) if (res == &iomem_resource)
continue; continue;
dev_printk(KERN_INFO, &s->cb_dev->dev, dev_printk(KERN_INFO, &s->cb_dev->dev,
"pcmcia: parent PCI bridge Memory " "pcmcia: parent PCI bridge window: %pR\n",
"window: 0x%llx - 0x%llx\n", res);
(unsigned long long)res->start,
(unsigned long long)res->end);
if (!adjust_memory(s, ADD_MANAGED_RESOURCE, res->start, res->end)) if (!adjust_memory(s, ADD_MANAGED_RESOURCE, res->start, res->end))
done |= IORESOURCE_MEM; done |= IORESOURCE_MEM;
} }
......
...@@ -112,12 +112,14 @@ struct resource_list { ...@@ -112,12 +112,14 @@ struct resource_list {
extern struct resource ioport_resource; extern struct resource ioport_resource;
extern struct resource iomem_resource; extern struct resource iomem_resource;
extern struct resource *request_resource_conflict(struct resource *root, struct resource *new);
extern int request_resource(struct resource *root, struct resource *new); extern int request_resource(struct resource *root, struct resource *new);
extern int release_resource(struct resource *new); extern int release_resource(struct resource *new);
void release_child_resources(struct resource *new); void release_child_resources(struct resource *new);
extern void reserve_region_with_split(struct resource *root, extern void reserve_region_with_split(struct resource *root,
resource_size_t start, resource_size_t end, resource_size_t start, resource_size_t end,
const char *name); const char *name);
extern struct resource *insert_resource_conflict(struct resource *parent, struct resource *new);
extern int insert_resource(struct resource *parent, struct resource *new); extern int insert_resource(struct resource *parent, struct resource *new);
extern void insert_resource_expand_to_fit(struct resource *root, struct resource *new); extern void insert_resource_expand_to_fit(struct resource *root, struct resource *new);
extern int allocate_resource(struct resource *root, struct resource *new, extern int allocate_resource(struct resource *root, struct resource *new,
......
...@@ -219,19 +219,34 @@ void release_child_resources(struct resource *r) ...@@ -219,19 +219,34 @@ void release_child_resources(struct resource *r)
} }
/** /**
* request_resource - request and reserve an I/O or memory resource * request_resource_conflict - request and reserve an I/O or memory resource
* @root: root resource descriptor * @root: root resource descriptor
* @new: resource descriptor desired by caller * @new: resource descriptor desired by caller
* *
* Returns 0 for success, negative error code on error. * Returns 0 for success, conflict resource on error.
*/ */
int request_resource(struct resource *root, struct resource *new) struct resource *request_resource_conflict(struct resource *root, struct resource *new)
{ {
struct resource *conflict; struct resource *conflict;
write_lock(&resource_lock); write_lock(&resource_lock);
conflict = __request_resource(root, new); conflict = __request_resource(root, new);
write_unlock(&resource_lock); write_unlock(&resource_lock);
return conflict;
}
/**
* request_resource - request and reserve an I/O or memory resource
* @root: root resource descriptor
* @new: resource descriptor desired by caller
*
* Returns 0 for success, negative error code on error.
*/
int request_resource(struct resource *root, struct resource *new)
{
struct resource *conflict;
conflict = request_resource_conflict(root, new);
return conflict ? -EBUSY : 0; return conflict ? -EBUSY : 0;
} }
...@@ -474,25 +489,40 @@ static struct resource * __insert_resource(struct resource *parent, struct resou ...@@ -474,25 +489,40 @@ static struct resource * __insert_resource(struct resource *parent, struct resou
} }
/** /**
* insert_resource - Inserts a resource in the resource tree * insert_resource_conflict - Inserts resource in the resource tree
* @parent: parent of the new resource * @parent: parent of the new resource
* @new: new resource to insert * @new: new resource to insert
* *
* Returns 0 on success, -EBUSY if the resource can't be inserted. * Returns 0 on success, conflict resource if the resource can't be inserted.
* *
* This function is equivalent to request_resource when no conflict * This function is equivalent to request_resource_conflict when no conflict
* happens. If a conflict happens, and the conflicting resources * happens. If a conflict happens, and the conflicting resources
* entirely fit within the range of the new resource, then the new * entirely fit within the range of the new resource, then the new
* resource is inserted and the conflicting resources become children of * resource is inserted and the conflicting resources become children of
* the new resource. * the new resource.
*/ */
int insert_resource(struct resource *parent, struct resource *new) struct resource *insert_resource_conflict(struct resource *parent, struct resource *new)
{ {
struct resource *conflict; struct resource *conflict;
write_lock(&resource_lock); write_lock(&resource_lock);
conflict = __insert_resource(parent, new); conflict = __insert_resource(parent, new);
write_unlock(&resource_lock); write_unlock(&resource_lock);
return conflict;
}
/**
* insert_resource - Inserts a resource in the resource tree
* @parent: parent of the new resource
* @new: new resource to insert
*
* Returns 0 on success, -EBUSY if the resource can't be inserted.
*/
int insert_resource(struct resource *parent, struct resource *new)
{
struct resource *conflict;
conflict = insert_resource_conflict(parent, new);
return conflict ? -EBUSY : 0; return conflict ? -EBUSY : 0;
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册