diff options
author | Milian Wolff <milian.wolff@kdab.com> | 2017-05-24 02:21:24 -0400 |
---|---|---|
committer | Ingo Molnar <mingo@kernel.org> | 2017-05-24 02:41:48 -0400 |
commit | b21cc97810932a551f7aac46f0b89c469c828b3f (patch) | |
tree | 60683bc954f18bf94e76ecff9fad7410a5081f72 | |
parent | 7d4df089d77306914426a604c890175f91a9a459 (diff) |
perf report: Fix memory leak in addr2line when called by addr2inlines
When a filename was found in addr2line it was duplicated via strdup()
but never freed. Now we pass NULL and handle this gracefully in
addr2line.
Detected by Valgrind:
==16331== 1,680 bytes in 21 blocks are definitely lost in loss record 148 of 220
==16331== at 0x4C2AF1F: malloc (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
==16331== by 0x672FA69: strdup (in /usr/lib/libc-2.25.so)
==16331== by 0x52769F: addr2line (srcline.c:256)
==16331== by 0x52769F: addr2inlines (srcline.c:294)
==16331== by 0x52769F: dso__parse_addr_inlines (srcline.c:502)
==16331== by 0x574D7A: inline__fprintf (hist.c:41)
==16331== by 0x574D7A: ipchain__fprintf_graph (hist.c:147)
==16331== by 0x57518A: __callchain__fprintf_graph (hist.c:212)
==16331== by 0x5753CF: callchain__fprintf_graph.constprop.6 (hist.c:337)
==16331== by 0x57738E: hist_entry__fprintf (hist.c:628)
==16331== by 0x57738E: hists__fprintf (hist.c:882)
==16331== by 0x44A20F: perf_evlist__tty_browse_hists (builtin-report.c:399)
==16331== by 0x44A20F: report__browse_hists (builtin-report.c:491)
==16331== by 0x44A20F: __cmd_report (builtin-report.c:624)
==16331== by 0x44A20F: cmd_report (builtin-report.c:1054)
==16331== by 0x4A49CE: run_builtin (perf.c:296)
==16331== by 0x4A4CC0: handle_internal_command (perf.c:348)
==16331== by 0x434371: run_argv (perf.c:392)
==16331== by 0x434371: main (perf.c:530)
Signed-off-by: Milian Wolff <milian.wolff@kdab.com>
Signed-off-by: Namhyung Kim <namhyung@kernel.org>
Cc: Arnaldo Carvalho de Melo <acme@kernel.org>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: David Ahern <dsahern@gmail.com>
Cc: Jiri Olsa <jolsa@kernel.org>
Cc: Jiri Olsa <jolsa@redhat.com>
Cc: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Yao Jin <yao.jin@linux.intel.com>
Cc: kernel-team@lge.com
Link: http://lkml.kernel.org/r/20170524062129.32529-3-namhyung@kernel.org
Signed-off-by: Ingo Molnar <mingo@kernel.org>
-rw-r--r-- | tools/perf/util/srcline.c | 23 |
1 files changed, 13 insertions, 10 deletions
diff --git a/tools/perf/util/srcline.c b/tools/perf/util/srcline.c index df051a52393c..5e376d64d59e 100644 --- a/tools/perf/util/srcline.c +++ b/tools/perf/util/srcline.c | |||
@@ -230,7 +230,10 @@ static int addr2line(const char *dso_name, u64 addr, | |||
230 | 230 | ||
231 | bfd_map_over_sections(a2l->abfd, find_address_in_section, a2l); | 231 | bfd_map_over_sections(a2l->abfd, find_address_in_section, a2l); |
232 | 232 | ||
233 | if (a2l->found && unwind_inlines) { | 233 | if (!a2l->found) |
234 | return 0; | ||
235 | |||
236 | if (unwind_inlines) { | ||
234 | int cnt = 0; | 237 | int cnt = 0; |
235 | 238 | ||
236 | while (bfd_find_inliner_info(a2l->abfd, &a2l->filename, | 239 | while (bfd_find_inliner_info(a2l->abfd, &a2l->filename, |
@@ -243,6 +246,8 @@ static int addr2line(const char *dso_name, u64 addr, | |||
243 | a2l->line, node, | 246 | a2l->line, node, |
244 | dso) != 0) | 247 | dso) != 0) |
245 | return 0; | 248 | return 0; |
249 | // found at least one inline frame | ||
250 | ret = 1; | ||
246 | } | 251 | } |
247 | } | 252 | } |
248 | 253 | ||
@@ -252,14 +257,14 @@ static int addr2line(const char *dso_name, u64 addr, | |||
252 | } | 257 | } |
253 | } | 258 | } |
254 | 259 | ||
255 | if (a2l->found && a2l->filename) { | 260 | if (file) { |
256 | *file = strdup(a2l->filename); | 261 | *file = a2l->filename ? strdup(a2l->filename) : NULL; |
257 | *line = a2l->line; | 262 | ret = *file ? 1 : 0; |
258 | |||
259 | if (*file) | ||
260 | ret = 1; | ||
261 | } | 263 | } |
262 | 264 | ||
265 | if (line) | ||
266 | *line = a2l->line; | ||
267 | |||
263 | return ret; | 268 | return ret; |
264 | } | 269 | } |
265 | 270 | ||
@@ -278,8 +283,6 @@ void dso__free_a2l(struct dso *dso) | |||
278 | static struct inline_node *addr2inlines(const char *dso_name, u64 addr, | 283 | static struct inline_node *addr2inlines(const char *dso_name, u64 addr, |
279 | struct dso *dso) | 284 | struct dso *dso) |
280 | { | 285 | { |
281 | char *file = NULL; | ||
282 | unsigned int line = 0; | ||
283 | struct inline_node *node; | 286 | struct inline_node *node; |
284 | 287 | ||
285 | node = zalloc(sizeof(*node)); | 288 | node = zalloc(sizeof(*node)); |
@@ -291,7 +294,7 @@ static struct inline_node *addr2inlines(const char *dso_name, u64 addr, | |||
291 | INIT_LIST_HEAD(&node->val); | 294 | INIT_LIST_HEAD(&node->val); |
292 | node->addr = addr; | 295 | node->addr = addr; |
293 | 296 | ||
294 | if (!addr2line(dso_name, addr, &file, &line, dso, TRUE, node)) | 297 | if (!addr2line(dso_name, addr, NULL, NULL, dso, TRUE, node)) |
295 | goto out_free_inline_node; | 298 | goto out_free_inline_node; |
296 | 299 | ||
297 | if (list_empty(&node->val)) | 300 | if (list_empty(&node->val)) |