perf sched: Fix memory leak in perf_sched__map()
authorYang Jihong <yangjihong1@huawei.com>
Tue, 6 Feb 2024 08:32:25 +0000 (08:32 +0000)
committerNamhyung Kim <namhyung@kernel.org>
Fri, 9 Feb 2024 22:08:41 +0000 (14:08 -0800)
perf_sched__map() needs to free memory of map_cpus, color_pids and
color_cpus in normal path and rollback allocated memory in error path.

Signed-off-by: Yang Jihong <yangjihong1@huawei.com>
Signed-off-by: Namhyung Kim <namhyung@kernel.org>
Link: https://lore.kernel.org/r/20240206083228.172607-3-yangjihong1@huawei.com
tools/perf/builtin-sched.c

index 08dec557e6be4a744fcd8ef073222b311cddb272..1513e87ce4f22334510aae15c9c3d4bd87cb7657 100644 (file)
@@ -3208,8 +3208,6 @@ static int perf_sched__lat(struct perf_sched *sched)
 
 static int setup_map_cpus(struct perf_sched *sched)
 {
-       struct perf_cpu_map *map;
-
        sched->max_cpu.cpu  = sysconf(_SC_NPROCESSORS_CONF);
 
        if (sched->map.comp) {
@@ -3218,16 +3216,15 @@ static int setup_map_cpus(struct perf_sched *sched)
                        return -1;
        }
 
-       if (!sched->map.cpus_str)
-               return 0;
-
-       map = perf_cpu_map__new(sched->map.cpus_str);
-       if (!map) {
-               pr_err("failed to get cpus map from %s\n", sched->map.cpus_str);
-               return -1;
+       if (sched->map.cpus_str) {
+               sched->map.cpus = perf_cpu_map__new(sched->map.cpus_str);
+               if (!sched->map.cpus) {
+                       pr_err("failed to get cpus map from %s\n", sched->map.cpus_str);
+                       zfree(&sched->map.comp_cpus);
+                       return -1;
+               }
        }
 
-       sched->map.cpus = map;
        return 0;
 }
 
@@ -3267,20 +3264,34 @@ static int setup_color_cpus(struct perf_sched *sched)
 
 static int perf_sched__map(struct perf_sched *sched)
 {
+       int rc = -1;
+
        if (setup_map_cpus(sched))
-               return -1;
+               return rc;
 
        if (setup_color_pids(sched))
-               return -1;
+               goto out_put_map_cpus;
 
        if (setup_color_cpus(sched))
-               return -1;
+               goto out_put_color_pids;
 
        setup_pager();
        if (perf_sched__read_events(sched))
-               return -1;
+               goto out_put_color_cpus;
+
+       rc = 0;
        print_bad_events(sched);
-       return 0;
+
+out_put_color_cpus:
+       perf_cpu_map__put(sched->map.color_cpus);
+
+out_put_color_pids:
+       perf_thread_map__put(sched->map.color_pids);
+
+out_put_map_cpus:
+       zfree(&sched->map.comp_cpus);
+       perf_cpu_map__put(sched->map.cpus);
+       return rc;
 }
 
 static int perf_sched__replay(struct perf_sched *sched)