]> www.infradead.org Git - users/hch/dma-mapping.git/commitdiff
PM: domains: Fix alloc/free in dev_pm_domain_attach|detach_list()
authorUlf Hansson <ulf.hansson@linaro.org>
Wed, 2 Oct 2024 12:22:23 +0000 (14:22 +0200)
committerUlf Hansson <ulf.hansson@linaro.org>
Thu, 10 Oct 2024 11:55:17 +0000 (13:55 +0200)
The dev_pm_domain_attach|detach_list() functions are not resource managed,
hence they should not use devm_* helpers to manage allocation/freeing of
data. Let's fix this by converting to the traditional alloc/free functions.

Fixes: 161e16a5e50a ("PM: domains: Add helper functions to attach/detach multiple PM domains")
Cc: stable@vger.kernel.org
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
Acked-by: Viresh Kumar <viresh.kumar@linaro.org>
Link: https://lore.kernel.org/r/20241002122232.194245-3-ulf.hansson@linaro.org
drivers/base/power/common.c

index 8c34ae1cd8d554d5e9e8deb86a2d49688d82ad22..cca2fd0a1aed64b368c8c8e33ec4a8ad554b37af 100644 (file)
@@ -195,6 +195,7 @@ int dev_pm_domain_attach_list(struct device *dev,
        struct device *pd_dev = NULL;
        int ret, i, num_pds = 0;
        bool by_id = true;
+       size_t size;
        u32 pd_flags = data ? data->pd_flags : 0;
        u32 link_flags = pd_flags & PD_FLAG_NO_DEV_LINK ? 0 :
                        DL_FLAG_STATELESS | DL_FLAG_PM_RUNTIME;
@@ -217,19 +218,17 @@ int dev_pm_domain_attach_list(struct device *dev,
        if (num_pds <= 0)
                return 0;
 
-       pds = devm_kzalloc(dev, sizeof(*pds), GFP_KERNEL);
+       pds = kzalloc(sizeof(*pds), GFP_KERNEL);
        if (!pds)
                return -ENOMEM;
 
-       pds->pd_devs = devm_kcalloc(dev, num_pds, sizeof(*pds->pd_devs),
-                                   GFP_KERNEL);
-       if (!pds->pd_devs)
-               return -ENOMEM;
-
-       pds->pd_links = devm_kcalloc(dev, num_pds, sizeof(*pds->pd_links),
-                                    GFP_KERNEL);
-       if (!pds->pd_links)
-               return -ENOMEM;
+       size = sizeof(*pds->pd_devs) + sizeof(*pds->pd_links);
+       pds->pd_devs = kcalloc(num_pds, size, GFP_KERNEL);
+       if (!pds->pd_devs) {
+               ret = -ENOMEM;
+               goto free_pds;
+       }
+       pds->pd_links = (void *)(pds->pd_devs + num_pds);
 
        if (link_flags && pd_flags & PD_FLAG_DEV_LINK_ON)
                link_flags |= DL_FLAG_RPM_ACTIVE;
@@ -272,6 +271,9 @@ err_attach:
                        device_link_del(pds->pd_links[i]);
                dev_pm_domain_detach(pds->pd_devs[i], true);
        }
+       kfree(pds->pd_devs);
+free_pds:
+       kfree(pds);
        return ret;
 }
 EXPORT_SYMBOL_GPL(dev_pm_domain_attach_list);
@@ -363,6 +365,9 @@ void dev_pm_domain_detach_list(struct dev_pm_domain_list *list)
                        device_link_del(list->pd_links[i]);
                dev_pm_domain_detach(list->pd_devs[i], true);
        }
+
+       kfree(list->pd_devs);
+       kfree(list);
 }
 EXPORT_SYMBOL_GPL(dev_pm_domain_detach_list);