PCI: Rework of_pci_get_host_bridge_resources() to devm_of_pci_get_host_bridge_resources()
[linux-2.6-microblaze.git] / drivers / pci / of.c
index a28355c..d088c91 100644 (file)
@@ -244,8 +244,9 @@ EXPORT_SYMBOL_GPL(of_pci_check_probe_only);
 
 #if defined(CONFIG_OF_ADDRESS)
 /**
- * of_pci_get_host_bridge_resources - Parse PCI host bridge resources from DT
- * @dev: device node of the host bridge having the range property
+ * devm_of_pci_get_host_bridge_resources() - Resource-managed parsing of PCI
+ *                                           host bridge resources from DT
+ * @dev: host bridge device
  * @busno: bus number associated with the bridge root bus
  * @bus_max: maximum number of buses for this bridge
  * @resources: list where the range of resources will be added after DT parsing
@@ -253,8 +254,6 @@ EXPORT_SYMBOL_GPL(of_pci_check_probe_only);
  * address for the start of the I/O range. Can be NULL if the caller doesn't
  * expect I/O ranges to be present in the device tree.
  *
- * It is the caller's job to free the @resources list.
- *
  * This function will parse the "ranges" property of a PCI host bridge device
  * node and setup the resource mapping based on its content. It is expected
  * that the property conforms with the Power ePAPR document.
@@ -262,11 +261,11 @@ EXPORT_SYMBOL_GPL(of_pci_check_probe_only);
  * It returns zero if the range parsing has been successful or a standard error
  * value if it failed.
  */
-int of_pci_get_host_bridge_resources(struct device_node *dev,
+int devm_of_pci_get_host_bridge_resources(struct device *dev,
                        unsigned char busno, unsigned char bus_max,
                        struct list_head *resources, resource_size_t *io_base)
 {
-       struct resource_entry *window;
+       struct device_node *dev_node = dev->of_node;
        struct resource *res;
        struct resource *bus_range;
        struct of_pci_range range;
@@ -277,19 +276,19 @@ int of_pci_get_host_bridge_resources(struct device_node *dev,
        if (io_base)
                *io_base = (resource_size_t)OF_BAD_ADDR;
 
-       bus_range = kzalloc(sizeof(*bus_range), GFP_KERNEL);
+       bus_range = devm_kzalloc(dev, sizeof(*bus_range), GFP_KERNEL);
        if (!bus_range)
                return -ENOMEM;
 
-       pr_info("host bridge %pOF ranges:\n", dev);
+       dev_info(dev, "host bridge %pOF ranges:\n", dev_node);
 
-       err = of_pci_parse_bus_range(dev, bus_range);
+       err = of_pci_parse_bus_range(dev_node, bus_range);
        if (err) {
                bus_range->start = busno;
                bus_range->end = bus_max;
                bus_range->flags = IORESOURCE_BUS;
-               pr_info("  No bus range found for %pOF, using %pR\n",
-                       dev, bus_range);
+               dev_info(dev, "  No bus range found for %pOF, using %pR\n",
+                        dev_node, bus_range);
        } else {
                if (bus_range->end > bus_range->start + bus_max)
                        bus_range->end = bus_range->start + bus_max;
@@ -297,11 +296,11 @@ int of_pci_get_host_bridge_resources(struct device_node *dev,
        pci_add_resource(resources, bus_range);
 
        /* Check for ranges property */
-       err = of_pci_range_parser_init(&parser, dev);
+       err = of_pci_range_parser_init(&parser, dev_node);
        if (err)
-               goto parse_failed;
+               goto failed;
 
-       pr_debug("Parsing ranges property...\n");
+       dev_dbg(dev, "Parsing ranges property...\n");
        for_each_of_pci_range(&parser, &range) {
                /* Read next ranges element */
                if ((range.flags & IORESOURCE_TYPE_BITS) == IORESOURCE_IO)
@@ -310,9 +309,9 @@ int of_pci_get_host_bridge_resources(struct device_node *dev,
                        snprintf(range_type, 4, "MEM");
                else
                        snprintf(range_type, 4, "err");
-               pr_info("  %s %#010llx..%#010llx -> %#010llx\n", range_type,
-                       range.cpu_addr, range.cpu_addr + range.size - 1,
-                       range.pci_addr);
+               dev_info(dev, "  %s %#010llx..%#010llx -> %#010llx\n",
+                        range_type, range.cpu_addr,
+                        range.cpu_addr + range.size - 1, range.pci_addr);
 
                /*
                 * If we failed translation or got a zero-sized region
@@ -321,28 +320,28 @@ int of_pci_get_host_bridge_resources(struct device_node *dev,
                if (range.cpu_addr == OF_BAD_ADDR || range.size == 0)
                        continue;
 
-               res = kzalloc(sizeof(struct resource), GFP_KERNEL);
+               res = devm_kzalloc(dev, sizeof(struct resource), GFP_KERNEL);
                if (!res) {
                        err = -ENOMEM;
-                       goto parse_failed;
+                       goto failed;
                }
 
-               err = of_pci_range_to_resource(&range, dev, res);
+               err = of_pci_range_to_resource(&range, dev_node, res);
                if (err) {
-                       kfree(res);
+                       devm_kfree(dev, res);
                        continue;
                }
 
                if (resource_type(res) == IORESOURCE_IO) {
                        if (!io_base) {
-                               pr_err("I/O range found for %pOF. Please provide an io_base pointer to save CPU base address\n",
-                                       dev);
+                               dev_err(dev, "I/O range found for %pOF. Please provide an io_base pointer to save CPU base address\n",
+                                       dev_node);
                                err = -EINVAL;
-                               goto conversion_failed;
+                               goto failed;
                        }
                        if (*io_base != (resource_size_t)OF_BAD_ADDR)
-                               pr_warn("More than one I/O resource converted for %pOF. CPU base address for old range lost!\n",
-                                       dev);
+                               dev_warn(dev, "More than one I/O resource converted for %pOF. CPU base address for old range lost!\n",
+                                        dev_node);
                        *io_base = range.cpu_addr;
                }
 
@@ -351,15 +350,11 @@ int of_pci_get_host_bridge_resources(struct device_node *dev,
 
        return 0;
 
-conversion_failed:
-       kfree(res);
-parse_failed:
-       resource_list_for_each_entry(window, resources)
-               kfree(window->res);
+failed:
        pci_free_resource_list(resources);
        return err;
 }
-EXPORT_SYMBOL_GPL(of_pci_get_host_bridge_resources);
+EXPORT_SYMBOL_GPL(devm_of_pci_get_host_bridge_resources);
 #endif /* CONFIG_OF_ADDRESS */
 
 /**
@@ -599,12 +594,12 @@ int pci_parse_request_of_pci_ranges(struct device *dev,
                                    struct resource **bus_range)
 {
        int err, res_valid = 0;
-       struct device_node *np = dev->of_node;
        resource_size_t iobase;
        struct resource_entry *win, *tmp;
 
        INIT_LIST_HEAD(resources);
-       err = of_pci_get_host_bridge_resources(np, 0, 0xff, resources, &iobase);
+       err = devm_of_pci_get_host_bridge_resources(dev, 0, 0xff, resources,
+                                                   &iobase);
        if (err)
                return err;