| 
							- /*
 -  * DISTRHO Plugin Toolkit (DPT)
 -  * Copyright (C) 2012-2013 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_UTILS_HPP_INCLUDED
 - #define DISTRHO_UTILS_HPP_INCLUDED
 - 
 - #include "src/DistrhoDefines.h"
 - 
 - #include <cassert>
 - #include <cstdarg>
 - #include <cstdio>
 - #include <cstdlib>
 - #include <cstring>
 - 
 - #ifdef PROPER_CPP11_SUPPORT
 - # include <cstdint>
 - #else
 - # include <stdint.h>
 - #endif
 - 
 - #ifdef DISTRHO_OS_WINDOWS
 - # include <windows.h>
 - #else
 - # include <unistd.h>
 - #endif
 - 
 - #if defined(DISTRHO_OS_MAC) && ! defined(CARLA_OS_MAC)
 - namespace std {
 - inline float
 -   fmin(float __x, float __y)
 -   { return __builtin_fminf(__x, __y); }
 - inline float
 -   fmax(float __x, float __y)
 -   { return __builtin_fmaxf(__x, __y); }
 - inline float
 -   rint(float __x)
 -   { return __builtin_rintf(__x); }
 - }
 - #endif
 - 
 - // -----------------------------------------------------------------------
 - // misc functions
 - 
 - static inline
 - long d_cconst(int a, int b, int c, int d) noexcept
 - {
 -     return (a << 24) | (b << 16) | (c << 8) | (d << 0);
 - }
 - 
 - // -----------------------------------------------------------------------
 - // string print functions
 - 
 - #ifndef DEBUG
 - # define d_debug(...)
 - #else
 - static inline
 - void d_debug(const char* const fmt, ...)
 - {
 -     va_list args;
 -     va_start(args, fmt);
 -     std::fprintf(stdout, "\x1b[30;1m");
 -     std::vfprintf(stdout, fmt, args);
 -     std::fprintf(stdout, "\x1b[0m\n");
 -     va_end(args);
 - }
 - #endif
 - 
 - static inline
 - void d_stdout(const char* const fmt, ...)
 - {
 -     va_list args;
 -     va_start(args, fmt);
 -     std::vfprintf(stdout, fmt, args);
 -     std::fprintf(stdout, "\n");
 -     va_end(args);
 - }
 - 
 - static inline
 - void d_stderr(const char* const fmt, ...)
 - {
 -     va_list args;
 -     va_start(args, fmt);
 -     std::vfprintf(stderr, fmt, args);
 -     std::fprintf(stderr, "\n");
 -     va_end(args);
 - }
 - 
 - static inline
 - void d_stderr2(const char* const fmt, ...)
 - {
 -     va_list args;
 -     va_start(args, fmt);
 -     std::fprintf(stderr, "\x1b[31m");
 -     std::vfprintf(stderr, fmt, args);
 -     std::fprintf(stderr, "\x1b[0m\n");
 -     va_end(args);
 - }
 - 
 - // -----------------------------------------------------------------------
 - // d_*sleep
 - 
 - static inline
 - void d_sleep(unsigned int secs)
 - {
 - #ifdef DISTRHO_OS_WINDOWS
 -     Sleep(secs * 1000);
 - #else
 -     sleep(secs);
 - #endif
 - }
 - 
 - static inline
 - void d_msleep(unsigned int msecs)
 - {
 - #ifdef DISTRHO_OS_WINDOWS
 -     Sleep(msecs);
 - #else
 -     usleep(msecs * 1000);
 - #endif
 - }
 - 
 - // -----------------------------------------------------------------------
 - // d_string class
 - 
 - class d_string
 - {
 - public:
 -     // -------------------------------------------------------------------
 -     // constructors (no explicit conversions allowed)
 - 
 -     /*
 -      * Empty string.
 -      */
 -     explicit d_string()
 -     {
 -         _init();
 -         _dup(nullptr);
 -     }
 - 
 -     /*
 -      * Simple character.
 -      */
 -     explicit d_string(const char c)
 -     {
 -         char ch[2];
 -         ch[0] = c;
 -         ch[1] = '\0';
 - 
 -         _init();
 -         _dup(ch);
 -     }
 - 
 -     /*
 -      * Simple char string.
 -      */
 -     explicit d_string(char* const strBuf)
 -     {
 -         _init();
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Simple const char string.
 -      */
 -     explicit d_string(const char* const strBuf)
 -     {
 -         _init();
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Integer.
 -      */
 -     explicit d_string(const int value)
 -     {
 -         char strBuf[0xff+1];
 -         std::memset(strBuf, 0, (0xff+1)*sizeof(char));
 -         std::snprintf(strBuf, 0xff, "%d", value);
 - 
 -         _init();
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Unsigned integer, possibly in hexadecimal.
 -      */
 -     explicit d_string(const unsigned int value, const bool hexadecimal = false)
 -     {
 -         char strBuf[0xff+1];
 -         std::memset(strBuf, 0, (0xff+1)*sizeof(char));
 -         std::snprintf(strBuf, 0xff, hexadecimal ? "0x%x" : "%u", value);
 - 
 -         _init();
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Long integer.
 -      */
 -     explicit d_string(const long int value)
 -     {
 -         char strBuf[0xff+1];
 -         std::memset(strBuf, 0, (0xff+1)*sizeof(char));
 -         std::snprintf(strBuf, 0xff, "%ld", value);
 - 
 -         _init();
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Long unsigned integer, possibly hexadecimal.
 -      */
 -     explicit d_string(const unsigned long int value, const bool hexadecimal = false)
 -     {
 -         char strBuf[0xff+1];
 -         std::memset(strBuf, 0, (0xff+1)*sizeof(char));
 -         std::snprintf(strBuf, 0xff, hexadecimal ? "0x%lx" : "%lu", value);
 - 
 -         _init();
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Single-precision floating point number.
 -      */
 -     explicit d_string(const float value)
 -     {
 -         char strBuf[0xff+1];
 -         std::memset(strBuf, 0, (0xff+1)*sizeof(char));
 -         std::snprintf(strBuf, 0xff, "%f", value);
 - 
 -         _init();
 -         _dup(strBuf);
 -     }
 - 
 -     /*
 -      * Double-precision floating point number.
 -      */
 -     explicit d_string(const double value)
 -     {
 -         char strBuf[0xff+1];
 -         std::memset(strBuf, 0, (0xff+1)*sizeof(char));
 -         std::snprintf(strBuf, 0xff, "%g", value);
 - 
 -         _init();
 -         _dup(strBuf);
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // non-explicit constructor
 - 
 -     /*
 -      * Create string from another string.
 -      */
 -     d_string(const d_string& str)
 -     {
 -         _init();
 -         _dup(str.fBuffer);
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // destructor
 - 
 -     /*
 -      * Destructor.
 -      */
 -     ~d_string()
 -     {
 -         assert(fBuffer != nullptr);
 - 
 -         delete[] fBuffer;
 -         fBuffer = nullptr;
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // public methods
 - 
 -     /*
 -      * Get length of the string.
 -      */
 -     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 another string, optionally ignoring case.
 -      */
 -     bool contains(const char* const strBuf, const bool ignoreCase = false) const
 -     {
 -         if (strBuf == nullptr)
 -             return false;
 - 
 -         if (ignoreCase)
 -         {
 - #ifdef __USE_GNU
 -             return (strcasestr(fBuffer, strBuf) != nullptr);
 - #else
 -             d_string tmp1(fBuffer), tmp2(strBuf);
 -             tmp1.toLower();
 -             tmp2.toLower();
 -             return (std::strstr((const char*)tmp1, (const char*)tmp2) != nullptr);
 - #endif
 -         }
 - 
 -         return (std::strstr(fBuffer, strBuf) != nullptr);
 -     }
 - 
 -     /*
 -      * Overloaded function.
 -      */
 -     bool contains(const d_string& str, const bool ignoreCase = false) const
 -     {
 -         return contains(str.fBuffer, ignoreCase);
 -     }
 - 
 -     /*
 -      * Check if character at 'pos' is a digit.
 -      */
 -     bool isDigit(const size_t pos) const noexcept
 -     {
 -         if (pos >= fBufferLen)
 -             return false;
 - 
 -         return (fBuffer[pos] >= '0' && fBuffer[pos] <= '9');
 -     }
 - 
 -     /*
 -      * Check if the string starts with the character 'c'.
 -      */
 -     bool startsWith(const char c) const
 -     {
 -         if (c == '\0')
 -             return false;
 - 
 -         return (fBufferLen > 0 && fBuffer[0] == c);
 -     }
 - 
 -     /*
 -      * Check if the string starts with the string 'prefix'.
 -      */
 -     bool startsWith(const char* const prefix) const
 -     {
 -         if (prefix == nullptr)
 -             return false;
 - 
 -         const size_t prefixLen(std::strlen(prefix));
 - 
 -         if (fBufferLen < prefixLen)
 -             return false;
 - 
 -         return (std::strncmp(fBuffer + (fBufferLen-prefixLen), prefix, prefixLen) == 0);
 -     }
 - 
 -     /*
 -      * Check if the string ends with the character 'c'.
 -      */
 -     bool endsWith(const char c) const
 -     {
 -         if (c == '\0')
 -             return false;
 - 
 -         return (fBufferLen > 0 && fBuffer[fBufferLen] == c);
 -     }
 - 
 -     /*
 -      * Check if the string ends with the string 'suffix'.
 -      */
 -     bool endsWith(const char* const suffix) const
 -     {
 -         if (suffix == nullptr)
 -             return false;
 - 
 -         const size_t suffixLen(std::strlen(suffix));
 - 
 -         if (fBufferLen < suffixLen)
 -             return false;
 - 
 -         return (std::strncmp(fBuffer + (fBufferLen-suffixLen), suffix, suffixLen) == 0);
 -     }
 - 
 -     /*
 -      * Clear the string.
 -      */
 -     void clear() noexcept
 -     {
 -         truncate(0);
 -     }
 - 
 -     /*
 -      * Replace all occurrences of character 'before' with character 'after'.
 -      */
 -     void replace(const char before, const char after) noexcept
 -     {
 -         if (before == '\0' || after == '\0')
 -             return;
 - 
 -         for (size_t i=0; i < fBufferLen; ++i)
 -         {
 -             if (fBuffer[i] == before)
 -                 fBuffer[i] = after;
 -             else if (fBuffer[i] == '\0')
 -                 break;
 -         }
 -     }
 - 
 -     /*
 -      * Truncate the string to size 'n'.
 -      */
 -     void truncate(const size_t n) noexcept
 -     {
 -         if (n >= fBufferLen)
 -             return;
 - 
 -         for (size_t i=n; i < fBufferLen; ++i)
 -             fBuffer[i] = '\0';
 - 
 -         fBufferLen = n;
 -     }
 - 
 -     /*
 -      * Convert all non-basic characters to '_'.
 -      */
 -     void toBasic() noexcept
 -     {
 -         for (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] = '_';
 -         }
 -     }
 - 
 -     /*
 -      * Convert to all ascii characters to lowercase.
 -      */
 -     void toLower() noexcept
 -     {
 -         static const char kCharDiff('a' - 'A');
 - 
 -         for (size_t i=0; i < fBufferLen; ++i)
 -         {
 -             if (fBuffer[i] >= 'A' && fBuffer[i] <= 'Z')
 -                 fBuffer[i] = static_cast<char>(fBuffer[i] + kCharDiff);
 -         }
 -     }
 - 
 -     /*
 -      * Convert to all ascii characters to uppercase.
 -      */
 -     void toUpper() noexcept
 -     {
 -         static const char kCharDiff('a' - 'A');
 - 
 -         for (size_t i=0; i < fBufferLen; ++i)
 -         {
 -             if (fBuffer[i] >= 'a' && fBuffer[i] <= 'z')
 -                 fBuffer[i] = static_cast<char>(fBuffer[i] - kCharDiff);
 -         }
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // public operators
 - 
 -     operator const char*() const noexcept
 -     {
 -         return fBuffer;
 -     }
 - 
 -     char& operator[](const size_t pos) const noexcept
 -     {
 -         return fBuffer[pos];
 -     }
 - 
 -     bool operator==(const char* const strBuf) const
 -     {
 -         return (strBuf != nullptr && std::strcmp(fBuffer, strBuf) == 0);
 -     }
 - 
 -     bool operator==(const d_string& str) const
 -     {
 -         return operator==(str.fBuffer);
 -     }
 - 
 -     bool operator!=(const char* const strBuf) const
 -     {
 -         return !operator==(strBuf);
 -     }
 - 
 -     bool operator!=(const d_string& str) const
 -     {
 -         return !operator==(str.fBuffer);
 -     }
 - 
 -     d_string& operator=(const char* const strBuf)
 -     {
 -         _dup(strBuf);
 - 
 -         return *this;
 -     }
 - 
 -     d_string& operator=(const d_string& str)
 -     {
 -         return operator=(str.fBuffer);
 -     }
 - 
 -     d_string& operator+=(const char* const strBuf)
 -     {
 -         if (strBuf == nullptr)
 -             return *this;
 - 
 -         const size_t newBufSize = fBufferLen + std::strlen(strBuf) + 1;
 -         char         newBuf[newBufSize];
 - 
 -         std::strcpy(newBuf, fBuffer);
 -         std::strcat(newBuf, strBuf);
 - 
 -         _dup(newBuf, newBufSize-1);
 - 
 -         return *this;
 -     }
 - 
 -     d_string& operator+=(const d_string& str)
 -     {
 -         return operator+=(str.fBuffer);
 -     }
 - 
 -     d_string operator+(const char* const strBuf)
 -     {
 -         const size_t newBufSize = fBufferLen + ((strBuf != nullptr) ? std::strlen(strBuf) : 0) + 1;
 -         char         newBuf[newBufSize];
 - 
 -         std::strcpy(newBuf, fBuffer);
 - 
 -         if (strBuf != nullptr)
 -             std::strcat(newBuf, strBuf);
 - 
 -         return d_string(newBuf);
 -     }
 - 
 -     d_string operator+(const d_string& str)
 -     {
 -         return operator+(str.fBuffer);
 -     }
 - 
 -     // -------------------------------------------------------------------
 - 
 - private:
 -     char*  fBuffer;    // the actual string buffer
 -     size_t fBufferLen; // string length
 -     bool   fFirstInit; // true when first initiated
 - 
 -     /*
 -      * Shared init function.
 -      * Called on all constructors.
 -      */
 -     void _init() noexcept
 -     {
 -         fBuffer    = nullptr;
 -         fBufferLen = 0;
 -         fFirstInit = true;
 -     }
 - 
 -     /*
 -      * Helper function.
 -      * Called whenever the string needs to be allocated.
 -      *
 -      * Notes:
 -      * - Allocates string only if first initiated, or 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 size_t size = 0)
 -     {
 -         if (strBuf != nullptr)
 -         {
 -             // don't recreate string if contents match
 -             if (fFirstInit || std::strcmp(fBuffer, strBuf) != 0)
 -             {
 -                 if (! fFirstInit)
 -                 {
 -                     assert(fBuffer != nullptr);
 -                     delete[] fBuffer;
 -                 }
 - 
 -                 fBufferLen = (size > 0) ? size : std::strlen(strBuf);
 -                 fBuffer    = new char[fBufferLen+1];
 - 
 -                 std::strcpy(fBuffer, strBuf);
 - 
 -                 fBuffer[fBufferLen] = '\0';
 - 
 -                 fFirstInit = false;
 -             }
 -         }
 -         else
 -         {
 -             assert(size == 0);
 - 
 -             // don't recreate null string
 -             if (fFirstInit || fBufferLen != 0)
 -             {
 -                 if (! fFirstInit)
 -                 {
 -                     assert(fBuffer != nullptr);
 -                     delete[] fBuffer;
 -                 }
 - 
 -                 fBufferLen = 0;
 -                 fBuffer    = new char[1];
 -                 fBuffer[0] = '\0';
 - 
 -                 fFirstInit = false;
 -             }
 -         }
 -     }
 - };
 - 
 - // -----------------------------------------------------------------------
 - 
 - static inline
 - d_string operator+(const d_string& strBefore, const char* const strBufAfter)
 - {
 -     const char* const strBufBefore = (const char*)strBefore;
 -     const size_t newBufSize        = strBefore.length() + ((strBufAfter != nullptr) ? std::strlen(strBufAfter) : 0) + 1;
 -     char newBuf[newBufSize];
 - 
 -     std::strcpy(newBuf, strBufBefore);
 -     std::strcat(newBuf, strBufAfter);
 - 
 -     return d_string(newBuf);
 - }
 - 
 - static inline
 - d_string operator+(const char* const strBufBefore, const d_string& strAfter)
 - {
 -     const char* const strBufAfter = (const char*)strAfter;
 -     const size_t      newBufSize  = ((strBufBefore != nullptr) ? std::strlen(strBufBefore) : 0) + strAfter.length() + 1;
 -     char newBuf[newBufSize];
 - 
 -     std::strcpy(newBuf, strBufBefore);
 -     std::strcat(newBuf, strBufAfter);
 - 
 -     return d_string(newBuf);
 - }
 - 
 - // -----------------------------------------------------------------------
 - 
 - #endif // DISTRHO_UTILS_HPP_INCLUDED
 
 
  |