Socket/Protocols/INet: Allow socket address string representation to omit the address...
[senf.git] / Utils / Exception.cc
1 // $Id$
2 //
3 // Copyright (C) 2006
4 // Fraunhofer Institute for Open Communication Systems (FOKUS)
5 // Competence Center NETwork research (NET), St. Augustin, GERMANY
6 //     Stefan Bund <g0dil@berlios.de>
7 //
8 // This program is free software; you can redistribute it and/or modify
9 // it under the terms of the GNU General Public License as published by
10 // the Free Software Foundation; either version 2 of the License, or
11 // (at your option) any later version.
12 //
13 // This program is distributed in the hope that it will be useful,
14 // but WITHOUT ANY WARRANTY; without even the implied warranty of
15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 // GNU General Public License for more details.
17 //
18 // You should have received a copy of the GNU General Public License
19 // along with this program; if not, write to the
20 // Free Software Foundation, Inc.,
21 // 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22
23 /** \file
24     \brief Exception non-inline non-template implementation */
25
26 #include "Exception.hh"
27 //#include "Exception.ih"
28
29 // Custom includes
30 #include <execinfo.h>
31 #include <sstream>
32 #include "../config.hh"
33 #include "impl/demangle.h"
34
35 #define prefix_
36 ///////////////////////////////cc.p////////////////////////////////////////
37
38 ///////////////////////////////////////////////////////////////////////////
39 // senf::Exception
40
41 #ifdef SENF_DEBUG
42 prefix_ void senf::ExceptionMixin::addBacktrace()
43 {
44     void * entries[SENF_DEBUG_BACKTRACE_NUMCALLERS];
45     unsigned nEntries( ::backtrace(entries, SENF_DEBUG_BACKTRACE_NUMCALLERS) );
46     char ** symbols = ::backtrace_symbols(entries, nEntries);
47     
48     std::stringstream ss;
49     ss << "\nException at\n";
50     for (unsigned i=0; i<nEntries; ++i) {
51         std::string sym (symbols[i]);
52         std::string::size_type fnStart (sym.find("("));
53         if (fnStart != std::string::npos) {
54             std::string::size_type fnEnd (sym.find(")",fnStart+1));
55             if (fnEnd != std::string::npos) {
56                 std::string fn (sym,fnStart+1, fnEnd-fnStart-1);
57                 char * demangled ( ::cplus_demangle(fn.c_str(), 
58                                                     DMGL_TYPES|DMGL_AUTO) );
59                 if (demangled) {
60                     ss << "  " << demangled << "( ... )" << std::string(sym,fnEnd+1) << "\n";
61                     continue;
62                 }
63             }
64         }
65         ss << "  " << sym << "\n";
66     }
67     ss << "-- \n" << message_;
68     message_ = ss.str();
69     free(symbols);
70 }
71 #endif
72
73 /////////////////////////////////////////////////////////////////////////// 
74 // senf::Exception
75
76 prefix_ senf::Exception::~Exception()
77     throw()
78 {}
79
80 prefix_ char const * senf::Exception::what()
81     const throw()
82 {
83     return message().c_str();
84 }
85
86 ///////////////////////////////////////////////////////////////////////////
87 // senf::SystemException
88
89 prefix_ void senf::SystemException::init(std::string const & descr, int code
90                                          _SENF_EXC_DEBUG_ARGS_ND)
91 {
92     code_ = code;
93 #   ifdef SENF_DEBUG
94     if (file && line)
95         (*this) << "Exception at " << file << ":" << line << "\n";
96 #   endif
97     (*this) << "[" << errorString() << "]";
98     if (! descr.empty()) (*this) << " " << descr;
99 }
100
101 ///////////////////////////////cc.e////////////////////////////////////////
102 #undef prefix_
103
104 \f
105 // Local Variables:
106 // mode: c++
107 // fill-column: 100
108 // c-file-style: "senf"
109 // indent-tabs-mode: nil
110 // ispell-local-dictionary: "american"
111 // compile-command: "scons -u test"
112 // comment-column: 40
113 // End: