/* FORMAT-OBJ.hpp - simple means to display an object Copyright (C) Lumiera.org 2016, Hermann Vosseler 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., 675 Mass Ave, Cambridge, MA 02139, USA. */ /** @file format-obj.hpp ** Simple functions to represent objects, for debugging and diagnostics. ** The helpers provided here are rather commonplace, but written in a way ** as to incur only modest header inclusion load. It should be OK to use ** these even on widely used interface headers. ** - util::toString() performs a failsafe to-String conversion, thereby preferring a ** built-in conversion operator, falling back to a lexical conversion (boost) ** or just a unmangled and simplified type string as default. ** - util::typedString() combines this with a always visible type display ** - lib::meta::demangleCxx() uses the built-in compiler support to translate a mangled ** type-ID (as given by `typeid(TY).name()`) into a readable, fully qualified ** C++ type name. This is only supported for GNU compatible compilers. ** ** @todo is it possible to stash away the `boost::lexical_cast` behind a custom facade, ** the way we did it for `boost::format`? This would reduce inclusion cost... ** ** @see FormatHelper_test ** @see [frontend for boost::format, printf-style](format-string.hpp) ** */ #ifndef LIB_FORMAT_OBJ_H #define LIB_FORMAT_OBJ_H #include "lib/symbol.hpp" #include "lib/meta/trait.hpp" #include namespace std { // forward declaration to avoid including template struct char_traits; template class basic_ostream; using ostream = basic_ostream>; } namespace lib { class Literal; namespace meta { std::string demangleCxx (lib::Literal rawName); std::string humanReadableTypeID (lib::Literal); std::string primaryTypeComponent (lib::Literal); std::string sanitisedFullTypeName(lib::Literal); }}// namespace lib::meta namespace util { std::string showDouble (double) noexcept; std::string showFloat (float) noexcept; std::string showAddr (void const* addr) noexcept; /** preconfigured format for pretty-printing of addresses */ std::ostream& showAddr (std::ostream&, void const* addr); namespace { /** toggle to prefer specialisation with direct lexical conversion */ template using enable_LexicalConversion = lib::meta::enable_if< lib::meta::use_LexicalConversion>; template using show_SmartPointer = lib::meta::enable_if< lib::meta::is_smart_ptr::TypeReferred>>; } /* === generalise the failsafe string conversion === */ /** @note base case is defined in meta/util.hpp */ template struct StringConv> { static std::string invoke (X const& val) noexcept try { return boost::lexical_cast (val); } catch(...) { return FAILURE_INDICATOR; } }; template struct StringConv> { static std::string invoke (SP const& smP) noexcept try { return showSmartPtr (smP, lib::meta::typeSymbol(smP)); } catch(...) { return FAILURE_INDICATOR; } }; /** explicit specialisation to control precision of double values. * @note we set an explicit precision, since this is a diagnostic facility * and we typically do not want to see all digits, but, for test code, * we do want a predictable string representation of simple fractional * values like `0.1` (which can not be represented as binary floats) */ template<> struct StringConv { static std::string invoke (double val) noexcept { return util::showDouble (val); } }; template<> struct StringConv { static std::string invoke (float val) noexcept { return util::showFloat (val); } }; template<> struct StringConv { static std::string invoke (bool val) noexcept { return util::showBool (val); } }; /** * get some string representation of any object, reliably. * A custom string conversion operator is invoked, if applicable, * while all lexically convertible types (numbers etc) are treated * by boost::lexical_cast. For double or float values, hard wired * rounding to a fixed number of digits will be performed, to yield * a predictable display of printed unit-test results. */ template inline std::string toString (TY const& val) noexcept { return StringConv::invoke (val); } /** * indicate type and possibly a (custom) conversion to string * @return human readable type name '|' string representation. * or just the type, when no string representation available */ template inline std::string typedString (TY const& val) noexcept try { std::string repr = StringConv::invoke (val); return 0 == repr.rfind("«", 0)? repr : "«"+typeStr(val)+"»|"+repr; } catch(...) { return FAILURE_INDICATOR; } } // namespace util #endif /*LIB_FORMAT_OBJ_H*/