indiaai-hackathon
/
datalab
/web
/node_modules
/node-pty
/deps
/winpty
/src
/shared
/StringBuilder.h
| // Copyright (c) 2016 Ryan Prichard | |
| // | |
| // Permission is hereby granted, free of charge, to any person obtaining a copy | |
| // of this software and associated documentation files (the "Software"), to | |
| // deal in the Software without restriction, including without limitation the | |
| // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or | |
| // sell copies of the Software, and to permit persons to whom the Software is | |
| // furnished to do so, subject to the following conditions: | |
| // | |
| // The above copyright notice and this permission notice shall be included in | |
| // all copies or substantial portions of the Software. | |
| // | |
| // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |
| // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |
| // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |
| // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |
| // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | |
| // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS | |
| // IN THE SOFTWARE. | |
| // Efficient integer->string conversion and string concatenation. The | |
| // hexadecimal conversion may optionally have leading zeros. Other ways to | |
| // convert integers to strings in C++ suffer these drawbacks: | |
| // | |
| // * std::stringstream: Inefficient, even more so than stdio. | |
| // | |
| // * std::to_string: No hexadecimal output, tends to use heap allocation, not | |
| // supported on Cygwin. | |
| // | |
| // * stdio routines: Requires parsing a format string (inefficient). The | |
| // caller *must* know how large the content is for correctness. The | |
| // string-printf functions are extremely inconsistent on Windows. In | |
| // particular, 64-bit integers, wide strings, and return values are | |
| // problem areas. | |
| // | |
| // StringBuilderTest.cc is a standalone program that tests this header. | |
| template <typename C, size_t sz> | |
| struct ValueString { | |
| std::array<C, sz> m_array; | |
| size_t m_offset; | |
| size_t m_size; | |
| const C *c_str() const { return m_array.data() + m_offset; } | |
| const C *data() const { return m_array.data() + m_offset; } | |
| size_t size() const { return m_size; } | |
| std::basic_string<C> str() const { | |
| return std::basic_string<C>(data(), m_size); | |
| } | |
| }; | |
| // Disable an MSVC /SDL error that forbids unsigned negation. Signed negation | |
| // invokes undefined behavior for INTxx_MIN, so unsigned negation is simpler to | |
| // reason about. (We assume twos-complement in any case.) | |
| // Formats an integer as decimal without leading zeros. | |
| template <typename C, typename I> | |
| ValueString<C, sizeof(I) * 3 + 1 + 1> gdecOfInt(const I value) { | |
| typedef typename std::make_unsigned<I>::type U; | |
| auto unsValue = static_cast<U>(value); | |
| const bool isNegative = (value < 0); | |
| if (isNegative) { | |
| unsValue = STRING_BUILDER_ALLOW_UNSIGNED_NEGATE(-unsValue); | |
| } | |
| decltype(gdecOfInt<C, I>(value)) out; | |
| auto &arr = out.m_array; | |
| C *const endp = arr.data() + arr.size(); | |
| C *outp = endp; | |
| *(--outp) = '\0'; | |
| STRING_BUILDER_CHECK(outp >= arr.data()); | |
| do { | |
| const int digit = unsValue % 10; | |
| unsValue /= 10; | |
| *(--outp) = '0' + digit; | |
| STRING_BUILDER_CHECK(outp >= arr.data()); | |
| } while (unsValue != 0); | |
| if (isNegative) { | |
| *(--outp) = '-'; | |
| STRING_BUILDER_CHECK(outp >= arr.data()); | |
| } | |
| out.m_offset = outp - arr.data(); | |
| out.m_size = endp - outp - 1; | |
| return out; | |
| } | |
| template <typename I> decltype(gdecOfInt<char, I>(0)) decOfInt(I i) { | |
| return gdecOfInt<char>(i); | |
| } | |
| template <typename I> decltype(gdecOfInt<wchar_t, I>(0)) wdecOfInt(I i) { | |
| return gdecOfInt<wchar_t>(i); | |
| } | |
| // Formats an integer as hexadecimal, with or without leading zeros. | |
| template <typename C, bool leadingZeros=false, typename I> | |
| ValueString<C, sizeof(I) * 2 + 1> ghexOfInt(const I value) { | |
| typedef typename std::make_unsigned<I>::type U; | |
| const auto unsValue = static_cast<U>(value); | |
| static const C hex[16] = {'0','1','2','3','4','5','6','7', | |
| '8','9','a','b','c','d','e','f'}; | |
| decltype(ghexOfInt<C, leadingZeros, I>(value)) out; | |
| auto &arr = out.m_array; | |
| C *outp = arr.data(); | |
| int inIndex = 0; | |
| int shift = sizeof(I) * 8 - 4; | |
| const int len = sizeof(I) * 2; | |
| if (!leadingZeros) { | |
| for (; inIndex < len - 1; ++inIndex, shift -= 4) { | |
| STRING_BUILDER_CHECK(shift >= 0 && shift < sizeof(unsValue) * 8); | |
| const int digit = (unsValue >> shift) & 0xF; | |
| if (digit != 0) { | |
| break; | |
| } | |
| } | |
| } | |
| for (; inIndex < len; ++inIndex, shift -= 4) { | |
| const int digit = (unsValue >> shift) & 0xF; | |
| *(outp++) = hex[digit]; | |
| STRING_BUILDER_CHECK(outp <= arr.data() + arr.size()); | |
| } | |
| *(outp++) = '\0'; | |
| STRING_BUILDER_CHECK(outp <= arr.data() + arr.size()); | |
| out.m_offset = 0; | |
| out.m_size = outp - arr.data() - 1; | |
| return out; | |
| } | |
| template <bool leadingZeros=false, typename I> | |
| decltype(ghexOfInt<char, leadingZeros, I>(0)) hexOfInt(I i) { | |
| return ghexOfInt<char, leadingZeros, I>(i); | |
| } | |
| template <bool leadingZeros=false, typename I> | |
| decltype(ghexOfInt<wchar_t, leadingZeros, I>(0)) whexOfInt(I i) { | |
| return ghexOfInt<wchar_t, leadingZeros, I>(i); | |
| } | |
| template <typename C> | |
| class GStringBuilder { | |
| public: | |
| typedef std::basic_string<C> StringType; | |
| GStringBuilder() {} | |
| GStringBuilder(size_t capacity) { | |
| m_out.reserve(capacity); | |
| } | |
| GStringBuilder &operator<<(C ch) { m_out.push_back(ch); return *this; } | |
| GStringBuilder &operator<<(const C *str) { m_out.append(str); return *this; } | |
| GStringBuilder &operator<<(const StringType &str) { m_out.append(str); return *this; } | |
| template <size_t sz> | |
| GStringBuilder &operator<<(const ValueString<C, sz> &str) { | |
| m_out.append(str.data(), str.size()); | |
| return *this; | |
| } | |
| private: | |
| // Forbid output of char/wchar_t for GStringBuilder if the type doesn't | |
| // exactly match the builder element type. The code still allows | |
| // signed char and unsigned char, but I'm a little worried about what | |
| // happens if a user tries to output int8_t or uint8_t. | |
| template <typename P> | |
| typename std::enable_if< | |
| (std::is_same<P, char>::value || std::is_same<P, wchar_t>::value) && | |
| !std::is_same<C, P>::value, GStringBuilder&>::type | |
| operator<<(P ch) { | |
| ASSERT(false && "Method was not supposed to be reachable."); | |
| return *this; | |
| } | |
| public: | |
| GStringBuilder &operator<<(short i) { return *this << gdecOfInt<C>(i); } | |
| GStringBuilder &operator<<(unsigned short i) { return *this << gdecOfInt<C>(i); } | |
| GStringBuilder &operator<<(int i) { return *this << gdecOfInt<C>(i); } | |
| GStringBuilder &operator<<(unsigned int i) { return *this << gdecOfInt<C>(i); } | |
| GStringBuilder &operator<<(long i) { return *this << gdecOfInt<C>(i); } | |
| GStringBuilder &operator<<(unsigned long i) { return *this << gdecOfInt<C>(i); } | |
| GStringBuilder &operator<<(long long i) { return *this << gdecOfInt<C>(i); } | |
| GStringBuilder &operator<<(unsigned long long i) { return *this << gdecOfInt<C>(i); } | |
| GStringBuilder &operator<<(const void *p) { | |
| m_out.push_back(static_cast<C>('0')); | |
| m_out.push_back(static_cast<C>('x')); | |
| *this << ghexOfInt<C>(reinterpret_cast<uintptr_t>(p)); | |
| return *this; | |
| } | |
| StringType str() { return m_out; } | |
| StringType str_moved() { return std::move(m_out); } | |
| const C *c_str() const { return m_out.c_str(); } | |
| private: | |
| StringType m_out; | |
| }; | |
| typedef GStringBuilder<char> StringBuilder; | |
| typedef GStringBuilder<wchar_t> WStringBuilder; | |