diff options
| -rw-r--r-- | arch/frv/mb93090-mb00/pci-frv.c | 4 | ||||
| -rw-r--r-- | arch/x86/pci/acpi.c | 22 | ||||
| -rw-r--r-- | arch/x86/pci/i386.c | 5 | ||||
| -rw-r--r-- | drivers/gpu/drm/radeon/radeon_irq_kms.c | 8 | ||||
| -rw-r--r-- | drivers/pci/hotplug/pciehp_hpc.c | 5 | ||||
| -rw-r--r-- | drivers/pci/ioapic.c | 9 | ||||
| -rw-r--r-- | drivers/pci/pci.c | 44 | ||||
| -rw-r--r-- | drivers/pci/probe.c | 53 | ||||
| -rw-r--r-- | drivers/pci/quirks.c | 59 | ||||
| -rw-r--r-- | drivers/pci/setup-res.c | 14 | ||||
| -rw-r--r-- | drivers/pcmcia/rsrc_nonstatic.c | 12 | ||||
| -rw-r--r-- | include/linux/ioport.h | 2 | ||||
| -rw-r--r-- | kernel/resource.c | 44 | 
13 files changed, 184 insertions, 97 deletions
diff --git a/arch/frv/mb93090-mb00/pci-frv.c b/arch/frv/mb93090-mb00/pci-frv.c index 1ed15d7fea2..16bc2cb5649 100644 --- a/arch/frv/mb93090-mb00/pci-frv.c +++ b/arch/frv/mb93090-mb00/pci-frv.c @@ -94,8 +94,7 @@ static void __init pcibios_allocate_bus_resources(struct list_head *bus_list)  				r = &dev->resource[idx];  				if (!r->start)  					continue; -				if (pci_claim_resource(dev, idx) < 0) -					printk(KERN_ERR "PCI: Cannot allocate resource region %d of bridge %s\n", idx, pci_name(dev)); +				pci_claim_resource(dev, idx);  			}  		}  		pcibios_allocate_bus_resources(&bus->children); @@ -125,7 +124,6 @@ static void __init pcibios_allocate_resources(int pass)  				DBG("PCI: Resource %08lx-%08lx (f=%lx, d=%d, p=%d)\n",  				    r->start, r->end, r->flags, disabled, pass);  				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 */  					r->end -= r->start;  					r->start = 0; diff --git a/arch/x86/pci/acpi.c b/arch/x86/pci/acpi.c index 6e22454bfaa..e31160216ef 100644 --- a/arch/x86/pci/acpi.c +++ b/arch/x86/pci/acpi.c @@ -122,8 +122,8 @@ setup_resource(struct acpi_resource *acpi_res, void *data)  	struct acpi_resource_address64 addr;  	acpi_status status;  	unsigned long flags; -	struct resource *root; -	u64 start, end; +	struct resource *root, *conflict; +	u64 start, end, max_len;  	status = resource_to_addr(acpi_res, &addr);  	if (!ACPI_SUCCESS(status)) @@ -140,6 +140,17 @@ setup_resource(struct acpi_resource *acpi_res, void *data)  	} else  		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;  	end = start + addr.address_length - 1; @@ -157,9 +168,12 @@ setup_resource(struct acpi_resource *acpi_res, void *data)  		return AE_OK;  	} -	if (insert_resource(root, res)) { +	conflict = insert_resource_conflict(root, res); +	if (conflict) {  		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 {  		pci_bus_add_resource(info->bus, res, 0);  		info->res_num++; diff --git a/arch/x86/pci/i386.c b/arch/x86/pci/i386.c index dece3eb9c90..46fd43f7910 100644 --- a/arch/x86/pci/i386.c +++ b/arch/x86/pci/i386.c @@ -127,9 +127,6 @@ static void __init pcibios_allocate_bus_resources(struct list_head *bus_list)  					continue;  				if (!r->start ||  				    pci_claim_resource(dev, idx) < 0) { -					dev_info(&dev->dev, -						 "can't reserve window %pR\n", -						 r);  					/*  					 * Something is wrong with the region.  					 * Invalidate the resource to prevent @@ -181,8 +178,6 @@ static void __init pcibios_allocate_resources(int pass)  					"BAR %d: reserving %pr (d=%d, p=%d)\n",  					idx, r, disabled, pass);  				if (pci_claim_resource(dev, idx) < 0) { -					dev_info(&dev->dev, -						 "can't reserve %pR\n", r);  					/* We'll assign a new address later */  					r->end -= r->start;  					r->start = 0; diff --git a/drivers/gpu/drm/radeon/radeon_irq_kms.c b/drivers/gpu/drm/radeon/radeon_irq_kms.c index 3cfd60fd008..ea4c645ece1 100644 --- a/drivers/gpu/drm/radeon/radeon_irq_kms.c +++ b/drivers/gpu/drm/radeon/radeon_irq_kms.c @@ -116,13 +116,7 @@ int radeon_irq_kms_init(struct radeon_device *rdev)  	}  	/* enable msi */  	rdev->msi_enabled = 0; -	/* MSIs don't seem to work on my rs780; -	 * not sure about rs880 or other rs780s. -	 * Needs more investigation. -	 */ -	if ((rdev->family >= CHIP_RV380) && -	    (rdev->family != CHIP_RS780) && -	    (rdev->family != CHIP_RS880)) { +	if (rdev->family >= CHIP_RV380) {  		int ret = pci_enable_msi(rdev->pdev);  		if (!ret) {  			rdev->msi_enabled = 1; diff --git a/drivers/pci/hotplug/pciehp_hpc.c b/drivers/pci/hotplug/pciehp_hpc.c index 40b48f569b1..9665d6b17a2 100644 --- a/drivers/pci/hotplug/pciehp_hpc.c +++ b/drivers/pci/hotplug/pciehp_hpc.c @@ -832,9 +832,8 @@ static inline void dbg_ctrl(struct controller *ctrl)  	for (i = 0; i < DEVICE_COUNT_RESOURCE; i++) {  		if (!pci_resource_len(pdev, i))  			continue; -		ctrl_info(ctrl, "  PCI resource [%d]     : 0x%llx@0x%llx\n", -			  i, (unsigned long long)pci_resource_len(pdev, i), -			  (unsigned long long)pci_resource_start(pdev, i)); +		ctrl_info(ctrl, "  PCI resource [%d]     : %pR\n", +			  i, &pdev->resource[i]);  	}  	ctrl_info(ctrl, "Slot Capabilities      : 0x%08x\n", ctrl->slot_cap);  	ctrl_info(ctrl, "  Physical Slot Number : %d\n", PSN(ctrl)); diff --git a/drivers/pci/ioapic.c b/drivers/pci/ioapic.c index 3e0d7b5dd1b..fb9fdf4a42b 100644 --- a/drivers/pci/ioapic.c +++ b/drivers/pci/ioapic.c @@ -31,9 +31,9 @@ static int ioapic_probe(struct pci_dev *dev, const struct pci_device_id *ent)  	acpi_status status;  	unsigned long long gsb;  	struct ioapic *ioapic; -	u64 addr;  	int ret;  	char *type; +	struct resource *res;  	handle = DEVICE_ACPI_HANDLE(&dev->dev);  	if (!handle) @@ -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))  		goto exit_disable; -	addr = pci_resource_start(dev, 0); -	if (acpi_register_ioapic(ioapic->handle, addr, ioapic->gsi_base)) +	res = &dev->resource[0]; +	if (acpi_register_ioapic(ioapic->handle, res->start, ioapic->gsi_base))  		goto exit_release;  	pci_set_drvdata(dev, ioapic); -	dev_info(&dev->dev, "%s at %#llx, GSI %u\n", type, addr, -		 ioapic->gsi_base); +	dev_info(&dev->dev, "%s at %pR, GSI %u\n", type, res, ioapic->gsi_base);  	return 0;  exit_release: diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index cb1dd5f4988..1531f3a4987 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c @@ -2576,18 +2576,17 @@ EXPORT_SYMBOL_GPL(pci_reset_function);   */  int pcix_get_max_mmrbc(struct pci_dev *dev)  { -	int err, cap; +	int cap;  	u32 stat;  	cap = pci_find_capability(dev, PCI_CAP_ID_PCIX);  	if (!cap)  		return -EINVAL; -	err = pci_read_config_dword(dev, cap + PCI_X_STATUS, &stat); -	if (err) +	if (pci_read_config_dword(dev, cap + PCI_X_STATUS, &stat))  		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); @@ -2600,18 +2599,17 @@ EXPORT_SYMBOL(pcix_get_max_mmrbc);   */  int pcix_get_mmrbc(struct pci_dev *dev)  { -	int ret, cap; -	u32 cmd; +	int cap; +	u16 cmd;  	cap = pci_find_capability(dev, PCI_CAP_ID_PCIX);  	if (!cap)  		return -EINVAL; -	ret = pci_read_config_dword(dev, cap + PCI_X_CMD, &cmd); -	if (!ret) -		ret = 512 << ((cmd & PCI_X_CMD_MAX_READ) >> 2); +	if (pci_read_config_word(dev, cap + PCI_X_CMD, &cmd)) +		return -EINVAL; -	return ret; +	return 512 << ((cmd & PCI_X_CMD_MAX_READ) >> 2);  }  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 cap, err = -EINVAL; -	u32 stat, cmd, v, o; +	int cap; +	u32 stat, v, o; +	u16 cmd;  	if (mmrbc < 512 || mmrbc > 4096 || !is_power_of_2(mmrbc)) -		goto out; +		return -EINVAL;  	v = ffs(mmrbc) - 10;  	cap = pci_find_capability(dev, PCI_CAP_ID_PCIX);  	if (!cap) -		goto out; +		return -EINVAL; -	err = pci_read_config_dword(dev, cap + PCI_X_STATUS, &stat); -	if (err) -		goto out; +	if (pci_read_config_dword(dev, cap + PCI_X_STATUS, &stat)) +		return -EINVAL;  	if (v > (stat & PCI_X_STATUS_MAX_READ) >> 21)  		return -E2BIG; -	err = pci_read_config_dword(dev, cap + PCI_X_CMD, &cmd); -	if (err) -		goto out; +	if (pci_read_config_word(dev, cap + PCI_X_CMD, &cmd)) +		return -EINVAL;  	o = (cmd & PCI_X_CMD_MAX_READ) >> 2;  	if (o != v) { @@ -2657,10 +2654,10 @@ int pcix_set_mmrbc(struct pci_dev *dev, int mmrbc)  		cmd &= ~PCI_X_CMD_MAX_READ;  		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 err; +	return 0;  }  EXPORT_SYMBOL(pcix_set_mmrbc); @@ -3023,7 +3020,6 @@ EXPORT_SYMBOL(pcim_pin_device);  EXPORT_SYMBOL(pci_disable_device);  EXPORT_SYMBOL(pci_find_capability);  EXPORT_SYMBOL(pci_bus_find_capability); -EXPORT_SYMBOL(pci_register_set_vga_state);  EXPORT_SYMBOL(pci_release_regions);  EXPORT_SYMBOL(pci_request_regions);  EXPORT_SYMBOL(pci_request_regions_exclusive); diff --git a/drivers/pci/probe.c b/drivers/pci/probe.c index 2a943090a3b..882bd8d29fe 100644 --- a/drivers/pci/probe.c +++ b/drivers/pci/probe.c @@ -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_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. -	 * If the BAR isn't implemented, all bits must be 0.  If it's a -	 * memory BAR or a ROM, bit 0 must be clear; if it's an io BAR, bit -	 * 1 must be clear. +	 * If it's a memory BAR or a ROM, bit 0 must be clear; if it's +	 * an io BAR, bit 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; +	}  	/*  	 * 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,  				   pos, res);  		}  	} 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; +		}  		res->start = l; -		res->end = l + sz; +		res->end = l + size;  		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)  		dev_printk(KERN_DEBUG, &dev->dev, "  bridge window %pR\n", res);  	} else {  		dev_printk(KERN_DEBUG, &dev->dev, -			 "  bridge window [io  %04lx - %04lx] reg reading\n", +			 "  bridge window [io  %#06lx-%#06lx] (disabled)\n",  				 base, limit);  	}  } @@ -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);  	} else {  		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);  	}  } @@ -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);  	} else {  		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);  	}  } @@ -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);  	u32 buses, i, j = 0;  	u16 bctl; +	u8 primary, secondary, subordinate;  	int broken = 0;  	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", -		buses & 0xffffff, pass); +	dev_dbg(&dev->dev, "scanning [bus %02x-%02x] behind bridge, pass %d\n", +		secondary, subordinate, pass);  	/* Check if setup is sensible at all */  	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");  		broken = 1;  	} @@ -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,  			      bctl & ~PCI_BRIDGE_CTL_MASTER_ABORT); -	if ((buses & 0xffff00) && !pcibios_assign_all_busses() && !is_cardbus && !broken) { -		unsigned int cmax, busnr; +	if ((secondary || subordinate) && !pcibios_assign_all_busses() && +	    !is_cardbus && !broken) { +		unsigned int cmax;  		/*  		 * Bus already configured by firmware, process it in the first  		 * pass and just note the configuration.  		 */  		if (pass)  			goto out; -		busnr = (buses >> 8) & 0xFF;  		/*  		 * 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,  		 * However, we continue to descend down the hierarchy and  		 * scan remaining child buses.  		 */ -		child = pci_find_bus(pci_domain_nr(bus), busnr); +		child = pci_find_bus(pci_domain_nr(bus), secondary);  		if (!child) { -			child = pci_add_new_bus(bus, dev, busnr); +			child = pci_add_new_bus(bus, dev, secondary);  			if (!child)  				goto out; -			child->primary = buses & 0xFF; -			child->subordinate = (buses >> 16) & 0xFF; +			child->primary = primary; +			child->subordinate = subordinate;  			child->bridge_ctl = bctl;  		} diff --git a/drivers/pci/quirks.c b/drivers/pci/quirks.c index 81d19d5683a..3ea0b29c010 100644 --- a/drivers/pci/quirks.c +++ b/drivers/pci/quirks.c @@ -368,8 +368,9 @@ static void __devinit quirk_io_region(struct pci_dev *dev, unsigned region,  		bus_region.end = res->end;  		pcibios_bus_to_resource(dev, res, &bus_region); -		pci_claim_resource(dev, nr); -		dev_info(&dev->dev, "quirk: %pR claimed by %s\n", res, name); +		if (pci_claim_resource(dev, nr) == 0) +			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)  	/*  	 * Disable PCI Bus Parking and PCI Master read caching on CX700  	 * which causes unspecified timing errors with a VT6212L on the PCI -	 * bus leading to USB2.0 packet loss. The defaults are that these -	 * features are turned off but some BIOSes turn them on. +	 * bus leading to USB2.0 packet loss. +	 * +	 * 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; + +	/* 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 (b & 0x40) {  			/* Turn off PCI Bus Parking */ @@ -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 @@ -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_VIA, 0xa238, quirk_disable_msi);  /* Go through the list of Hypertransport capabilities and   * return 1 if a HT MSI capability is found and enabled */ @@ -2479,6 +2495,39 @@ DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATI, 0x4374,  DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_ATI, 0x4375,  			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 */  #ifdef CONFIG_PCI_IOV diff --git a/drivers/pci/setup-res.c b/drivers/pci/setup-res.c index 7d678bb15ff..17bed18d24a 100644 --- a/drivers/pci/setup-res.c +++ b/drivers/pci/setup-res.c @@ -93,8 +93,7 @@ void pci_update_resource(struct pci_dev *dev, int resno)  int pci_claim_resource(struct pci_dev *dev, int resource)  {  	struct resource *res = &dev->resource[resource]; -	struct resource *root; -	int err; +	struct resource *root, *conflict;  	root = pci_find_parent_resource(dev, res);  	if (!root) { @@ -103,12 +102,15 @@ int pci_claim_resource(struct pci_dev *dev, int resource)  		return -EINVAL;  	} -	err = request_resource(root, res); -	if (err) +	conflict = request_resource_conflict(root, res); +	if (conflict) {  		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); diff --git a/drivers/pcmcia/rsrc_nonstatic.c b/drivers/pcmcia/rsrc_nonstatic.c index dcc602134d9..2e47991eccf 100644 --- a/drivers/pcmcia/rsrc_nonstatic.c +++ b/drivers/pcmcia/rsrc_nonstatic.c @@ -874,10 +874,8 @@ static int nonstatic_autoadd_resources(struct pcmcia_socket *s)  			if (res == &ioport_resource)  				continue;  			dev_printk(KERN_INFO, &s->cb_dev->dev, -				   "pcmcia: parent PCI bridge I/O " -				   "window: 0x%llx - 0x%llx\n", -				   (unsigned long long)res->start, -				   (unsigned long long)res->end); +				   "pcmcia: parent PCI bridge window: %pR\n", +				   res);  			if (!adjust_io(s, ADD_MANAGED_RESOURCE, res->start, res->end))  				done |= IORESOURCE_IO; @@ -887,10 +885,8 @@ static int nonstatic_autoadd_resources(struct pcmcia_socket *s)  			if (res == &iomem_resource)  				continue;  			dev_printk(KERN_INFO, &s->cb_dev->dev, -				   "pcmcia: parent PCI bridge Memory " -				   "window: 0x%llx - 0x%llx\n", -				   (unsigned long long)res->start, -				   (unsigned long long)res->end); +				   "pcmcia: parent PCI bridge window: %pR\n", +				   res);  			if (!adjust_memory(s, ADD_MANAGED_RESOURCE, res->start, res->end))  				done |= IORESOURCE_MEM;  		} diff --git a/include/linux/ioport.h b/include/linux/ioport.h index 71ab79da7e7..26fad187d66 100644 --- a/include/linux/ioport.h +++ b/include/linux/ioport.h @@ -112,12 +112,14 @@ struct resource_list {  extern struct resource ioport_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 release_resource(struct resource *new);  void release_child_resources(struct resource *new);  extern void reserve_region_with_split(struct resource *root,  			     resource_size_t start, resource_size_t end,  			     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 void insert_resource_expand_to_fit(struct resource *root, struct resource *new);  extern int allocate_resource(struct resource *root, struct resource *new, diff --git a/kernel/resource.c b/kernel/resource.c index 2d5be5d9bf5..9c358e26353 100644 --- a/kernel/resource.c +++ b/kernel/resource.c @@ -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   * @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;  	write_lock(&resource_lock);  	conflict = __request_resource(root, new);  	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;  } @@ -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   * @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   * entirely fit within the range of the new resource, then the new   * resource is inserted and the conflicting resources become children of   * 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;  	write_lock(&resource_lock);  	conflict = __insert_resource(parent, new);  	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;  }  |