/* * Copyright (C) 2000-2007 Paul Davis * Copyright (C) 2015 Tim Mayberry * * 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 Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include #include "pbd/demangle.h" #if defined(__GLIBCXX__) || defined(__APPLE__) #include #endif #ifdef __APPLE__ #include #endif std::string PBD::demangle_symbol (const std::string& mangled_symbol) { #if defined(__GLIBCXX__) || defined(__APPLE__) try { int status; char* realname = abi::__cxa_demangle (mangled_symbol.c_str(), 0, 0, &status); if (!status) { std::string demangled_symbol (realname); free (realname); return demangled_symbol; } return mangled_symbol; } catch (...) { /* may happen if realname == NULL */ } #endif /* Note: on win32, you can use UnDecorateSymbolName. See http://msdn.microsoft.com/en-us/library/ms681400%28VS.85%29.aspx See also: http://msdn.microsoft.com/en-us/library/ms680344%28VS.85%29.aspx */ return mangled_symbol; } std::string PBD::demangle (std::string const& str) { #ifdef __APPLE__ std::string foo; std::stringstream sstr (str); /* format is: [ DIGITS LIBRARY_NAME ADDRESS SYMBOL OFFSET ] We just need symbol. If this was speed-critical code, we'd likely use C style code to get symbol, but it's not. */ sstr >> foo; sstr >> foo; sstr >> foo; /* Read as far as the "offset" */ char sym[1024]; sstr.getline (sym, sizeof (sym), '+'); if (sstr.bad() || strlen (sym) < 2) { return str; } /* There's a space at the beginning which we don't care about, and one * at the end too */ sym[strlen(sym)-1] = '\0'; std::string symbol = &sym[1]; if (symbol.size() > 2) { if (symbol[0] == '-' && symbol[1] == '[') { /* Objective C */ std::string::size_type bracket = symbol.find_last_of (']'); if (bracket == std::string::npos) { /* Apparently no Objective C, despite early indications that it was */ return demangle_symbol (symbol); } return symbol.substr (0, bracket + 1); } else { /* Not Objective C */ return demangle_symbol (symbol); } } return str; #else std::string::size_type const b = str.find_first_of ("("); if (b == std::string::npos) { return demangle_symbol (str); } std::string::size_type const p = str.find_last_of ("+"); if (p == std::string::npos) { return demangle_symbol (str); } if ((p - b) <= 1) { return demangle_symbol (str); } std::string const symbol = str.substr (b + 1, p - b - 1); return demangle_symbol (symbol); #endif }