summaryrefslogtreecommitdiffstats
path: root/lib/libproc
diff options
context:
space:
mode:
authorrpaulo <rpaulo@FreeBSD.org>2012-09-07 02:38:07 +0000
committerrpaulo <rpaulo@FreeBSD.org>2012-09-07 02:38:07 +0000
commitf85f236da92356ecc56e0a580df735dffaa9ee00 (patch)
treebd37abd3895967676ad26b5ecdf5371f2f2e7459 /lib/libproc
parent55f61bbb9b449d690b864b0a940d1853c77bf70d (diff)
downloadFreeBSD-src-f85f236da92356ecc56e0a580df735dffaa9ee00.zip
FreeBSD-src-f85f236da92356ecc56e0a580df735dffaa9ee00.tar.gz
When calling the C++ demangler, make sure to free the returned buffer,
which might have been reallocated. Pointed out by: stefanf
Diffstat (limited to 'lib/libproc')
-rw-r--r--lib/libproc/proc_sym.c30
1 files changed, 24 insertions, 6 deletions
diff --git a/lib/libproc/proc_sym.c b/lib/libproc/proc_sym.c
index 0940df8..4bef7f0 100644
--- a/lib/libproc/proc_sym.c
+++ b/lib/libproc/proc_sym.c
@@ -51,6 +51,26 @@ extern char *__cxa_demangle(const char *, char *, size_t *, int *);
static void proc_rdl2prmap(rd_loadobj_t *, prmap_t *);
static void
+demangle(const char *symbol, char *buf, size_t len)
+{
+ char *dembuf;
+ size_t demlen = len;
+
+ dembuf = malloc(len);
+ if (!dembuf)
+ goto fail;
+ dembuf = __cxa_demangle(symbol, dembuf, &demlen, NULL);
+ if (!dembuf)
+ goto fail;
+ strlcpy(buf, dembuf, len);
+ free(dembuf);
+
+ return;
+fail:
+ strlcpy(buf, symbol, len);
+}
+
+static void
proc_rdl2prmap(rd_loadobj_t *rdl, prmap_t *map)
{
map->pr_vaddr = rdl->rdl_saddr;
@@ -268,9 +288,8 @@ proc_addr2sym(struct proc_handle *p, uintptr_t addr, char *name,
if (addr >= rsym && addr <= (rsym + sym.st_size)) {
s = elf_strptr(e, dynsymstridx, sym.st_name);
if (s) {
- if (strlen(s) > 2 &&
- s[0] == '_' && s[1] == 'Z')
- __cxa_demangle(s, name, &namesz, NULL);
+ if (s[0] == '_' && s[1] == 'Z' && s[2])
+ demangle(s, name, namesz);
else
strlcpy(name, s, namesz);
memcpy(symcopy, &sym, sizeof(sym));
@@ -308,9 +327,8 @@ symtab:
if (addr >= rsym && addr <= (rsym + sym.st_size)) {
s = elf_strptr(e, symtabstridx, sym.st_name);
if (s) {
- if (strlen(s) > 2 &&
- s[0] == '_' && s[1] == 'Z')
- __cxa_demangle(s, name, &namesz, NULL);
+ if (s[0] == '_' && s[1] == 'Z' && s[2])
+ demangle(s, name, namesz);
else
strlcpy(name, s, namesz);
memcpy(symcopy, &sym, sizeof(sym));
OpenPOWER on IntegriCloud