#pragma once /* // https://stackoverflow.com/questions/2164827/explicitly-exporting-shared-library-functions-in-linux // generate import export macro #if defined(_MSC_VER) // Microsoft #define LIBCMO_RAW_EXPORT __declspec(dllexport) #define LIBCMO_RAW_IMPORT __declspec(dllimport) #elif defined(__GNUC__) // GCC #define LIBCMO_RAW_EXPORT __attribute__((visibility("default"))) #define LIBCMO_RAW_IMPORT #elif defined(__clang__) // GCC #define LIBCMO_RAW_EXPORT __attribute__((visibility("default"))) #define LIBCMO_RAW_IMPORT #else // do nothing and hope for the best? #define LIBCMO_RAW_EXPORT #define LIBCMO_RAW_IMPORT #pragma warning Unknown dynamic link import/export semantics. #endif // choosee proper style #if defined(LIBCMO_EXPORTING) #define LIBCMO_NAKED_EXPORT LIBCMO_RAW_EXPORT #else #define LIBCMO_NAKED_EXPORT LIBCMO_RAW_IMPORT #endif // some work for cpp #if defined(__cplusplus) #define LIBCMO_EXPORT extern "C" LIBCMO_NAKED_EXPORT #else #define LIBCMO_EXPORT LIBCMO_NAKED_EXPORT #endif */ #if defined(_WIN32) #define LIBCMO_OS_WIN32 // disable annoy win32 macro #define WIN32_LEAN_AND_MEAN #define NOMINMAX #endif #include #include #include #include #include #include #pragma region Batch Ctor operator= Operations #define LIBCMO_DISABLE_COPY_MOVE(CLSNAME) \ CLSNAME(const CLSNAME&) = delete; \ CLSNAME(CLSNAME&&) = delete; \ CLSNAME& operator=(const CLSNAME&) = delete; \ CLSNAME& operator=(CLSNAME&&) = delete; #define LIBCMO_DEFAULT_COPY_MOVE(CLSNAME) \ CLSNAME(const CLSNAME&) = default; \ CLSNAME(CLSNAME&&) = default; \ CLSNAME& operator=(const CLSNAME&) = default; \ CLSNAME& operator=(CLSNAME&&) = default; #pragma endregion namespace LibCmo { [[noreturn]] void LibPanic(int line, const char* file, const char* errmsg); #define LIBPANIC(msg) LibCmo::LibPanic(__LINE__, __FILE__, msg); namespace EnumsHelper { template, int> = 0> inline TEnum Merge(std::initializer_list il) { std::underlying_type_t result = 0; for (auto it = il.begin(); it != il.end(); ++it) { result |= static_cast>(*it); } return static_cast(result); } template, int> = 0> inline TEnum Inv(TEnum e) { return static_cast(~(static_cast>(e))); } template, int> = 0> inline void Rm(TEnum& e1, TEnum e2) { e1 = static_cast(static_cast>(e1) & static_cast>(Inv(e2))); } template, int> = 0> inline void Mask(TEnum& e1, TEnum e2) { e1 = static_cast(static_cast>(e1) & static_cast>(e2)); } template, int> = 0> inline void Add(TEnum& e1, TEnum e2) { e1 = static_cast(static_cast>(e1) | static_cast>(e2)); } template, int> = 0> inline bool Has(TEnum e, TEnum probe) { return static_cast(static_cast>(e) & static_cast>(probe)); } } }