2023-02-06 15:58:16 +08:00
|
|
|
#pragma once
|
|
|
|
|
2023-02-11 15:29:51 +08:00
|
|
|
/*
|
2023-02-06 15:58:16 +08:00
|
|
|
// https://stackoverflow.com/questions/2164827/explicitly-exporting-shared-library-functions-in-linux
|
|
|
|
// generate import export macro
|
|
|
|
#if defined(_MSC_VER)
|
2023-02-11 15:29:51 +08:00
|
|
|
// Microsoft
|
2023-02-06 15:58:16 +08:00
|
|
|
#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
|
2023-02-11 15:29:51 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#if defined(_WIN32)
|
|
|
|
#define LIBCMO_OS_WIN32
|
|
|
|
#endif
|
|
|
|
|
2023-02-25 22:58:28 +08:00
|
|
|
#include <cstdio>
|
2023-02-26 21:48:03 +08:00
|
|
|
#include <type_traits>
|
|
|
|
#include <cinttypes>
|
|
|
|
#include <cstdint>
|
2023-03-01 10:48:50 +08:00
|
|
|
#include <initializer_list>
|
2023-02-25 22:58:28 +08:00
|
|
|
|
2023-02-11 15:29:51 +08:00
|
|
|
namespace LibCmo {
|
2023-02-26 21:48:03 +08:00
|
|
|
|
|
|
|
namespace EnumsHelper {
|
|
|
|
template<typename TEnum>
|
2023-02-28 14:04:38 +08:00
|
|
|
inline TEnum FlagEnumMerge(std::initializer_list<TEnum> il) {
|
|
|
|
std::underlying_type_t<TEnum> result = 0u;
|
|
|
|
for (auto it = il.begin(); it != il.end(); ++it) {
|
|
|
|
result |= static_cast<std::underlying_type_t<TEnum>>(*it);
|
|
|
|
}
|
|
|
|
return static_cast<TEnum>(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename TEnum>
|
|
|
|
inline TEnum FlagEnumInv(TEnum e) {
|
|
|
|
return static_cast<TEnum>(~(static_cast<std::underlying_type_t<TEnum>>(e)));
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename TEnum>
|
|
|
|
inline void FlagEnumRm(TEnum& e, std::initializer_list<TEnum> il) {
|
|
|
|
auto mask = FlagEnumInv(FlagEnumMerge(il));
|
|
|
|
e = static_cast<TEnum>(static_cast<std::underlying_type_t<TEnum>>(e) & static_cast<std::underlying_type_t<TEnum>>(mask));
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename TEnum>
|
|
|
|
inline void FlagEnumAdd(TEnum& e, std::initializer_list<TEnum> il) {
|
|
|
|
auto mask = FlagEnumMerge(il);
|
|
|
|
e = static_cast<TEnum>(static_cast<std::underlying_type_t<TEnum>>(e) | static_cast<std::underlying_type_t<TEnum>>(mask));
|
2023-02-26 21:48:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename TEnum>
|
|
|
|
inline bool FlagEnumHas(TEnum e, TEnum probe) {
|
|
|
|
return static_cast<bool>(static_cast<std::underlying_type_t<TEnum>>(e) & static_cast<std::underlying_type_t<TEnum>>(probe));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace StreamHelper {
|
2023-02-11 15:29:51 +08:00
|
|
|
|
2023-02-25 22:58:28 +08:00
|
|
|
void CopyStream(const void* src, FILE* dest, size_t len);
|
|
|
|
void CopyStream(FILE* src, void* dest, size_t len);
|
2023-02-11 15:29:51 +08:00
|
|
|
|
2023-02-25 17:39:39 +08:00
|
|
|
}
|
2023-02-11 15:29:51 +08:00
|
|
|
}
|