| 
							- /*
 -  * DISTRHO Plugin Framework (DPF)
 -  * Copyright (C) 2012-2025 Filipe Coelho <falktx@falktx.com>
 -  *
 -  * Permission to use, copy, modify, and/or distribute this software for any purpose with
 -  * or without fee is hereby granted, provided that the above copyright notice and this
 -  * permission notice appear in all copies.
 -  *
 -  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD
 -  * TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN
 -  * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
 -  * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER
 -  * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
 -  * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 -  */
 - 
 - #ifndef DISTRHO_STRING_HPP_INCLUDED
 - #define DISTRHO_STRING_HPP_INCLUDED
 - 
 - #include "../DistrhoUtils.hpp"
 - #include "../extra/ScopedSafeLocale.hpp"
 - 
 - #include <algorithm>
 - 
 - #if __cplusplus >= 201703L
 - # include <string_view>
 - #endif
 - 
 - START_NAMESPACE_DISTRHO
 - 
 - // -----------------------------------------------------------------------
 - // String class
 - 
 - class String
 - {
 - public:
 -     // -------------------------------------------------------------------
 -     // constructors (no explicit conversions allowed)
 - 
 -     /*
 -      * Empty string.
 -      */
 -     explicit String() noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false) {}
 - 
 -     /*
 -      * Simple character.
 -      */
 -     explicit String(const char c) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         const char ch[2] = { c, '\0' };
 -         _dup(ch);
 -     }
 - 
 -     /*
 -      * Simple char string.
 -      */
 -     explicit String(char* const strBuf, const bool reallocData = true) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         if (reallocData || strBuf == nullptr)
 -         {
 -             _dup(strBuf);
 -         }
 -         else
 -         {
 -             fBuffer      = strBuf;
 -             fBufferLen   = std::strlen(strBuf);
 -             fBufferAlloc = true;
 -         }
 -     }
 - 
 -     /*
 -      * Simple const char string.
 -      */
 -     explicit String(const char* const strBuf) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         _dup(strBuf);
 -     }
 - 
 -    #if __cplusplus >= 201703L
 -     /*
 -      * std::string_view compatible variant.
 -      */
 -     explicit String(const std::string_view& strView) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         _dup(strView.data(), strView.size());
 -     }
 -    #endif
 - 
 -     /*
 -      * Integer.
 -      */
 -     explicit String(const int value) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         char strBuf[0xff+1];
 -         std::snprintf(strBuf, 0xff, "%d", value);
 -         strBuf[0xff] = '\0';
 - 
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Unsigned integer, possibly in hexadecimal.
 -      */
 -     explicit String(const unsigned int value, const bool hexadecimal = false) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         char strBuf[0xff+1];
 -         std::snprintf(strBuf, 0xff, hexadecimal ? "0x%x" : "%u", value);
 -         strBuf[0xff] = '\0';
 - 
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Long integer.
 -      */
 -     explicit String(const long value) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         char strBuf[0xff+1];
 -         std::snprintf(strBuf, 0xff, "%ld", value);
 -         strBuf[0xff] = '\0';
 - 
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Long unsigned integer, possibly hexadecimal.
 -      */
 -     explicit String(const unsigned long value, const bool hexadecimal = false) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         char strBuf[0xff+1];
 -         std::snprintf(strBuf, 0xff, hexadecimal ? "0x%lx" : "%lu", value);
 -         strBuf[0xff] = '\0';
 - 
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Long long integer.
 -      */
 -     explicit String(const long long value) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         char strBuf[0xff+1];
 -         std::snprintf(strBuf, 0xff, "%lld", value);
 -         strBuf[0xff] = '\0';
 - 
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Long long unsigned integer, possibly hexadecimal.
 -      */
 -     explicit String(const unsigned long long value, const bool hexadecimal = false) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         char strBuf[0xff+1];
 -         std::snprintf(strBuf, 0xff, hexadecimal ? "0x%llx" : "%llu", value);
 -         strBuf[0xff] = '\0';
 - 
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Single-precision floating point number.
 -      */
 -     explicit String(const float value) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         char strBuf[0xff+1];
 - 
 -         {
 -             const ScopedSafeLocale ssl;
 -             std::snprintf(strBuf, 0xff, "%.12g", static_cast<double>(value));
 -         }
 - 
 -         strBuf[0xff] = '\0';
 - 
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Double-precision floating point number.
 -      */
 -     explicit String(const double value) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         char strBuf[0xff+1];
 - 
 -         {
 -             const ScopedSafeLocale ssl;
 -             std::snprintf(strBuf, 0xff, "%.24g", value);
 -         }
 - 
 -         strBuf[0xff] = '\0';
 - 
 -         _dup(strBuf);
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // non-explicit constructor
 - 
 -     /*
 -      * Create string from another string.
 -      */
 -     String(const String& str) noexcept
 -         : fBuffer(_null()),
 -           fBufferLen(0),
 -           fBufferAlloc(false)
 -     {
 -         _dup(str.fBuffer);
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // destructor
 - 
 -     /*
 -      * Destructor.
 -      */
 -     ~String() noexcept
 -     {
 -         DISTRHO_SAFE_ASSERT_RETURN(fBuffer != nullptr,);
 - 
 -         if (fBufferAlloc)
 -             std::free(fBuffer);
 - 
 -         fBuffer      = nullptr;
 -         fBufferLen   = 0;
 -         fBufferAlloc = false;
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // public methods
 - 
 -     /*
 -      * Get length of the string.
 -      */
 -     std::size_t length() const noexcept
 -     {
 -         return fBufferLen;
 -     }
 - 
 -     /*
 -      * Check if the string is empty.
 -      */
 -     bool isEmpty() const noexcept
 -     {
 -         return (fBufferLen == 0);
 -     }
 - 
 -     /*
 -      * Check if the string is not empty.
 -      */
 -     bool isNotEmpty() const noexcept
 -     {
 -         return (fBufferLen != 0);
 -     }
 - 
 -     /*
 -      * Check if the string contains a specific character, case-sensitive.
 -      */
 -     bool contains(const char c) const noexcept
 -     {
 -         for (std::size_t i=0; i<fBufferLen; ++i)
 -         {
 -             if (fBuffer[i] == c)
 -                 return true;
 -         }
 - 
 -         return false;
 -     }
 - 
 -     /*
 -      * Check if the string contains another string, optionally ignoring case.
 -      */
 -     bool contains(const char* const strBuf, const bool ignoreCase = false) const noexcept
 -     {
 -         DISTRHO_SAFE_ASSERT_RETURN(strBuf != nullptr, false);
 - 
 -         if (ignoreCase)
 -         {
 - #ifdef __USE_GNU
 -             return (strcasestr(fBuffer, strBuf) != nullptr);
 - #else
 -             String tmp1(fBuffer), tmp2(strBuf);
 - 
 -             // memory allocation failed or empty string(s)
 -             if (tmp1.fBuffer == _null() || tmp2.fBuffer == _null())
 -                 return false;
 - 
 -             tmp1.toLower();
 -             tmp2.toLower();
 -             return (std::strstr(tmp1, tmp2) != nullptr);
 - #endif
 -         }
 - 
 -         return (std::strstr(fBuffer, strBuf) != nullptr);
 -     }
 - 
 -     /*
 -      * Check if character at 'pos' is a digit.
 -      */
 -     bool isDigit(const std::size_t pos) const noexcept
 -     {
 -         DISTRHO_SAFE_ASSERT_RETURN(pos < fBufferLen, false);
 - 
 -         return (fBuffer[pos] >= '0' && fBuffer[pos] <= '9');
 -     }
 - 
 -     /*
 -      * Check if the string starts with the character 'c'.
 -      */
 -     bool startsWith(const char c) const noexcept
 -     {
 -         DISTRHO_SAFE_ASSERT_RETURN(c != '\0', false);
 - 
 -         return (fBufferLen > 0 && fBuffer[0] == c);
 -     }
 - 
 -     /*
 -      * Check if the string starts with the string 'prefix'.
 -      */
 -     bool startsWith(const char* const prefix) const noexcept
 -     {
 -         DISTRHO_SAFE_ASSERT_RETURN(prefix != nullptr, false);
 - 
 -         const std::size_t prefixLen(std::strlen(prefix));
 - 
 -         if (fBufferLen < prefixLen)
 -             return false;
 - 
 -         return (std::strncmp(fBuffer, prefix, prefixLen) == 0);
 -     }
 - 
 -     /*
 -      * Check if the string ends with the character 'c'.
 -      */
 -     bool endsWith(const char c) const noexcept
 -     {
 -         DISTRHO_SAFE_ASSERT_RETURN(c != '\0', false);
 - 
 -         return (fBufferLen > 0 && fBuffer[fBufferLen-1] == c);
 -     }
 - 
 -     /*
 -      * Check if the string ends with the string 'suffix'.
 -      */
 -     bool endsWith(const char* const suffix) const noexcept
 -     {
 -         DISTRHO_SAFE_ASSERT_RETURN(suffix != nullptr, false);
 - 
 -         const std::size_t suffixLen(std::strlen(suffix));
 - 
 -         if (fBufferLen < suffixLen)
 -             return false;
 - 
 -         return (std::strncmp(fBuffer + (fBufferLen-suffixLen), suffix, suffixLen) == 0);
 -     }
 - 
 -     /*
 -      * Find the first occurrence of character 'c' in the string.
 -      * Returns "length()" if the character is not found.
 -      */
 -     std::size_t find(const char c, bool* const found = nullptr) const noexcept
 -     {
 -         if (fBufferLen == 0 || c == '\0')
 -         {
 -             if (found != nullptr)
 -                 *found = false;
 -             return fBufferLen;
 -         }
 - 
 -         for (std::size_t i=0; i < fBufferLen; ++i)
 -         {
 -             if (fBuffer[i] == c)
 -             {
 -                 if (found != nullptr)
 -                     *found = true;
 -                 return i;
 -             }
 -         }
 - 
 -         if (found != nullptr)
 -             *found = false;
 -         return fBufferLen;
 -     }
 - 
 -     /*
 -      * Find the first occurrence of string 'strBuf' in the string.
 -      * Returns "length()" if the string is not found.
 -      */
 -     std::size_t find(const char* const strBuf, bool* const found = nullptr) const noexcept
 -     {
 -         if (fBufferLen == 0 || strBuf == nullptr || strBuf[0] == '\0')
 -         {
 -             if (found != nullptr)
 -                 *found = false;
 -             return fBufferLen;
 -         }
 - 
 -         if (char* const subStrBuf = std::strstr(fBuffer, strBuf))
 -         {
 -             const ssize_t ret(subStrBuf - fBuffer);
 - 
 -             if (ret < 0)
 -             {
 -                 // should never happen!
 -                 d_safe_assert_int("ret >= 0", __FILE__, __LINE__, int(ret));
 - 
 -                 if (found != nullptr)
 -                     *found = false;
 -                 return fBufferLen;
 -             }
 - 
 -             if (found != nullptr)
 -                 *found = true;
 -             return static_cast<std::size_t>(ret);
 -         }
 - 
 -         if (found != nullptr)
 -             *found = false;
 -         return fBufferLen;
 -     }
 - 
 -     /*
 -      * Find the last occurrence of character 'c' in the string.
 -      * Returns "length()" if the character is not found.
 -      */
 -     std::size_t rfind(const char c, bool* const found = nullptr) const noexcept
 -     {
 -         if (fBufferLen == 0 || c == '\0')
 -         {
 -             if (found != nullptr)
 -                 *found = false;
 -             return fBufferLen;
 -         }
 - 
 -         for (std::size_t i=fBufferLen; i > 0; --i)
 -         {
 -             if (fBuffer[i-1] == c)
 -             {
 -                 if (found != nullptr)
 -                     *found = true;
 -                 return i-1;
 -             }
 -         }
 - 
 -         if (found != nullptr)
 -             *found = false;
 -         return fBufferLen;
 -     }
 - 
 -     /*
 -      * Find the last occurrence of string 'strBuf' in the string.
 -      * Returns "length()" if the string is not found.
 -      */
 -     std::size_t rfind(const char* const strBuf, bool* const found = nullptr) const noexcept
 -     {
 -         if (found != nullptr)
 -             *found = false;
 - 
 -         if (fBufferLen == 0 || strBuf == nullptr || strBuf[0] == '\0')
 -             return fBufferLen;
 - 
 -         const std::size_t strBufLen(std::strlen(strBuf));
 - 
 -         std::size_t ret = fBufferLen;
 -         const char* tmpBuf = fBuffer;
 - 
 -         for (std::size_t i=0; i < fBufferLen; ++i)
 -         {
 -             if (std::strstr(tmpBuf+1, strBuf) == nullptr && std::strncmp(tmpBuf, strBuf, strBufLen) == 0)
 -             {
 -                 if (found != nullptr)
 -                     *found = true;
 -                 break;
 -             }
 - 
 -             --ret;
 -             ++tmpBuf;
 -         }
 - 
 -         return fBufferLen-ret;
 -     }
 - 
 -     /*
 -      * Clear the string.
 -      */
 -     void clear() noexcept
 -     {
 -         truncate(0);
 -     }
 - 
 -     /*
 -      * Replace all occurrences of character 'before' with character 'after'.
 -      */
 -     String& replace(const char before, const char after) noexcept
 -     {
 -         DISTRHO_SAFE_ASSERT_RETURN(before != '\0' /* && after != '\0' */, *this);
 - 
 -         for (std::size_t i=0; i < fBufferLen; ++i)
 -         {
 -             if (fBuffer[i] == before)
 -                 fBuffer[i] = after;
 -         }
 - 
 -         return *this;
 -     }
 - 
 -     /*
 -      * Remove all occurrences of character 'c', shifting and truncating the string as necessary.
 -      */
 -     String& remove(const char c) noexcept
 -     {
 -         DISTRHO_SAFE_ASSERT_RETURN(c != '\0', *this);
 - 
 -         if (fBufferLen == 0)
 -             return *this;
 - 
 -         for (std::size_t i=0; i < fBufferLen; ++i)
 -         {
 -             if (fBuffer[i] == c)
 -             {
 -                 --fBufferLen;
 -                 std::memmove(fBuffer+i, fBuffer+i+1, fBufferLen-i);
 -             }
 -         }
 - 
 -         fBuffer[fBufferLen] = '\0';
 -         return *this;
 -     }
 - 
 -     /*
 -      * Truncate the string to size 'n'.
 -      */
 -     String& truncate(const std::size_t n) noexcept
 -     {
 -         if (n >= fBufferLen)
 -             return *this;
 - 
 -         fBuffer[n] = '\0';
 -         fBufferLen = n;
 - 
 -         return *this;
 -     }
 - 
 -     /*
 -      * Convert all non-basic characters to '_'.
 -      */
 -     String& toBasic() noexcept
 -     {
 -         for (std::size_t i=0; i < fBufferLen; ++i)
 -         {
 -             if (fBuffer[i] >= '0' && fBuffer[i] <= '9')
 -                 continue;
 -             if (fBuffer[i] >= 'A' && fBuffer[i] <= 'Z')
 -                 continue;
 -             if (fBuffer[i] >= 'a' && fBuffer[i] <= 'z')
 -                 continue;
 -             if (fBuffer[i] == '_')
 -                 continue;
 - 
 -             fBuffer[i] = '_';
 -         }
 - 
 -         return *this;
 -     }
 - 
 -     /*
 -      * Convert all ascii characters to lowercase.
 -      */
 -     String& toLower() noexcept
 -     {
 -         static constexpr const char kCharDiff = 'a' - 'A';
 - 
 -         for (std::size_t i=0; i < fBufferLen; ++i)
 -         {
 -             if (fBuffer[i] >= 'A' && fBuffer[i] <= 'Z')
 -                 fBuffer[i] = static_cast<char>(fBuffer[i] + kCharDiff);
 -         }
 - 
 -         return *this;
 -     }
 - 
 -     /*
 -      * Convert all ascii characters to uppercase.
 -      */
 -     String& toUpper() noexcept
 -     {
 -         static constexpr const char kCharDiff = 'a' - 'A';
 - 
 -         for (std::size_t i=0; i < fBufferLen; ++i)
 -         {
 -             if (fBuffer[i] >= 'a' && fBuffer[i] <= 'z')
 -                 fBuffer[i] = static_cast<char>(fBuffer[i] - kCharDiff);
 -         }
 - 
 -         return *this;
 -     }
 - 
 -     /*
 -      * Create a new string where all non-basic characters are converted to '_'.
 -      * @see toBasic()
 -      */
 -     String asBasic() const noexcept
 -     {
 -         String s(*this);
 -         return s.toBasic();
 -     }
 - 
 -     /*
 -      * Create a new string where all ascii characters are converted lowercase.
 -      * @see toLower()
 -      */
 -     String asLower() const noexcept
 -     {
 -         String s(*this);
 -         return s.toLower();
 -     }
 - 
 -     /*
 -      * Create a new string where all ascii characters are converted to uppercase.
 -      * @see toUpper()
 -      */
 -     String asUpper() const noexcept
 -     {
 -         String s(*this);
 -         return s.toUpper();
 -     }
 - 
 -     /*
 -      * Direct access to the string buffer (read-only).
 -      */
 -     const char* buffer() const noexcept
 -     {
 -         return fBuffer;
 -     }
 - 
 -     /*
 -      * Get and release the string buffer, while also clearing this string.
 -      * This allows to keep a pointer to the buffer after this object is deleted.
 -      * Result must be freed.
 -      */
 -     char* getAndReleaseBuffer() noexcept
 -     {
 -         char* ret = fBufferLen > 0 ? fBuffer : nullptr;
 -         fBuffer = _null();
 -         fBufferLen = 0;
 -         fBufferAlloc = false;
 -         return ret;
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // base64 stuff, based on http://www.adp-gmbh.ch/cpp/common/base64.html
 -     // Copyright (C) 2004-2008 René Nyffenegger
 - 
 -     static String asBase64(const void* const data, const std::size_t dataSize)
 -     {
 -         static constexpr const char* const kBase64Chars =
 -             "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
 -             "abcdefghijklmnopqrstuvwxyz"
 -             "0123456789+/";
 - 
 -        #ifndef _MSC_VER
 -         const std::size_t kTmpBufSize = std::min(d_nextPowerOf2(static_cast<uint32_t>(dataSize/3)), 65536U);
 -        #else
 -         constexpr std::size_t kTmpBufSize = 65536U;
 -        #endif
 - 
 -         const uchar* bytesToEncode((const uchar*)data);
 - 
 -         uint i=0, j=0;
 -         uint charArray3[3], charArray4[4];
 - 
 -         char strBuf[kTmpBufSize + 1];
 -         strBuf[kTmpBufSize] = '\0';
 -         std::size_t strBufIndex = 0;
 - 
 -         String ret;
 - 
 -         for (std::size_t s=0; s<dataSize; ++s)
 -         {
 -             charArray3[i++] = *(bytesToEncode++);
 - 
 -             if (i == 3)
 -             {
 -                 charArray4[0] =  (charArray3[0] & 0xfc) >> 2;
 -                 charArray4[1] = ((charArray3[0] & 0x03) << 4) + ((charArray3[1] & 0xf0) >> 4);
 -                 charArray4[2] = ((charArray3[1] & 0x0f) << 2) + ((charArray3[2] & 0xc0) >> 6);
 -                 charArray4[3] =   charArray3[2] & 0x3f;
 - 
 -                 for (i=0; i<4; ++i)
 -                     strBuf[strBufIndex++] = kBase64Chars[charArray4[i]];
 - 
 -                 if (strBufIndex >= kTmpBufSize-7)
 -                 {
 -                     strBuf[strBufIndex] = '\0';
 -                     strBufIndex = 0;
 -                     ret += strBuf;
 -                 }
 - 
 -                 i = 0;
 -             }
 -         }
 - 
 -         if (i != 0)
 -         {
 -             for (j=i; j<3; ++j)
 -               charArray3[j] = '\0';
 - 
 -             charArray4[0] =  (charArray3[0] & 0xfc) >> 2;
 -             charArray4[1] = ((charArray3[0] & 0x03) << 4) + ((charArray3[1] & 0xf0) >> 4);
 -             charArray4[2] = ((charArray3[1] & 0x0f) << 2) + ((charArray3[2] & 0xc0) >> 6);
 -             charArray4[3] =   charArray3[2] & 0x3f;
 - 
 -             for (j=0; j<4 && i<3 && j<i+1; ++j)
 -                 strBuf[strBufIndex++] = kBase64Chars[charArray4[j]];
 - 
 -             for (; i++ < 3;)
 -                 strBuf[strBufIndex++] = '=';
 -         }
 - 
 -         if (strBufIndex != 0)
 -         {
 -             strBuf[strBufIndex] = '\0';
 -             ret += strBuf;
 -         }
 - 
 -         return ret;
 -     }
 - 
 -     /*
 -      * Convert to a URL encoded string.
 -      */
 -     String& urlEncode() noexcept
 -     {
 -         static constexpr const char* const kHexChars = "0123456789ABCDEF";
 - 
 -         if (fBufferLen == 0)
 -             return *this;
 - 
 -         char* const newbuf = static_cast<char*>(std::malloc(fBufferLen * 3 + 1));
 -         DISTRHO_SAFE_ASSERT_RETURN(newbuf != nullptr, *this);
 - 
 -         char* newbufptr = newbuf;
 - 
 -         for (std::size_t i=0; i < fBufferLen; ++i)
 -         {
 -             const char c = fBuffer[i];
 - 
 -             switch (c)
 -             {
 -             case '!': // 33
 -             case '#': // 35
 -             case '$': // 36
 -             case '&': // 38
 -             case '\'': // 39
 -             case '(': // 40
 -             case ')': // 41
 -             case '*': // 42
 -             case '+': // 43
 -             case ',': // 44
 -             case '-': // 45
 -             case '.': // 46
 -             case '/': // 47
 -             case '0': // 48
 -             case '1': // 49
 -             case '2': // 50
 -             case '3': // 51
 -             case '4': // 52
 -             case '5': // 53
 -             case '6': // 54
 -             case '7': // 55
 -             case '8': // 56
 -             case '9': // 57
 -             case ':': // 58
 -             case ';': // 59
 -             case '=': // 61
 -             case '?': // 63
 -             case '@': // 64
 -             case 'A': // 65
 -             case 'B': // 66
 -             case 'C': // 67
 -             case 'D': // 68
 -             case 'E': // 69
 -             case 'F': // 70
 -             case 'G': // 71
 -             case 'H': // 72
 -             case 'I': // 73
 -             case 'J': // 74
 -             case 'K': // 75
 -             case 'L': // 76
 -             case 'M': // 77
 -             case 'N': // 78
 -             case 'O': // 79
 -             case 'P': // 80
 -             case 'Q': // 81
 -             case 'R': // 82
 -             case 'S': // 83
 -             case 'T': // 84
 -             case 'U': // 85
 -             case 'V': // 86
 -             case 'W': // 87
 -             case 'X': // 88
 -             case 'Y': // 89
 -             case 'Z': // 90
 -             case '[': // 91
 -             case ']': // 93
 -             case '_': // 95
 -             case 'a': // 97
 -             case 'b': // 98
 -             case 'c': // 99
 -             case 'd': // 100
 -             case 'e': // 101
 -             case 'f': // 102
 -             case 'g': // 103
 -             case 'h': // 104
 -             case 'i': // 105
 -             case 'j': // 106
 -             case 'k': // 107
 -             case 'l': // 108
 -             case 'm': // 109
 -             case 'n': // 110
 -             case 'o': // 111
 -             case 'p': // 112
 -             case 'q': // 113
 -             case 'r': // 114
 -             case 's': // 115
 -             case 't': // 116
 -             case 'u': // 117
 -             case 'v': // 118
 -             case 'w': // 119
 -             case 'x': // 120
 -             case 'y': // 121
 -             case 'z': // 122
 -             case '~': // 126
 -                 *newbufptr++ = c;
 -                 break;
 -             default:
 -                 *newbufptr++ = '%';
 -                 *newbufptr++ = kHexChars[(c >> 4) & 0xf];
 -                 *newbufptr++ = kHexChars[c & 0xf];
 -                 break;
 -             }
 -         }
 - 
 -         *newbufptr = '\0';
 - 
 -         std::free(fBuffer);
 -         fBuffer = newbuf;
 -         fBufferLen = std::strlen(newbuf);
 -         fBufferAlloc = true;
 - 
 -         return *this;
 -     }
 - 
 -     /*
 -      * Convert to a URL decoded string.
 -      */
 -     String& urlDecode() noexcept
 -     {
 -         if (fBufferLen == 0)
 -             return *this;
 - 
 -         char* const newbuf = static_cast<char*>(std::malloc(fBufferLen + 1));
 -         DISTRHO_SAFE_ASSERT_RETURN(newbuf != nullptr, *this);
 - 
 -         char* newbufptr = newbuf;
 - 
 -         for (std::size_t i=0; i < fBufferLen; ++i)
 -         {
 -             const char c = fBuffer[i];
 - 
 -             if (c == '%')
 -             {
 -                 DISTRHO_SAFE_ASSERT_CONTINUE(fBufferLen > i + 2);
 - 
 -                 char c1 = fBuffer[i + 1];
 -                 char c2 = fBuffer[i + 2];
 -                 i += 2;
 - 
 -                 /**/ if (c1 >= '0' && c1 <= '9')
 -                     c1 -= '0';
 -                 else if (c1 >= 'A' && c1 <= 'Z')
 -                     c1 -= 'A' - 10;
 -                 else if (c1 >= 'a' && c1 <= 'z')
 -                     c1 -= 'a' - 10;
 -                 else
 -                     continue;
 - 
 -                 /**/ if (c2 >= '0' && c2 <= '9')
 -                     c2 -= '0';
 -                 else if (c2 >= 'A' && c2 <= 'Z')
 -                     c2 -= 'A' - 10;
 -                 else if (c2 >= 'a' && c2 <= 'z')
 -                     c2 -= 'a' - 10;
 -                 else
 -                     continue;
 - 
 -                 *newbufptr++ = c1 << 4 | c2;
 -             }
 -             else
 -             {
 -                 *newbufptr++ = c;
 -             }
 -         }
 - 
 -         *newbufptr = '\0';
 - 
 -         std::free(fBuffer);
 -         fBuffer = newbuf;
 -         fBufferLen = std::strlen(newbuf);
 -         fBufferAlloc = true;
 - 
 -         return *this;
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // public operators
 - 
 -     operator const char*() const noexcept
 -     {
 -         return fBuffer;
 -     }
 - 
 -     char operator[](const std::size_t pos) const noexcept
 -     {
 -         if (pos < fBufferLen)
 -             return fBuffer[pos];
 - 
 -         d_safe_assert("pos < fBufferLen", __FILE__, __LINE__);
 - 
 -         static char fallback;
 -         fallback = '\0';
 -         return fallback;
 -     }
 - 
 -     char& operator[](const std::size_t pos) noexcept
 -     {
 -         if (pos < fBufferLen)
 -             return fBuffer[pos];
 - 
 -         d_safe_assert("pos < fBufferLen", __FILE__, __LINE__);
 - 
 -         static char fallback;
 -         fallback = '\0';
 -         return fallback;
 -     }
 - 
 -     bool operator==(const char* const strBuf) const noexcept
 -     {
 -         return (strBuf != nullptr && std::strcmp(fBuffer, strBuf) == 0);
 -     }
 - 
 -     bool operator==(const String& str) const noexcept
 -     {
 -         return operator==(str.fBuffer);
 -     }
 - 
 -     bool operator!=(const char* const strBuf) const noexcept
 -     {
 -         return !operator==(strBuf);
 -     }
 - 
 -     bool operator!=(const String& str) const noexcept
 -     {
 -         return !operator==(str.fBuffer);
 -     }
 - 
 -     String& operator=(const char* const strBuf) noexcept
 -     {
 -         _dup(strBuf);
 - 
 -         return *this;
 -     }
 - 
 -     String& operator=(const String& str) noexcept
 -     {
 -         _dup(str.fBuffer);
 - 
 -         return *this;
 -     }
 - 
 -     String& operator+=(const char* const strBuf) noexcept
 -     {
 -         if (strBuf == nullptr || strBuf[0] == '\0')
 -             return *this;
 - 
 -         const std::size_t strBufLen = std::strlen(strBuf);
 - 
 -         // for empty strings, we can just take the appended string as our entire data
 -         if (isEmpty())
 -         {
 -             _dup(strBuf, strBufLen);
 -             return *this;
 -         }
 - 
 -         // we have some data ourselves, reallocate to add the new stuff
 -         char* const newBuf = static_cast<char*>(std::realloc(fBufferAlloc ? fBuffer : nullptr, fBufferLen + strBufLen + 1));
 -         DISTRHO_SAFE_ASSERT_RETURN(newBuf != nullptr, *this);
 - 
 -         std::memcpy(newBuf + fBufferLen, strBuf, strBufLen + 1);
 - 
 -         fBuffer = newBuf;
 -         fBufferLen += strBufLen;
 -         fBufferAlloc = true;
 - 
 -         return *this;
 -     }
 - 
 -     String& operator+=(const String& str) noexcept
 -     {
 -         return operator+=(str.fBuffer);
 -     }
 - 
 -     String operator+(const char* const strBuf) noexcept
 -     {
 -         if (strBuf == nullptr || strBuf[0] == '\0')
 -             return *this;
 -         if (isEmpty())
 -             return String(strBuf);
 - 
 -         const std::size_t strBufLen = std::strlen(strBuf);
 -         const std::size_t newBufSize = fBufferLen + strBufLen;
 -         char* const newBuf = static_cast<char*>(std::malloc(newBufSize + 1));
 -         DISTRHO_SAFE_ASSERT_RETURN(newBuf != nullptr, String());
 - 
 -         std::memcpy(newBuf, fBuffer, fBufferLen);
 -         std::memcpy(newBuf + fBufferLen, strBuf, strBufLen + 1);
 - 
 -         return String(newBuf, false);
 -     }
 - 
 -     String operator+(const String& str) noexcept
 -     {
 -         return operator+(str.fBuffer);
 -     }
 - 
 -     // needed for std::map compatibility
 -     bool operator<(const String& str) const noexcept
 -     {
 -         return std::strcmp(fBuffer, str.fBuffer) < 0;
 -     }
 - 
 -     // -------------------------------------------------------------------
 - 
 - private:
 -     char*       fBuffer;      // the actual string buffer
 -     std::size_t fBufferLen;   // string length
 -     bool        fBufferAlloc; // wherever the buffer is allocated, not using _null()
 - 
 -     /*
 -      * Static null string.
 -      * Prevents allocation for new and/or empty strings.
 -      */
 -     static char* _null() noexcept
 -     {
 -         static char sNull = '\0';
 -         return &sNull;
 -     }
 - 
 -     /*
 -      * Helper function.
 -      * Called whenever the string needs to be allocated.
 -      *
 -      * Notes:
 -      * - Allocates string only if 'strBuf' is not null and new string contents are different
 -      * - If 'strBuf' is null, 'size' must be 0
 -      */
 -     void _dup(const char* const strBuf, const std::size_t size = 0) noexcept
 -     {
 -         if (strBuf != nullptr)
 -         {
 -             // don't recreate string if contents match
 -             if (std::strcmp(fBuffer, strBuf) == 0)
 -                 return;
 - 
 -             if (fBufferAlloc)
 -                 std::free(fBuffer);
 - 
 -             fBufferLen = (size > 0) ? size : std::strlen(strBuf);
 -             fBuffer    = static_cast<char*>(std::malloc(fBufferLen + 1));
 - 
 -             if (fBuffer == nullptr)
 -             {
 -                 fBuffer      = _null();
 -                 fBufferLen   = 0;
 -                 fBufferAlloc = false;
 -                 return;
 -             }
 - 
 -             fBufferAlloc = true;
 - 
 -             std::strcpy(fBuffer, strBuf);
 -             fBuffer[fBufferLen] = '\0';
 -         }
 -         else
 -         {
 -             DISTRHO_SAFE_ASSERT_UINT(size == 0, static_cast<uint>(size));
 - 
 -             // don't recreate null string
 -             if (! fBufferAlloc)
 -                 return;
 - 
 -             DISTRHO_SAFE_ASSERT(fBuffer != nullptr);
 -             std::free(fBuffer);
 - 
 -             fBuffer      = _null();
 -             fBufferLen   = 0;
 -             fBufferAlloc = false;
 -         }
 -     }
 - 
 -     DISTRHO_PREVENT_HEAP_ALLOCATION
 - };
 - 
 - // -----------------------------------------------------------------------
 - 
 - static inline
 - String operator+(const String& strBefore, const char* const strBufAfter) noexcept
 - {
 -     if (strBufAfter == nullptr || strBufAfter[0] == '\0')
 -         return strBefore;
 -     if (strBefore.isEmpty())
 -         return String(strBufAfter);
 - 
 -     const std::size_t strBeforeLen = strBefore.length();
 -     const std::size_t strBufAfterLen = std::strlen(strBufAfter);
 -     const std::size_t newBufSize = strBeforeLen + strBufAfterLen;
 -     char* const newBuf = static_cast<char*>(malloc(newBufSize + 1));
 -     DISTRHO_SAFE_ASSERT_RETURN(newBuf != nullptr, String());
 - 
 -     std::memcpy(newBuf, strBefore.buffer(), strBeforeLen);
 -     std::memcpy(newBuf + strBeforeLen, strBufAfter, strBufAfterLen + 1);
 - 
 -     return String(newBuf, false);
 - }
 - 
 - static inline
 - String operator+(const char* const strBufBefore, const String& strAfter) noexcept
 - {
 -     if (strAfter.isEmpty())
 -         return String(strBufBefore);
 -     if (strBufBefore == nullptr || strBufBefore[0] == '\0')
 -         return strAfter;
 - 
 -     const std::size_t strBufBeforeLen = std::strlen(strBufBefore);
 -     const std::size_t strAfterLen = strAfter.length();
 -     const std::size_t newBufSize = strBufBeforeLen + strAfterLen;
 -     char* const newBuf = static_cast<char*>(malloc(newBufSize + 1));
 -     DISTRHO_SAFE_ASSERT_RETURN(newBuf != nullptr, String());
 - 
 -     std::memcpy(newBuf, strBufBefore, strBufBeforeLen);
 -     std::memcpy(newBuf + strBufBeforeLen, strAfter.buffer(), strAfterLen + 1);
 - 
 -     return String(newBuf, false);
 - }
 - 
 - // -----------------------------------------------------------------------
 - 
 - END_NAMESPACE_DISTRHO
 - 
 - #endif // DISTRHO_STRING_HPP_INCLUDED
 
 
  |