#pragma once #include "json.hpp" #include #include #include #include #include #include #include #include #ifdef __cplusplus extern "C" { #endif #define LIMIT(x, min, max) \ ({ \ typeof(x) _x = x; \ typeof(min) _min = min; \ typeof(max) _max = max; \ (((_x) < (_min)) ? (_min) : ((_x) > (_max)) ? (_max) \ : (_x)); \ }) #define DegToRad(a) ((a) *0.01745329252f) #define RadToDeg(a) ((a) *57.29577951f) #define ALIGN4(x) (((x) + 3) & ~3) #define ALIGN32(x) (((x) + 31) & ~31) #define ALIGN_DATA(align) __attribute__((aligned(align))) #define ALIGN_DATA_0x40 ALIGN_DATA(0x40) // those work only in powers of 2 #define ROUNDDOWN(val, align) ((val) & ~(align - 1)) #define ROUNDUP(val, align) ROUNDDOWN(((val) + (align - 1)), align) #define le16(i) ((((uint16_t) ((i) &0xFF)) << 8) | ((uint16_t) (((i) &0xFF00) >> 8))) #define le32(i) ((((uint32_t) le16((i) &0xFFFF)) << 16) | ((uint32_t) le16(((i) &0xFFFF0000) >> 16))) #define le64(i) ((((uint64_t) le32((i) &0xFFFFFFFFLL)) << 32) | ((uint64_t) le32(((i) &0xFFFFFFFF00000000LL) >> 32))) //Needs to have log_init() called beforehand. void dumpHex(const void *data, size_t size); #ifdef __cplusplus } #endif template std::unique_ptr make_unique_nothrow(Args &&...args) noexcept(noexcept(T(std::forward(args)...))) { return std::unique_ptr(new (std::nothrow) T(std::forward(args)...)); } template inline typename std::unique_ptr make_unique_nothrow(size_t num) noexcept { return std::unique_ptr(new (std::nothrow) std::remove_extent_t[num]()); } template std::shared_ptr make_shared_nothrow(Args &&...args) noexcept(noexcept(T(std::forward(args)...))) { return std::shared_ptr(new (std::nothrow) T(std::forward(args)...)); } template std::enable_if_t>, bool> remove_first_if(Container &container, Predicate pred) { auto it = container.before_begin(); for (auto prev = it, current = ++it; current != container.end(); ++prev, ++current) { if (pred(*current)) { container.erase_after(prev); return true; } } return false; } template std::enable_if_t>, bool> remove_first_if(Container &container, Predicate pred) { auto it = container.begin(); while (it != container.end()) { if (pred(*it)) { container.erase(it); return true; } ++it; } return false; } template std::enable_if_t>, bool> remove_first_if(Container &container, Predicate pred) { auto it = container.begin(); while (it != container.end()) { if (pred(*it)) { container.erase(it); return true; } ++it; } return false; } template bool remove_locked_first_if(std::mutex &mutex, Container &container, Predicate pred) { std::lock_guard lock(mutex); return remove_first_if(container, pred); } template T pop_locked_first_if(std::mutex &mutex, std::vector &container, Predicate pred) { std::lock_guard lock(mutex); T result; auto it = container.begin(); while (it != container.end()) { if (pred(*it)) { result = std::move(*it); container.erase(it); return result; } ++it; } return result; } template void append_move_all_values(Container &dest, Container &src) { dest.insert(dest.end(), std::make_move_iterator(src.begin()), std::make_move_iterator(src.end())); src.clear(); } typedef enum { UTILS_IO_ERROR_SUCCESS = 0, /**< Success. */ UTILS_IO_ERROR_INVALID_ARGS = -0x01, /**< Invalid arguments passed to the function. */ UTILS_IO_ERROR_MALLOC_FAILED = -0x02, /**< Memory allocation failed. */ UTILS_IO_ERROR_GENERIC = -0x03, /**< Generic IO error during saving or loading. */ UTILS_IO_ERROR_NOT_FOUND = -0x4, /**< Item not found. */ } UtilsIOError; std::string getPluginPath(); std::string getModulePath(); OSDynLoad_Error CustomDynLoadAlloc(int32_t size, int32_t align, void **outAddr); void CustomDynLoadFree(void *addr); UtilsIOError ParseJsonFromFile(const std::string &filePath, nlohmann::json &outJson); std::vector getPluginFilePaths(std::string_view basePath); std::vector getNonBaseAromaPluginFilenames(std::string_view basePath);