demangle.c 2.19 KB
Newer Older
1 2
/*
 * This file is part of ltrace.
3
 * Copyright (C) 2012,2013 Petr Machata, Red Hat Inc.
4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
 * Copyright (C) 1998,1999,2003,2004,2008,2009 Juan Cespedes
 * Copyright (C) 2006 Ian Wienand
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
 * published by the Free Software Foundation; either version 2 of the
 * License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA
 */

Juan Cespedes's avatar
Juan Cespedes committed
23 24 25
#include "config.h"

#include <string.h>
Juan Cespedes's avatar
Juan Cespedes committed
26 27
#include <stdlib.h>
#include <stdio.h>
Juan Cespedes's avatar
Juan Cespedes committed
28

29 30 31
#include "demangle.h"
#include "dict.h"
#include "debug.h"
Juan Cespedes's avatar
Juan Cespedes committed
32

Juan Cespedes's avatar
Juan Cespedes committed
33 34
#ifdef USE_DEMANGLE

Juan Cespedes's avatar
Juan Cespedes committed
35
/*****************************************************************************/
Juan Cespedes's avatar
Juan Cespedes committed
36

37
static struct dict *name_cache = NULL;
Juan Cespedes's avatar
Juan Cespedes committed
38

39 40
const char *
my_demangle(const char *function_name) {
41
#ifdef USE_CXA_DEMANGLE
42
	extern char *__cxa_demangle(const char *, char *, size_t *, int *);
Juan Cespedes's avatar
Juan Cespedes committed
43
#endif
Juan Cespedes's avatar
Juan Cespedes committed
44

45 46
	debug(DEBUG_FUNCTION, "my_demangle(name=%s)", function_name);

47 48 49
	if (name_cache == NULL) {
		name_cache = malloc(sizeof(*name_cache));
		if (name_cache != NULL)
50
			DICT_INIT(name_cache, char *, const char *,
51 52 53
				  dict_hash_string, dict_eq_string, NULL);
	}

54 55 56 57
	const char *tmp = NULL;
	if (name_cache != NULL
	    && DICT_FIND_VAL(name_cache, &function_name, &tmp) == 0)
		return tmp;
58

Juan Cespedes's avatar
Juan Cespedes committed
59
#ifdef HAVE_LIBIBERTY
60
	tmp = cplus_demangle(function_name,
61
					 DMGL_ANSI | DMGL_PARAMS);
62
#elif defined USE_CXA_DEMANGLE
63
	int status = 0;
64
	tmp = __cxa_demangle(function_name, NULL, NULL, &status);
Juan Cespedes's avatar
Juan Cespedes committed
65
#endif
66 67 68 69 70
	if (name_cache == NULL || tmp == NULL) {
	fail:
		if (tmp == NULL)
			return function_name;
		return tmp;
Juan Cespedes's avatar
Juan Cespedes committed
71
	}
72 73 74 75 76 77 78 79 80 81

	const char *fn_copy = strdup(function_name);
	if (fn_copy == NULL)
		goto fail;

	if (DICT_INSERT(name_cache, &fn_copy, &tmp) < 0) {
		free((char *)fn_copy);
		goto fail;
	}

Juan Cespedes's avatar
Juan Cespedes committed
82
	return tmp;
Juan Cespedes's avatar
Juan Cespedes committed
83 84 85
}

#endif