forked from Minki/linux
PCI/IA64: embed pci hostbridge resources into pci_root_info
Currently, pcibios_resource_to_bus() and pcibios_bus_to_resource() functions use pci_host_bridge to translate bus side address from/to cpu side address. The pci_window in pci_controller never be used again. So we remove pci_window in pci_controller and embed hostbridge resource into pci_root_info. Bjorn suggested to implement hostbridge resources release in IA64 like in X86, this patch is to prepare for that. Signed-off-by: Yijing Wang <wangyijing@huawei.com> Signed-off-by: Jiang Liu <jiang.liu@huawei.com> Cc: Fenghua Yu <fenghua.yu@intel.com> Cc: Yinghai Lu <yinghai@kernel.org> Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org> Cc: Thierry Reding <thierry.reding@avionic-design.de> Cc: linux-ia64@vger.kernel.org Signed-off-by: Tony Luck <tony.luck@intel.com>
This commit is contained in:
parent
2ead66b547
commit
5cd7595dea
@ -89,23 +89,16 @@ extern int pci_mmap_legacy_page_range(struct pci_bus *bus,
|
|||||||
#define pci_legacy_read platform_pci_legacy_read
|
#define pci_legacy_read platform_pci_legacy_read
|
||||||
#define pci_legacy_write platform_pci_legacy_write
|
#define pci_legacy_write platform_pci_legacy_write
|
||||||
|
|
||||||
struct pci_window {
|
|
||||||
struct resource resource;
|
|
||||||
u64 offset;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct pci_controller {
|
struct pci_controller {
|
||||||
void *acpi_handle;
|
void *acpi_handle;
|
||||||
void *iommu;
|
void *iommu;
|
||||||
int segment;
|
int segment;
|
||||||
int node; /* nearest node with memory or -1 for global allocation */
|
int node; /* nearest node with memory or -1 for global allocation */
|
||||||
|
|
||||||
unsigned int windows;
|
|
||||||
struct pci_window *window;
|
|
||||||
|
|
||||||
void *platform_data;
|
void *platform_data;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
#define PCI_CONTROLLER(busdev) ((struct pci_controller *) busdev->sysdata)
|
#define PCI_CONTROLLER(busdev) ((struct pci_controller *) busdev->sysdata)
|
||||||
#define pci_domain_nr(busdev) (PCI_CONTROLLER(busdev)->segment)
|
#define pci_domain_nr(busdev) (PCI_CONTROLLER(busdev)->segment)
|
||||||
|
|
||||||
|
@ -134,6 +134,9 @@ struct pci_root_info {
|
|||||||
struct acpi_device *bridge;
|
struct acpi_device *bridge;
|
||||||
struct pci_controller *controller;
|
struct pci_controller *controller;
|
||||||
struct list_head resources;
|
struct list_head resources;
|
||||||
|
struct resource *res;
|
||||||
|
resource_size_t *res_offset;
|
||||||
|
unsigned int res_num;
|
||||||
char *name;
|
char *name;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -265,7 +268,7 @@ static acpi_status count_window(struct acpi_resource *resource, void *data)
|
|||||||
static acpi_status add_window(struct acpi_resource *res, void *data)
|
static acpi_status add_window(struct acpi_resource *res, void *data)
|
||||||
{
|
{
|
||||||
struct pci_root_info *info = data;
|
struct pci_root_info *info = data;
|
||||||
struct pci_window *window;
|
struct resource *resource;
|
||||||
struct acpi_resource_address64 addr;
|
struct acpi_resource_address64 addr;
|
||||||
acpi_status status;
|
acpi_status status;
|
||||||
unsigned long flags, offset = 0;
|
unsigned long flags, offset = 0;
|
||||||
@ -289,37 +292,36 @@ static acpi_status add_window(struct acpi_resource *res, void *data)
|
|||||||
} else
|
} else
|
||||||
return AE_OK;
|
return AE_OK;
|
||||||
|
|
||||||
window = &info->controller->window[info->controller->windows++];
|
resource = &info->res[info->res_num];
|
||||||
window->resource.name = info->name;
|
resource->name = info->name;
|
||||||
window->resource.flags = flags;
|
resource->flags = flags;
|
||||||
window->resource.start = addr.minimum + offset;
|
resource->start = addr.minimum + offset;
|
||||||
window->resource.end = window->resource.start + addr.address_length - 1;
|
resource->end = resource->start + addr.address_length - 1;
|
||||||
window->offset = offset;
|
info->res_offset[info->res_num] = offset;
|
||||||
|
|
||||||
if (insert_resource(root, &window->resource)) {
|
if (insert_resource(root, resource)) {
|
||||||
dev_err(&info->bridge->dev,
|
dev_err(&info->bridge->dev,
|
||||||
"can't allocate host bridge window %pR\n",
|
"can't allocate host bridge window %pR\n",
|
||||||
&window->resource);
|
resource);
|
||||||
} else {
|
} else {
|
||||||
if (offset)
|
if (offset)
|
||||||
dev_info(&info->bridge->dev, "host bridge window %pR "
|
dev_info(&info->bridge->dev, "host bridge window %pR "
|
||||||
"(PCI address [%#llx-%#llx])\n",
|
"(PCI address [%#llx-%#llx])\n",
|
||||||
&window->resource,
|
resource,
|
||||||
window->resource.start - offset,
|
resource->start - offset,
|
||||||
window->resource.end - offset);
|
resource->end - offset);
|
||||||
else
|
else
|
||||||
dev_info(&info->bridge->dev,
|
dev_info(&info->bridge->dev,
|
||||||
"host bridge window %pR\n",
|
"host bridge window %pR\n", resource);
|
||||||
&window->resource);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* HP's firmware has a hack to work around a Windows bug.
|
/* HP's firmware has a hack to work around a Windows bug.
|
||||||
* Ignore these tiny memory ranges */
|
* Ignore these tiny memory ranges */
|
||||||
if (!((window->resource.flags & IORESOURCE_MEM) &&
|
if (!((resource->flags & IORESOURCE_MEM) &&
|
||||||
(window->resource.end - window->resource.start < 16)))
|
(resource->end - resource->start < 16)))
|
||||||
pci_add_resource_offset(&info->resources, &window->resource,
|
pci_add_resource_offset(&info->resources, resource,
|
||||||
window->offset);
|
info->res_offset[info->res_num]);
|
||||||
|
|
||||||
|
info->res_num++;
|
||||||
return AE_OK;
|
return AE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -329,7 +331,6 @@ struct pci_bus *pci_acpi_scan_root(struct acpi_pci_root *root)
|
|||||||
int domain = root->segment;
|
int domain = root->segment;
|
||||||
int bus = root->secondary.start;
|
int bus = root->secondary.start;
|
||||||
struct pci_controller *controller;
|
struct pci_controller *controller;
|
||||||
unsigned int windows = 0;
|
|
||||||
struct pci_root_info info;
|
struct pci_root_info info;
|
||||||
struct pci_bus *pbus;
|
struct pci_bus *pbus;
|
||||||
char *name;
|
char *name;
|
||||||
@ -351,22 +352,29 @@ struct pci_bus *pci_acpi_scan_root(struct acpi_pci_root *root)
|
|||||||
/* insert busn resource at first */
|
/* insert busn resource at first */
|
||||||
pci_add_resource(&info.resources, &root->secondary);
|
pci_add_resource(&info.resources, &root->secondary);
|
||||||
acpi_walk_resources(device->handle, METHOD_NAME__CRS, count_window,
|
acpi_walk_resources(device->handle, METHOD_NAME__CRS, count_window,
|
||||||
&windows);
|
&info.res_num);
|
||||||
if (windows) {
|
if (info.res_num) {
|
||||||
controller->window =
|
info.res =
|
||||||
kzalloc_node(sizeof(*controller->window) * windows,
|
kzalloc_node(sizeof(*info.res) * info.res_num,
|
||||||
GFP_KERNEL, controller->node);
|
GFP_KERNEL, controller->node);
|
||||||
if (!controller->window)
|
if (!info.res)
|
||||||
goto out2;
|
goto out2;
|
||||||
|
|
||||||
|
info.res_offset =
|
||||||
|
kzalloc_node(sizeof(*info.res_offset) * info.res_num,
|
||||||
|
GFP_KERNEL, controller->node);
|
||||||
|
if (!info.res_offset)
|
||||||
|
goto out3;
|
||||||
|
|
||||||
name = kmalloc(16, GFP_KERNEL);
|
name = kmalloc(16, GFP_KERNEL);
|
||||||
if (!name)
|
if (!name)
|
||||||
goto out3;
|
goto out4;
|
||||||
|
|
||||||
sprintf(name, "PCI Bus %04x:%02x", domain, bus);
|
sprintf(name, "PCI Bus %04x:%02x", domain, bus);
|
||||||
info.bridge = device;
|
info.bridge = device;
|
||||||
info.controller = controller;
|
info.controller = controller;
|
||||||
info.name = name;
|
info.name = name;
|
||||||
|
info.res_num = 0;
|
||||||
acpi_walk_resources(device->handle, METHOD_NAME__CRS,
|
acpi_walk_resources(device->handle, METHOD_NAME__CRS,
|
||||||
add_window, &info);
|
add_window, &info);
|
||||||
}
|
}
|
||||||
@ -385,9 +393,10 @@ struct pci_bus *pci_acpi_scan_root(struct acpi_pci_root *root)
|
|||||||
|
|
||||||
pci_scan_child_bus(pbus);
|
pci_scan_child_bus(pbus);
|
||||||
return pbus;
|
return pbus;
|
||||||
|
out4:
|
||||||
|
kfree(info.res_offset);
|
||||||
out3:
|
out3:
|
||||||
kfree(controller->window);
|
kfree(info.res);
|
||||||
out2:
|
out2:
|
||||||
kfree(controller);
|
kfree(controller);
|
||||||
out1:
|
out1:
|
||||||
|
Loading…
Reference in New Issue
Block a user