Large rework of the project, specifically the String classes.
continuous-integration/drone/push Build is passing
Details
continuous-integration/drone/push Build is passing
Details
This commit is contained in:
parent
8eba3b28ff
commit
abca51d331
|
@ -1,4 +1,5 @@
|
||||||
/cmake-build-debug/
|
/cmake-build-debug/
|
||||||
|
/cmake-build-debug-coverage/
|
||||||
/cmake-build-release/
|
/cmake-build-release/
|
||||||
/build-release-windows/
|
/build-release-windows/
|
||||||
/.idea/
|
/.idea/
|
|
@ -2,7 +2,7 @@ cmake_minimum_required(VERSION 3.13)
|
||||||
project(Arbutils)
|
project(Arbutils)
|
||||||
|
|
||||||
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall -Werror")
|
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall -Werror")
|
||||||
set(CMAKE_CXX_STANDARD 17)
|
set(CMAKE_CXX_STANDARD 20)
|
||||||
|
|
||||||
if (WINDOWS)
|
if (WINDOWS)
|
||||||
ADD_DEFINITIONS(-D WINDOWS=1)
|
ADD_DEFINITIONS(-D WINDOWS=1)
|
||||||
|
@ -11,7 +11,7 @@ endif (WINDOWS)
|
||||||
file(GLOB_RECURSE SRC_FILES "src/*.cpp" "src/*.hpp")
|
file(GLOB_RECURSE SRC_FILES "src/*.cpp" "src/*.hpp")
|
||||||
set(LIBTYPE STATIC)
|
set(LIBTYPE STATIC)
|
||||||
if (SHARED)
|
if (SHARED)
|
||||||
set(LIBTYPE SHARED)
|
set(LIBTYPE SHARED src/StringView.hpp src/String/BasicStringView.hpp src/String/StringViewLiteral.hpp)
|
||||||
endif(SHARED)
|
endif(SHARED)
|
||||||
add_library(Arbutils ${LIBTYPE} ${SRC_FILES})
|
add_library(Arbutils ${LIBTYPE} ${SRC_FILES})
|
||||||
|
|
||||||
|
|
|
@ -1,4 +0,0 @@
|
||||||
#include "ConstString.hpp"
|
|
||||||
|
|
||||||
ArbUt::__ConstStringCharHolder* ArbUt::CaseInsensitiveConstString::__emptyString = new __ConstStringCharHolder("", 0);
|
|
||||||
ArbUt::__ConstStringCharHolder* ArbUt::ConstString::__emptyString = new __ConstStringCharHolder("", 0);
|
|
|
@ -1,65 +0,0 @@
|
||||||
#ifndef ARBUTILS_CONSTSTRING_HPP
|
|
||||||
#define ARBUTILS_CONSTSTRING_HPP
|
|
||||||
|
|
||||||
#include <atomic>
|
|
||||||
#include <cstddef>
|
|
||||||
#include <cstdint>
|
|
||||||
#include <cstring>
|
|
||||||
#include <string>
|
|
||||||
#include "__ConstStringCore.hpp"
|
|
||||||
|
|
||||||
namespace ArbUt {
|
|
||||||
class __ConstStringCharHolder {
|
|
||||||
char* _value;
|
|
||||||
std::atomic<size_t> _references;
|
|
||||||
|
|
||||||
__ConstStringCharHolder(const __ConstStringCharHolder& o) = delete;
|
|
||||||
__ConstStringCharHolder& operator=(const __ConstStringCharHolder& other) = delete;
|
|
||||||
|
|
||||||
public:
|
|
||||||
__ConstStringCharHolder(const char* value, size_t size) noexcept : _value(new char[size + 1]), _references(1) {
|
|
||||||
strncpy(_value, value, size + 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
~__ConstStringCharHolder() noexcept { delete[] _value; }
|
|
||||||
|
|
||||||
inline void RemoveReference() noexcept {
|
|
||||||
if (--_references <= 0) {
|
|
||||||
delete this;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
inline void AddReference() noexcept { _references++; }
|
|
||||||
inline constexpr const char* GetValue() const noexcept { return _value; }
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
ConstStringCore(
|
|
||||||
ConstString,
|
|
||||||
inline static uint32_t constexpr Hash(char const* input) {
|
|
||||||
return (*input) ? static_cast<uint32_t>((*input)) + 33 * Hash(input + 1) : 5381;
|
|
||||||
};)
|
|
||||||
|
|
||||||
inline constexpr ArbUt::ConstString_Literal
|
|
||||||
operator"" _const(const char* c, size_t l) {
|
|
||||||
return ArbUt::ConstString_Literal(c, l);
|
|
||||||
}
|
|
||||||
inline constexpr ArbUt::ConstString_Literal operator"" _c(const char* c, size_t l) {
|
|
||||||
return ArbUt::ConstString_Literal(c, l);
|
|
||||||
}
|
|
||||||
|
|
||||||
ConstStringCore(
|
|
||||||
CaseInsensitiveConstString,
|
|
||||||
inline static constexpr char charToLower(const char c) {
|
|
||||||
return (c >= 'A' && c <= 'Z') ? c + ('a' - 'A') : c;
|
|
||||||
} inline static uint32_t constexpr Hash(char const* input) {
|
|
||||||
return charToLower(*input) ? static_cast<uint32_t>(charToLower(*input)) + 33 * Hash(input + 1) : 5381;
|
|
||||||
};);
|
|
||||||
|
|
||||||
inline constexpr ArbUt::CaseInsensitiveConstString_Literal operator"" _const_nocase(const char* c, size_t l) {
|
|
||||||
return ArbUt::CaseInsensitiveConstString_Literal(c, l);
|
|
||||||
}
|
|
||||||
inline constexpr ArbUt::CaseInsensitiveConstString_Literal operator"" _cnc(const char* c, size_t l) {
|
|
||||||
return ArbUt::CaseInsensitiveConstString_Literal(c, l);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // ARBUTILS_CONSTSTRING_HPP
|
|
12
src/Enum.hpp
12
src/Enum.hpp
|
@ -4,12 +4,13 @@
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include "MacroUtils.hpp"
|
#include "MacroUtils.hpp"
|
||||||
|
#include "StringView.hpp"
|
||||||
|
|
||||||
#define ENUM_VALUE(x, value) x = value,
|
#define ENUM_VALUE(x, value) x = value,
|
||||||
#
|
#
|
||||||
#define ENUM_CASE(x, name) \
|
#define ENUM_CASE(x, name) \
|
||||||
case name::x: \
|
case name::x: \
|
||||||
return #x;
|
return ArbUt::StringViewLiteral(#x);
|
||||||
#
|
#
|
||||||
#define ENUM_PARSE_CASE(x, name) \
|
#define ENUM_PARSE_CASE(x, name) \
|
||||||
case ConstHash(#x): \
|
case ConstHash(#x): \
|
||||||
|
@ -64,14 +65,9 @@
|
||||||
} \
|
} \
|
||||||
\
|
\
|
||||||
public: \
|
public: \
|
||||||
constexpr static const char* ToString(name value) noexcept { \
|
constexpr static ArbUt::StringViewLiteral ToString(name value) noexcept { \
|
||||||
switch (value) { MACRO_UTILS_FOR_EACH(ENUM_CASE, name, values) } \
|
switch (value) { MACRO_UTILS_FOR_EACH(ENUM_CASE, name, values) } \
|
||||||
/*If we haven't found a value, we want to stringify the number*/ \
|
return "out of bounds"_cnc; \
|
||||||
auto v = static_cast<type>(value); \
|
|
||||||
auto size = (int)((ceil(log10(v)) + 1) * sizeof(char)); \
|
|
||||||
char* snum = new char[size + 1]; \
|
|
||||||
sprintf(snum, "%d", v); \
|
|
||||||
return snum; \
|
|
||||||
} \
|
} \
|
||||||
constexpr static name Parse(const char* input, bool caseInsensitive = false) { \
|
constexpr static name Parse(const char* input, bool caseInsensitive = false) { \
|
||||||
if (caseInsensitive) \
|
if (caseInsensitive) \
|
||||||
|
|
|
@ -0,0 +1,35 @@
|
||||||
|
#ifndef ARBUTILS_BASICSTRINGVIEW_HPP
|
||||||
|
#define ARBUTILS_BASICSTRINGVIEW_HPP
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
namespace ArbUt {
|
||||||
|
class BasicStringView {
|
||||||
|
protected:
|
||||||
|
size_t _length;
|
||||||
|
uint32_t _hash;
|
||||||
|
|
||||||
|
constexpr BasicStringView(size_t length, uint32_t hash) : _length(length), _hash(hash) {}
|
||||||
|
public:
|
||||||
|
|
||||||
|
[[nodiscard]] inline constexpr size_t Length() const noexcept { return _length; }
|
||||||
|
[[nodiscard]] inline constexpr uint32_t GetHash() const noexcept { return _hash; }
|
||||||
|
[[nodiscard]] inline constexpr std::size_t operator()(BasicStringView const& s) const noexcept { return _hash; }
|
||||||
|
[[nodiscard]] inline constexpr operator uint32_t() const noexcept { return _hash; }
|
||||||
|
[[nodiscard]] inline constexpr bool operator==(const BasicStringView& rhs) const noexcept {
|
||||||
|
return _hash == rhs._hash;
|
||||||
|
}
|
||||||
|
inline constexpr bool operator!=(const BasicStringView& rhs) const noexcept { return _hash != rhs._hash; }
|
||||||
|
inline constexpr bool Empty() const noexcept { return Length() == 0; }
|
||||||
|
|
||||||
|
[[nodiscard]] virtual constexpr const char* c_str() const noexcept = 0;
|
||||||
|
[[nodiscard]] virtual constexpr std::string_view std_str() const noexcept = 0;
|
||||||
|
|
||||||
|
virtual constexpr bool operator==(const std::string_view& rhs) const noexcept = 0;
|
||||||
|
virtual constexpr bool operator!=(const std::string_view& rhs) const noexcept = 0;
|
||||||
|
virtual constexpr bool operator==(const char* rhs) const noexcept = 0;
|
||||||
|
virtual constexpr bool operator!=(const char* rhs) const noexcept = 0;
|
||||||
|
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // ARBUTILS_BASICSTRINGVIEW_HPP
|
|
@ -0,0 +1,4 @@
|
||||||
|
#include "StringView.hpp"
|
||||||
|
|
||||||
|
ArbUt::__ConstStringCharHolder* ArbUt::StringView::__emptyString =
|
||||||
|
new __ConstStringCharHolder(std::string_view("", 0));
|
|
@ -0,0 +1,104 @@
|
||||||
|
#ifndef ARBUTILS_STRINGVIEW_HPP
|
||||||
|
#define ARBUTILS_STRINGVIEW_HPP
|
||||||
|
|
||||||
|
#include <atomic>
|
||||||
|
#include <cstddef>
|
||||||
|
#include <cstdint>
|
||||||
|
#include <cstring>
|
||||||
|
#include <string>
|
||||||
|
#include "BasicStringView.hpp"
|
||||||
|
|
||||||
|
#if WINDOWS
|
||||||
|
#define STDSTRINGCONSTEXPR
|
||||||
|
#else
|
||||||
|
#define STDSTRINGCONSTEXPR constexpr
|
||||||
|
#endif
|
||||||
|
|
||||||
|
namespace ArbUt {
|
||||||
|
class __ConstStringCharHolder {
|
||||||
|
std::string_view _value;
|
||||||
|
std::atomic<size_t> _references;
|
||||||
|
|
||||||
|
__ConstStringCharHolder(const __ConstStringCharHolder& o) = delete;
|
||||||
|
__ConstStringCharHolder& operator=(const __ConstStringCharHolder& other) = delete;
|
||||||
|
|
||||||
|
public:
|
||||||
|
__ConstStringCharHolder(const std::string_view& value) noexcept : _value(value), _references(1) {}
|
||||||
|
|
||||||
|
inline void RemoveReference() noexcept {
|
||||||
|
if (--_references <= 0) {
|
||||||
|
delete this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
inline void AddReference() noexcept { _references++; }
|
||||||
|
inline constexpr const std::string_view& GetValue() const noexcept { return _value; }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
inline static constexpr char charToLower(const char c) { return (c >= 'A' && c <= 'Z') ? c + ('a' - 'A') : c; }
|
||||||
|
static uint32_t constexpr Hash(char const* input) {
|
||||||
|
return charToLower(*input) ? static_cast<uint32_t>(charToLower(*input)) + 33 * Hash(input + 1) : 5381;
|
||||||
|
};
|
||||||
|
static int constexpr CalcLength(const char* str) { return *str ? 1 + CalcLength(str + 1) : 0; }
|
||||||
|
|
||||||
|
namespace ArbUt {
|
||||||
|
class StringView : public BasicStringView {
|
||||||
|
private:
|
||||||
|
static __ConstStringCharHolder* __emptyString;
|
||||||
|
static inline __ConstStringCharHolder* GetEmptyString() { return __emptyString; }
|
||||||
|
|
||||||
|
__ConstStringCharHolder* _str;
|
||||||
|
|
||||||
|
inline __ConstStringCharHolder* CloneHolder() const noexcept {
|
||||||
|
_str->AddReference();
|
||||||
|
return _str;
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
StringView(const std::string_view& str) noexcept
|
||||||
|
: BasicStringView(str.length(), Hash(str.data())), _str(new __ConstStringCharHolder(str)) {}
|
||||||
|
|
||||||
|
StringView() noexcept : BasicStringView(0, Hash("")), _str(GetEmptyString()) {
|
||||||
|
GetEmptyString()->AddReference();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Copy operators */
|
||||||
|
StringView(const StringView& other) noexcept
|
||||||
|
: BasicStringView(other._length, other._hash), _str(other.CloneHolder()) {}
|
||||||
|
StringView& operator=(const StringView& other) noexcept {
|
||||||
|
if (_str == other._str) {
|
||||||
|
_str->AddReference();
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
_str->RemoveReference();
|
||||||
|
_str = other.CloneHolder();
|
||||||
|
_length = other._length;
|
||||||
|
_hash = other._hash;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
~StringView() noexcept { _str->RemoveReference(); }
|
||||||
|
|
||||||
|
[[nodiscard]] inline constexpr const char* c_str() const noexcept override { return _str->GetValue().data(); }
|
||||||
|
[[nodiscard]] inline std::string_view std_str() const noexcept override { return _str->GetValue(); }
|
||||||
|
|
||||||
|
inline constexpr bool operator==(const std::string_view& rhs) const noexcept override {
|
||||||
|
return _hash == Hash(rhs.data());
|
||||||
|
}
|
||||||
|
inline constexpr bool operator!=(const std::string_view& rhs) const noexcept override {
|
||||||
|
return _hash != Hash(rhs.data());
|
||||||
|
}
|
||||||
|
inline constexpr bool operator==(const char* rhs) const noexcept override { return _hash == Hash(rhs); }
|
||||||
|
inline constexpr bool operator!=(const char* rhs) const noexcept override { return _hash != Hash(rhs); }
|
||||||
|
|
||||||
|
inline static constexpr uint32_t CalculateHash(const char* val) noexcept { return Hash(val); }
|
||||||
|
inline static constexpr uint32_t CalculateHash(const std::string_view& val) noexcept { return Hash(val.data()); }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace std {
|
||||||
|
template <> struct hash<ArbUt::StringView> {
|
||||||
|
constexpr std::size_t operator()(ArbUt::StringView const& s) const noexcept { return s.GetHash(); }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
#endif // ARBUTILS_STRINGVIEW_HPP
|
|
@ -0,0 +1,46 @@
|
||||||
|
#ifndef ARBUTILS_STRINGVIEWLITERAL_HPP
|
||||||
|
#define ARBUTILS_STRINGVIEWLITERAL_HPP
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
|
#include "BasicStringView.hpp"
|
||||||
|
#include "StringView.hpp"
|
||||||
|
|
||||||
|
namespace ArbUt {
|
||||||
|
class StringViewLiteral : public BasicStringView {
|
||||||
|
private:
|
||||||
|
const char* _str;
|
||||||
|
|
||||||
|
public:
|
||||||
|
constexpr StringViewLiteral(const char* str, size_t size) noexcept
|
||||||
|
: BasicStringView(size, Hash(str)), _str(str) {}
|
||||||
|
constexpr StringViewLiteral(const char* str) noexcept : StringViewLiteral(str, CalcLength(str)){};
|
||||||
|
[[nodiscard]] inline constexpr const char* c_str() const noexcept { return _str; }
|
||||||
|
constexpr std::string_view std_str() const noexcept { return std::string_view(_str, _length); }
|
||||||
|
constexpr std::size_t operator()(StringViewLiteral const& s) const noexcept { return s.GetHash(); }
|
||||||
|
inline operator StringView() const noexcept { return StringView(std::string_view(_str, _length)); }
|
||||||
|
|
||||||
|
inline constexpr bool operator==(const std::string_view& rhs) const noexcept {
|
||||||
|
return _hash == Hash(rhs.data());
|
||||||
|
}
|
||||||
|
inline constexpr bool operator!=(const std::string_view& rhs) const noexcept {
|
||||||
|
return _hash != Hash(rhs.data());
|
||||||
|
}
|
||||||
|
inline constexpr bool operator==(const char* rhs) const noexcept { return _hash == Hash(rhs); }
|
||||||
|
inline constexpr bool operator!=(const char* rhs) const noexcept { return _hash != Hash(rhs); }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace std {
|
||||||
|
template <> struct hash<ArbUt::StringViewLiteral> {
|
||||||
|
constexpr std::size_t operator()(ArbUt::StringViewLiteral const& s) const noexcept { return s.GetHash(); }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
inline constexpr ArbUt::StringViewLiteral operator"" _const_nocase(const char* c, size_t l) {
|
||||||
|
return ArbUt::StringViewLiteral(c, l);
|
||||||
|
}
|
||||||
|
inline constexpr ArbUt::StringViewLiteral operator"" _cnc(const char* c, size_t l) {
|
||||||
|
return ArbUt::StringViewLiteral(c, l);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // ARBUTILS_STRINGVIEWLITERAL_HPP
|
|
@ -0,0 +1,6 @@
|
||||||
|
#include "String/StringView.hpp"
|
||||||
|
#include "String/StringViewLiteral.hpp"
|
||||||
|
|
||||||
|
namespace ArbUt {
|
||||||
|
[[deprecated("Moved to StringView")]] typedef StringView CaseInsensitiveConstString;
|
||||||
|
}
|
|
@ -1,131 +0,0 @@
|
||||||
#ifndef ARBUTILS_CONSTSTRINGCORE_HPP
|
|
||||||
#define ARBUTILS_CONSTSTRINGCORE_HPP
|
|
||||||
|
|
||||||
#if WINDOWS
|
|
||||||
#define STDSTRINGCONSTEXPR
|
|
||||||
#else
|
|
||||||
#define STDSTRINGCONSTEXPR constexpr
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define ConstStringCore(name, hashFunction) \
|
|
||||||
namespace ArbUt { \
|
|
||||||
class name { \
|
|
||||||
private: \
|
|
||||||
__ConstStringCharHolder* _str; \
|
|
||||||
size_t _length; \
|
|
||||||
uint32_t _hash; \
|
|
||||||
hashFunction; \
|
|
||||||
\
|
|
||||||
static __ConstStringCharHolder* __emptyString; \
|
|
||||||
static inline __ConstStringCharHolder* GetEmptyString() { return __emptyString; } \
|
|
||||||
\
|
|
||||||
inline static int constexpr Length(const char* str) { return *str ? 1 + Length(str + 1) : 0; } \
|
|
||||||
inline __ConstStringCharHolder* CloneHolder() const noexcept { \
|
|
||||||
_str->AddReference(); \
|
|
||||||
return _str; \
|
|
||||||
} \
|
|
||||||
\
|
|
||||||
public: \
|
|
||||||
name(const char* str, size_t size) noexcept \
|
|
||||||
: _str(new __ConstStringCharHolder(str, size)), _length(size), _hash(Hash(str)) {} \
|
|
||||||
\
|
|
||||||
name() noexcept : _str(GetEmptyString()), _length(0), _hash(Hash("")) { \
|
|
||||||
GetEmptyString()->AddReference(); \
|
|
||||||
}; \
|
|
||||||
explicit name(const char* str) noexcept : name(str, Length(str)){}; \
|
|
||||||
explicit name(const std::string& str) noexcept : name(str.c_str(), str.size()){}; \
|
|
||||||
\
|
|
||||||
/* Copy operators */ \
|
|
||||||
name(const name& other) noexcept \
|
|
||||||
: _str(other.CloneHolder()), _length(other._length), _hash(other._hash) {} \
|
|
||||||
name& operator=(const name& other) noexcept { \
|
|
||||||
if (_str == other._str) { \
|
|
||||||
_str->AddReference(); \
|
|
||||||
return *this; \
|
|
||||||
} \
|
|
||||||
_str->RemoveReference(); \
|
|
||||||
_str = other.CloneHolder(); \
|
|
||||||
_length = other._length; \
|
|
||||||
_hash = other._hash; \
|
|
||||||
return *this; \
|
|
||||||
} \
|
|
||||||
\
|
|
||||||
~name() noexcept { _str->RemoveReference(); } \
|
|
||||||
\
|
|
||||||
[[nodiscard]] inline constexpr const char* c_str() const noexcept { return _str->GetValue(); } \
|
|
||||||
[[nodiscard]] inline std::string std_str() const noexcept { \
|
|
||||||
return std::string(_str->GetValue(), _length); \
|
|
||||||
} \
|
|
||||||
\
|
|
||||||
[[nodiscard]] inline constexpr size_t Length() const noexcept { return _length; } \
|
|
||||||
\
|
|
||||||
[[nodiscard]] inline constexpr uint32_t GetHash() const noexcept { return _hash; } \
|
|
||||||
inline constexpr bool Empty() const noexcept { return _length == 0; } \
|
|
||||||
\
|
|
||||||
constexpr std::size_t operator()(name const& s) const noexcept { return s.GetHash(); } \
|
|
||||||
inline constexpr operator uint32_t() const noexcept { return _hash; } \
|
|
||||||
\
|
|
||||||
inline constexpr bool operator==(const name& rhs) const noexcept { return _hash == rhs._hash; } \
|
|
||||||
inline constexpr bool operator!=(const name& rhs) const noexcept { return _hash != rhs._hash; } \
|
|
||||||
inline STDSTRINGCONSTEXPR bool operator==(const std::string& rhs) const noexcept { \
|
|
||||||
return _hash == Hash(rhs.c_str()); \
|
|
||||||
} \
|
|
||||||
inline STDSTRINGCONSTEXPR bool operator!=(const std::string& rhs) const noexcept { \
|
|
||||||
return _hash != Hash(rhs.c_str()); \
|
|
||||||
} \
|
|
||||||
inline constexpr bool operator==(const char* rhs) const noexcept { return _hash == Hash(rhs); } \
|
|
||||||
inline constexpr bool operator!=(const char* rhs) const noexcept { return _hash != Hash(rhs); } \
|
|
||||||
\
|
|
||||||
inline static constexpr uint32_t GetHash(const char* val) noexcept { return Hash(val); } \
|
|
||||||
inline static STDSTRINGCONSTEXPR uint32_t GetHash(const std::string& val) noexcept { \
|
|
||||||
return Hash(val.c_str()); \
|
|
||||||
} \
|
|
||||||
}; \
|
|
||||||
\
|
|
||||||
class name##_Literal { \
|
|
||||||
private: \
|
|
||||||
const char* _str; \
|
|
||||||
size_t _length; \
|
|
||||||
uint32_t _hash; \
|
|
||||||
\
|
|
||||||
hashFunction; \
|
|
||||||
\
|
|
||||||
inline static int constexpr Length(const char* str) noexcept { return *str ? 1 + Length(str + 1) : 0; } \
|
|
||||||
\
|
|
||||||
public: \
|
|
||||||
constexpr name##_Literal(const char* str, size_t size) noexcept \
|
|
||||||
: _str(str), _length(size), _hash(Hash(str)) {} \
|
|
||||||
explicit name##_Literal(const char* str) noexcept : name##_Literal(str, Length(str)){}; \
|
|
||||||
[[nodiscard]] inline constexpr const char* c_str() const noexcept { return _str; } \
|
|
||||||
[[nodiscard]] inline constexpr size_t Length() const noexcept { return _length; } \
|
|
||||||
\
|
|
||||||
[[nodiscard]] inline constexpr uint32_t GetHash() const noexcept { return _hash; } \
|
|
||||||
inline constexpr bool Empty() const noexcept { return _length == 0; } \
|
|
||||||
\
|
|
||||||
constexpr std::size_t operator()(name##_Literal const& s) const noexcept { return s.GetHash(); } \
|
|
||||||
inline constexpr operator uint32_t() const noexcept { return _hash; } \
|
|
||||||
inline operator name() const noexcept { return name(_str, _length); } \
|
|
||||||
\
|
|
||||||
inline constexpr bool operator==(const name##_Literal& rhs) const noexcept { return _hash == rhs._hash; } \
|
|
||||||
inline constexpr bool operator!=(const name##_Literal& rhs) const noexcept { return _hash != rhs._hash; } \
|
|
||||||
inline STDSTRINGCONSTEXPR bool operator==(const std::string& rhs) const noexcept { \
|
|
||||||
return _hash == Hash(rhs.c_str()); \
|
|
||||||
} \
|
|
||||||
inline STDSTRINGCONSTEXPR bool operator!=(const std::string& rhs) const noexcept { \
|
|
||||||
return _hash != Hash(rhs.c_str()); \
|
|
||||||
} \
|
|
||||||
inline constexpr bool operator==(const char* rhs) const noexcept { return _hash == Hash(rhs); } \
|
|
||||||
inline constexpr bool operator!=(const char* rhs) const noexcept { return _hash != Hash(rhs); } \
|
|
||||||
}; \
|
|
||||||
} \
|
|
||||||
\
|
|
||||||
namespace std { \
|
|
||||||
template <> struct hash<ArbUt::name> { \
|
|
||||||
constexpr std::size_t operator()(ArbUt::name const& s) const noexcept { return s.GetHash(); } \
|
|
||||||
}; \
|
|
||||||
template <> struct hash<ArbUt::name##_Literal> { \
|
|
||||||
constexpr std::size_t operator()(ArbUt::name##_Literal const& s) const noexcept { return s.GetHash(); } \
|
|
||||||
}; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -2,29 +2,23 @@
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
#include "../extern/catch.hpp"
|
#include "../extern/catch.hpp"
|
||||||
#include "../src/ConstString.hpp"
|
#include "../src/StringView.hpp"
|
||||||
|
|
||||||
TEST_CASE("Use const string in unordered_map", "[Utilities]") {
|
TEST_CASE("Initialize compile time", "[Utilities]") {
|
||||||
std::unordered_map<ArbUt::ConstString, int32_t> map;
|
static_assert("foo"_cnc.Length() == 3);
|
||||||
map.insert({"foo"_c, 1});
|
static_assert("bar"_cnc.Length() == 3);
|
||||||
map.insert({"bar"_c, 5});
|
|
||||||
|
|
||||||
CHECK(map["bar"_c] == 5);
|
|
||||||
CHECK(map["foo"_c] == 1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Use const string in switch case", "[Utilities]") {
|
TEST_CASE("Compare compile time", "[Utilities]") {
|
||||||
auto val = ArbUt::ConstString("foobar");
|
static_assert("foo"_cnc != "bar"_cnc);
|
||||||
switch (val) {
|
|
||||||
case "foo"_c: FAIL(); break;
|
|
||||||
case "bar"_c: FAIL(); break;
|
|
||||||
case "foobar"_c: SUCCEED(); break;
|
|
||||||
default: FAIL(); break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Compare compile time with CaseInsensitiveConstString", "[Utilities]") {
|
||||||
|
static_assert("foo"_cnc == ArbUt::StringViewLiteral("foo"));
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Use insensitive const string in unordered_map", "[Utilities]") {
|
TEST_CASE("Use insensitive const string in unordered_map", "[Utilities]") {
|
||||||
std::unordered_map<ArbUt::CaseInsensitiveConstString, int32_t> map;
|
std::unordered_map<ArbUt::StringView, int32_t> map;
|
||||||
map.insert({"foO"_cnc, 1});
|
map.insert({"foO"_cnc, 1});
|
||||||
map.insert({"bAR"_cnc, 5});
|
map.insert({"bAR"_cnc, 5});
|
||||||
|
|
||||||
|
@ -33,7 +27,7 @@ TEST_CASE("Use insensitive const string in unordered_map", "[Utilities]") {
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Use case insensitive const string in switch case", "[Utilities]") {
|
TEST_CASE("Use case insensitive const string in switch case", "[Utilities]") {
|
||||||
auto val = ArbUt::CaseInsensitiveConstString("foobar");
|
auto val = ArbUt::StringView("foobar");
|
||||||
switch (val) {
|
switch (val) {
|
||||||
case "foo"_cnc: FAIL(); break;
|
case "foo"_cnc: FAIL(); break;
|
||||||
case "bar"_cnc: FAIL(); break;
|
case "bar"_cnc: FAIL(); break;
|
||||||
|
@ -42,30 +36,8 @@ TEST_CASE("Use case insensitive const string in switch case", "[Utilities]") {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef WINDOWS
|
|
||||||
__attribute__((optnone))
|
|
||||||
#endif
|
|
||||||
static ArbUt::CaseInsensitiveConstString
|
|
||||||
TestCreateConstString() {
|
|
||||||
char originalVal[7];
|
|
||||||
originalVal[0] = 'f';
|
|
||||||
originalVal[1] = 'o';
|
|
||||||
originalVal[2] = 'o';
|
|
||||||
originalVal[3] = 'b';
|
|
||||||
originalVal[4] = 'a';
|
|
||||||
originalVal[5] = 'r';
|
|
||||||
originalVal[6] = '\0';
|
|
||||||
return ArbUt::CaseInsensitiveConstString(originalVal);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_CASE("Out of scope char* doesn't lose reference", "[Utilities]") {
|
|
||||||
ArbUt::CaseInsensitiveConstString val = TestCreateConstString();
|
|
||||||
INFO(val.c_str());
|
|
||||||
REQUIRE(strcmp(val.c_str(), "foobar") == 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_CASE("Literal conststring to non literal, then use", "[Utilities]") {
|
TEST_CASE("Literal conststring to non literal, then use", "[Utilities]") {
|
||||||
ArbUt::CaseInsensitiveConstString val;
|
ArbUt::StringView val;
|
||||||
{ val = "foobar"_cnc; }
|
{ val = "foobar"_cnc; }
|
||||||
INFO(val.c_str());
|
INFO(val.c_str());
|
||||||
REQUIRE(strcmp(val.c_str(), "foobar") == 0);
|
REQUIRE(strcmp(val.c_str(), "foobar") == 0);
|
||||||
|
|
|
@ -1,14 +1,15 @@
|
||||||
#ifdef TESTS_BUILD
|
#ifdef TESTS_BUILD
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
#include "../extern/catch.hpp"
|
#include "../extern/catch.hpp"
|
||||||
|
#include "../src/StringView.hpp"
|
||||||
#include "../src/Enum.hpp"
|
#include "../src/Enum.hpp"
|
||||||
|
|
||||||
ENUM(TestEnum, uint8_t, Val1, Val2, Val3)
|
ENUM(TestEnum, uint8_t, Val1, Val2, Val3)
|
||||||
|
|
||||||
TEST_CASE("Parse Enum case sensitive", "[Utilities]") {
|
TEST_CASE("Parse Enum case sensitive", "[Utilities]") {
|
||||||
CHECK(TestEnumHelper::Parse("Val1") == TestEnum::Val1);
|
STATIC_REQUIRE(TestEnumHelper::Parse("Val1") == TestEnum::Val1);
|
||||||
CHECK(TestEnumHelper::Parse("Val2") == TestEnum::Val2);
|
STATIC_REQUIRE(TestEnumHelper::Parse("Val2") == TestEnum::Val2);
|
||||||
CHECK(TestEnumHelper::Parse("Val3") == TestEnum::Val3);
|
STATIC_REQUIRE(TestEnumHelper::Parse("Val3") == TestEnum::Val3);
|
||||||
CHECK_THROWS(TestEnumHelper::Parse("Val4"));
|
CHECK_THROWS(TestEnumHelper::Parse("Val4"));
|
||||||
CHECK_THROWS(TestEnumHelper::Parse("val1"));
|
CHECK_THROWS(TestEnumHelper::Parse("val1"));
|
||||||
}
|
}
|
||||||
|
@ -53,12 +54,11 @@ TEST_CASE("Try Parse Enum case insensitive", "[Utilities]") {
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Enum To String", "[Utilities]") {
|
TEST_CASE("Enum To String", "[Utilities]") {
|
||||||
CHECK(strcmp(TestEnumHelper::ToString(TestEnum::Val1), "Val1") == 0);
|
STATIC_REQUIRE(TestEnumHelper::ToString(TestEnum::Val1) == "Val1");
|
||||||
CHECK(strcmp(TestEnumHelper::ToString(TestEnum::Val2), "Val2") == 0);
|
STATIC_REQUIRE(TestEnumHelper::ToString(TestEnum::Val2) == "Val2");
|
||||||
CHECK(strcmp(TestEnumHelper::ToString(TestEnum::Val3), "Val3") == 0);
|
STATIC_REQUIRE(TestEnumHelper::ToString(TestEnum::Val3) == "Val3");
|
||||||
auto s = TestEnumHelper::ToString((TestEnum)100);
|
|
||||||
CHECK(strcmp(s, "100") == 0);
|
CHECK(TestEnumHelper::ToString((TestEnum)100) == "out of bounds");
|
||||||
delete[] s;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Enum Get Values", "[Utilities]") {
|
TEST_CASE("Enum Get Values", "[Utilities]") {
|
||||||
|
@ -69,4 +69,12 @@ TEST_CASE("Enum Get Values", "[Utilities]") {
|
||||||
CHECK(vec[2] == TestEnum::Val3);
|
CHECK(vec[2] == TestEnum::Val3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Enum Get Highest", "[Utilities]") { STATIC_REQUIRE(TestEnumHelper::Highest() == TestEnum::Val3); }
|
||||||
|
|
||||||
|
TEST_CASE("Enum Get Lowest", "[Utilities]") { STATIC_REQUIRE(TestEnumHelper::Lowest() == TestEnum::Val1); }
|
||||||
|
|
||||||
|
TEST_CASE("Enum Get First", "[Utilities]") { STATIC_REQUIRE(TestEnumHelper::First() == TestEnum::Val1); }
|
||||||
|
|
||||||
|
TEST_CASE("Enum Get Last", "[Utilities]") { STATIC_REQUIRE(TestEnumHelper::Last() == TestEnum::Val3); }
|
||||||
|
|
||||||
#endif
|
#endif
|
|
@ -67,4 +67,11 @@ TEST_CASE("Test IndexOf", "[Utilities]") {
|
||||||
CHECK(ls.IndexOf(684) == -1);
|
CHECK(ls.IndexOf(684) == -1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Test list out of bounds", "[Utilities]") {
|
||||||
|
auto ls = List<int>({5, 200, 1500, -500, 5, 300, -500});
|
||||||
|
REQUIRE_THROWS(ls.At(-1));
|
||||||
|
REQUIRE_THROWS(ls.At(7));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
#endif
|
#endif
|
|
@ -41,4 +41,15 @@ TEST_CASE("Create Unique Ptr list, append, iterate") {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Test unique ptr list out of bounds", "[Utilities]") {
|
||||||
|
auto ls = UniquePtrList<uint32_t>();
|
||||||
|
auto v1 = new uint32_t(100);
|
||||||
|
auto v2 = new uint32_t(5000);
|
||||||
|
ls.Append(v1);
|
||||||
|
ls.Append(v2);
|
||||||
|
REQUIRE_THROWS(ls.At(-1));
|
||||||
|
REQUIRE_THROWS(ls.At(2));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
#endif
|
#endif
|
Loading…
Reference in New Issue