123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265 |
- /*
- * Created by Phil on 23/4/2014.
- * Copyright 2014 Two Blue Cubes Ltd. All rights reserved.
- *
- * Distributed under the Boost Software License, Version 1.0. (See accompanying
- * file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
- */
- #if defined(__clang__)
- # pragma clang diagnostic push
- # pragma clang diagnostic ignored "-Wexit-time-destructors"
- # pragma clang diagnostic ignored "-Wglobal-constructors"
- #endif
- // Enable specific decls locally
- #if !defined(CATCH_CONFIG_ENABLE_CHRONO_STRINGMAKER)
- #define CATCH_CONFIG_ENABLE_CHRONO_STRINGMAKER
- #endif
- #include "catch_tostring.h"
- #include "catch_interfaces_config.h"
- #include "catch_context.h"
- #include "catch_polyfills.hpp"
- #include <cmath>
- #include <iomanip>
- namespace Catch {
- namespace Detail {
- const std::string unprintableString = "{?}";
- namespace {
- const int hexThreshold = 255;
- struct Endianness {
- enum Arch { Big, Little };
- static Arch which() {
- int one = 1;
- // If the lowest byte we read is non-zero, we can assume
- // that little endian format is used.
- auto value = *reinterpret_cast<char*>(&one);
- return value ? Little : Big;
- }
- };
- }
- std::string rawMemoryToString( const void *object, std::size_t size ) {
- // Reverse order for little endian architectures
- int i = 0, end = static_cast<int>( size ), inc = 1;
- if( Endianness::which() == Endianness::Little ) {
- i = end-1;
- end = inc = -1;
- }
- unsigned char const *bytes = static_cast<unsigned char const *>(object);
- ReusableStringStream rss;
- rss << "0x" << std::setfill('0') << std::hex;
- for( ; i != end; i += inc )
- rss << std::setw(2) << static_cast<unsigned>(bytes[i]);
- return rss.str();
- }
- }
- template<typename T>
- std::string fpToString( T value, int precision ) {
- if (Catch::isnan(value)) {
- return "nan";
- }
- ReusableStringStream rss;
- rss << std::setprecision( precision )
- << std::fixed
- << value;
- std::string d = rss.str();
- std::size_t i = d.find_last_not_of( '0' );
- if( i != std::string::npos && i != d.size()-1 ) {
- if( d[i] == '.' )
- i++;
- d = d.substr( 0, i+1 );
- }
- return d;
- }
- //// ======================================================= ////
- //
- // Out-of-line defs for full specialization of StringMaker
- //
- //// ======================================================= ////
- std::string StringMaker<std::string>::convert(const std::string& str) {
- if (!getCurrentContext().getConfig()->showInvisibles()) {
- return '"' + str + '"';
- }
- std::string s("\"");
- for (char c : str) {
- switch (c) {
- case '\n':
- s.append("\\n");
- break;
- case '\t':
- s.append("\\t");
- break;
- default:
- s.push_back(c);
- break;
- }
- }
- s.append("\"");
- return s;
- }
- #ifdef CATCH_CONFIG_CPP17_STRING_VIEW
- std::string StringMaker<std::string_view>::convert(std::string_view str) {
- return ::Catch::Detail::stringify(std::string{ str });
- }
- #endif
- std::string StringMaker<char const*>::convert(char const* str) {
- if (str) {
- return ::Catch::Detail::stringify(std::string{ str });
- } else {
- return{ "{null string}" };
- }
- }
- std::string StringMaker<char*>::convert(char* str) {
- if (str) {
- return ::Catch::Detail::stringify(std::string{ str });
- } else {
- return{ "{null string}" };
- }
- }
- #ifdef CATCH_CONFIG_WCHAR
- std::string StringMaker<std::wstring>::convert(const std::wstring& wstr) {
- std::string s;
- s.reserve(wstr.size());
- for (auto c : wstr) {
- s += (c <= 0xff) ? static_cast<char>(c) : '?';
- }
- return ::Catch::Detail::stringify(s);
- }
- # ifdef CATCH_CONFIG_CPP17_STRING_VIEW
- std::string StringMaker<std::wstring_view>::convert(std::wstring_view str) {
- return StringMaker<std::wstring>::convert(std::wstring(str));
- }
- # endif
- std::string StringMaker<wchar_t const*>::convert(wchar_t const * str) {
- if (str) {
- return ::Catch::Detail::stringify(std::wstring{ str });
- } else {
- return{ "{null string}" };
- }
- }
- std::string StringMaker<wchar_t *>::convert(wchar_t * str) {
- if (str) {
- return ::Catch::Detail::stringify(std::wstring{ str });
- } else {
- return{ "{null string}" };
- }
- }
- #endif
- #if defined(CATCH_CONFIG_CPP17_BYTE)
- #include <cstddef>
- std::string StringMaker<std::byte>::convert(std::byte value) {
- return ::Catch::Detail::stringify(std::to_integer<unsigned long long>(value));
- }
- #endif // defined(CATCH_CONFIG_CPP17_BYTE)
- std::string StringMaker<int>::convert(int value) {
- return ::Catch::Detail::stringify(static_cast<long long>(value));
- }
- std::string StringMaker<long>::convert(long value) {
- return ::Catch::Detail::stringify(static_cast<long long>(value));
- }
- std::string StringMaker<long long>::convert(long long value) {
- ReusableStringStream rss;
- rss << value;
- if (value > Detail::hexThreshold) {
- rss << " (0x" << std::hex << value << ')';
- }
- return rss.str();
- }
- std::string StringMaker<unsigned int>::convert(unsigned int value) {
- return ::Catch::Detail::stringify(static_cast<unsigned long long>(value));
- }
- std::string StringMaker<unsigned long>::convert(unsigned long value) {
- return ::Catch::Detail::stringify(static_cast<unsigned long long>(value));
- }
- std::string StringMaker<unsigned long long>::convert(unsigned long long value) {
- ReusableStringStream rss;
- rss << value;
- if (value > Detail::hexThreshold) {
- rss << " (0x" << std::hex << value << ')';
- }
- return rss.str();
- }
- std::string StringMaker<bool>::convert(bool b) {
- return b ? "true" : "false";
- }
- std::string StringMaker<signed char>::convert(signed char value) {
- if (value == '\r') {
- return "'\\r'";
- } else if (value == '\f') {
- return "'\\f'";
- } else if (value == '\n') {
- return "'\\n'";
- } else if (value == '\t') {
- return "'\\t'";
- } else if ('\0' <= value && value < ' ') {
- return ::Catch::Detail::stringify(static_cast<unsigned int>(value));
- } else {
- char chstr[] = "' '";
- chstr[1] = value;
- return chstr;
- }
- }
- std::string StringMaker<char>::convert(char c) {
- return ::Catch::Detail::stringify(static_cast<signed char>(c));
- }
- std::string StringMaker<unsigned char>::convert(unsigned char c) {
- return ::Catch::Detail::stringify(static_cast<char>(c));
- }
- std::string StringMaker<std::nullptr_t>::convert(std::nullptr_t) {
- return "nullptr";
- }
- int StringMaker<float>::precision = 5;
- std::string StringMaker<float>::convert(float value) {
- return fpToString(value, precision) + 'f';
- }
- int StringMaker<double>::precision = 10;
- std::string StringMaker<double>::convert(double value) {
- return fpToString(value, precision);
- }
- std::string ratio_string<std::atto>::symbol() { return "a"; }
- std::string ratio_string<std::femto>::symbol() { return "f"; }
- std::string ratio_string<std::pico>::symbol() { return "p"; }
- std::string ratio_string<std::nano>::symbol() { return "n"; }
- std::string ratio_string<std::micro>::symbol() { return "u"; }
- std::string ratio_string<std::milli>::symbol() { return "m"; }
- } // end namespace Catch
- #if defined(__clang__)
- # pragma clang diagnostic pop
- #endif
|