/** * Copyright (C) 2021-2024 Saturneric * * This file is part of GpgFrontend. * * GpgFrontend is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * GpgFrontend is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with GpgFrontend. If not, see . * * The initial version of the source code is inherited from * the gpg4usb project, which is under GPL-3.0-or-later. * * All the source code of GpgFrontend was modified and released by * Saturneric starting on May 12, 2021. * * SPDX-License-Identifier: GPL-3.0-or-later * */ #pragma once #include #include #include #include #include #include #include #include #include #define DUP(v) GFModuleStrDup(v) #define UDUP(v) UnStrDup(v) #define QDUP(v) QStrDup(v) #define LISTEN(event) GFModuleListenEvent(GFGetModuleID(), DUP(event)) #define DEFINE_TRANSLATIONS_STRUCTURE(name) \ class GTrC { \ Q_DECLARE_TR_FUNCTIONS(GTrC) \ }; \ auto TranslatorDataReader(const char* p_l, char** p_d) -> int { \ auto locale = UDUP(p_l); \ QFile f(QString(":/i18n/%2.%1.qm").arg(locale).arg(#name)); \ if (f.exists() && f.open(QIODevice::ReadOnly)) { \ auto b = f.readAll(); \ *p_d = AllocBufferAndCopy(b); \ return b.size(); \ } \ FLOG_WARN("%3 loading, locale: %1, not found", locale, f.fileName(), \ UDUP(GFGetModuleID())); \ *p_d = nullptr; \ return 0; \ } #define REGISTER_TRANS_READER() \ GFAppRegisterTranslatorReader(GFGetModuleID(), TranslatorDataReader) #define GC_TR(text) QT_TRANSLATE_NOOP("GTrC", text) #define CONCATENATE_DETAIL(x, y) x##y #define CONCATENATE(x, y) CONCATENATE_DETAIL(x, y) #define GTRC_TR(name, src) CONCATENATE(GTrC_, name)::tr(src) #define STRINGIFY(x) #x #define TOSTRING(x) STRINGIFY(x) #define GTRC_AS_STRING(name) TOSTRING(GTrC_##name) #define EXECUTE_MODULE() \ auto GFExecuteModule(GFModuleEvent* p_event) -> int { \ auto event = ConvertEventToMap(p_event); #define END_EXECUTE_MODULE() } #define CB_SUCC(event) \ { \ CB(event, GFGetModuleID(), {{"ret", "0"}}); \ return 0; \ } #define CB_ERR(event, ret, err) \ { \ CB(event, GFGetModuleID(), \ {{"ret", QString::number(ret)}, {"err", QString(err)}}); \ return ret; \ } #define DEFINE_EXECUTE_API_USING_STANDARD_EVEN_HANDLE_MODEL \ EXECUTE_MODULE() { \ auto event_id = event["event_id"]; \ auto it = _gr_module_event_handlers.find(event_id); \ if (it != _gr_module_event_handlers.end()) { \ return it.value()(event); \ } \ CB_ERR(event, -1, QString("unsupported event id: %1").arg(event_id)); \ } \ END_EXECUTE_MODULE() #define REGISTER_EVENT_HANDLER(event_id, handler) \ static const bool _gv_register_event_handler_by_id_##event_id = [] { \ _gr_module_event_handlers[#event_id] = handler; \ return true; \ }(); inline void MLogDebug(const QString& s) { GFModuleLogDebug(s.toUtf8()); } inline void MLogInfo(const QString& s) { GFModuleLogInfo(s.toUtf8()); } inline void MLogWarn(const QString& s) { GFModuleLogWarn(s.toUtf8()); } inline void MLogError(const QString& s) { GFModuleLogError(s.toUtf8()); } #define LOG_DEBUG(format) MLogDebug(FormatString(QString(format))) #define LOG_INFO(format) MLogDebug(FormatString(QString(format))) #define LOG_WARN(format) MLogDebug(FormatString(QString(format))) #define LOG_ERROR(format) MLogDebug(FormatString(QString(format))) #define FLOG_DEBUG(format, ...) \ MLogDebug(FormatString(QString(format), __VA_ARGS__)) #define FLOG_INFO(format, ...) \ MLogInfo(FormatString(QString(format), __VA_ARGS__)) #define FLOG_WARN(format, ...) \ MLogWarn(FormatString(QString(format), __VA_ARGS__)) #define FLOG_ERROR(format, ...) \ MLogError(FormatString(QString(format), __VA_ARGS__)) inline auto QStrDup(QString str) -> char* { return DUP(str.toUtf8()); } inline auto UnStrDup(const char* s) -> QString { auto q_s = QString::fromUtf8(s == nullptr ? "" : s); if (s != nullptr) GFFreeMemory(static_cast(const_cast(s))); return q_s; } inline auto FormatStringHelper(const QString& format, const std::string& arg) -> QString { return format.arg(QString::fromStdString(arg)); } template auto FormatStringHelper(const QString& format, T arg) -> QString { return format.arg(arg); } template auto FormatStringHelper(const QString& format, const std::string& arg, Args... args) -> QString { return FormatStringHelper(format.arg(QString::fromStdString(arg)), args...); } template auto FormatStringHelper(const QString& format, T arg, Args... args) -> QString { return FormatStringHelper(format.arg(arg), args...); } inline auto FormatStringHelper(const QString& format) -> QString { return format; } template inline auto FormatStringHelper(const T& format) -> typename std::enable_if::value, QString>::type { return FormatStringHelper(QString::fromStdString(format)); } template auto FormatString(const QString& format, Args... args) -> QString { return FormatStringHelper(format, args...); } inline auto QMapToMetaDataArray(const QMap& map) -> MetaData** { auto** meta_data_array = static_cast(GFAllocateMemory(sizeof(MetaData*) * map.size())); int index = 0; for (auto it = map.begin(); it != map.end(); ++it) { meta_data_array[index] = static_cast(GFAllocateMemory(sizeof(MetaData))); QByteArray const key = it.key().toUtf8(); QByteArray const value = it.value().toUtf8(); meta_data_array[index]->key = GFModuleStrDup(key); meta_data_array[index]->value = GFModuleStrDup(value); ++index; } return meta_data_array; } inline auto QMapToGFModuleMetaDataList(const QMap& map) -> GFModuleMetaData* { GFModuleMetaData* head = nullptr; GFModuleMetaData* tail = nullptr; for (auto it = map.begin(); it != map.end(); ++it) { auto* new_node = static_cast( GFAllocateMemory(sizeof(GFModuleMetaData))); QByteArray const key = it.key().toUtf8(); QByteArray const value = it.value().toUtf8(); new_node->key = DUP(key); new_node->value = DUP(value); new_node->next = nullptr; if (tail != nullptr) { tail->next = new_node; } else { head = new_node; } tail = new_node; } return head; } inline auto ConvertEventParamsToMap(GFModuleEventParam* params) -> QMap { QMap param_map; GFModuleEventParam* current = params; GFModuleEventParam* last; while (current != nullptr) { const auto name = UDUP(current->name); const auto value = UDUP(current->value); if (!name.isEmpty()) param_map[name] = value; last = current; current = current->next; GFFreeMemory(last); } return param_map; } inline auto ConvertMapToParams(const QMap& param_map) -> GFModuleEventParam* { GFModuleEventParam* head = nullptr; GFModuleEventParam* prev = nullptr; #if QT_VERSION >= QT_VERSION_CHECK(6, 4, 0) for (const auto& [key, value] : param_map.asKeyValueRange()) { auto* param = static_cast( GFAllocateMemory(sizeof(GFModuleEventParam))); param->name = DUP(key.toUtf8()); param->value = DUP(value.toUtf8()); param->next = nullptr; if (prev == nullptr) { head = param; } else { prev->next = param; } prev = param; } #else for (auto it = param_map.keyValueBegin(); it != param_map.keyValueEnd(); ++it) { auto* param = static_cast( GFAllocateMemory(sizeof(GFModuleEventParam))); param->name = DUP(it->first.toUtf8()); param->value = DUP(it->second.toUtf8()); param->next = nullptr; if (prev == nullptr) { head = param; } else { prev->next = param; } prev = param; } #endif return head; } inline auto ConvertEventToMap(GFModuleEvent* event) -> QMap { QMap event_map; event_map["event_id"] = UDUP(event->id); event_map["trigger_id"] = UDUP(event->trigger_id); event_map.insert(ConvertEventParamsToMap(event->params)); GFFreeMemory(event); return event_map; } inline auto ConvertMapToEvent(QMap event_map) -> GFModuleEvent* { auto* event = static_cast(GFAllocateMemory(sizeof(GFModuleEvent))); event->id = DUP(event_map["event_id"].toUtf8()); event->trigger_id = DUP(event_map["trigger_id"].toUtf8()); event->params = nullptr; return event; } inline void CB(const QMap& event, const char* module, const QMap& params) { GFModuleTriggerModuleEventCallback(ConvertMapToEvent(event), module, ConvertMapToParams(params)); } inline auto AllocBufferAndCopy(const QByteArray& b) -> char* { auto* p = static_cast(GFAllocateMemory(sizeof(char) * b.size())); memcpy(p, b.constData(), b.size()); return p; } template auto SecureCreateSharedObject(Args&&... args) -> std::shared_ptr { void* mem = GFAllocateMemory(sizeof(T)); if (!mem) throw std::bad_alloc(); try { T* obj = new (mem) T(std::forward(args)...); return std::shared_ptr(obj, [](T* ptr) { ptr->~T(); GFFreeMemory(ptr); }); } catch (...) { GFFreeMemory(mem); throw; } } template class PointerConverter { public: explicit PointerConverter(void* ptr) : ptr_(ptr) {} auto AsType() const -> T* { return static_cast(ptr_); } private: void* ptr_; }; /** * @brief * * @tparam T * @return T* */ template auto SecureMallocAsType(std::size_t size) -> T* { return PointerConverter(GFAllocateMemory(size)).AsType(); } /** * @brief * * @return void* */ template auto SecureReallocAsType(T* ptr, std::size_t size) -> T* { return PointerConverter(GFReallocateMemory(ptr, size)).AsType(); } template auto SecureCreateQSharedObject(Args&&... args) -> QSharedPointer { void* mem = GFAllocateMemory(sizeof(T)); if (!mem) throw std::bad_alloc(); try { T* obj = new (mem) T(std::forward(args)...); return QSharedPointer(obj, [](T* ptr) { ptr->~T(); GFFreeMemory(ptr); }); } catch (...) { GFFreeMemory(mem); throw; } } inline auto CharArrayToQStringList(char** pl_components, int size) -> QStringList { QStringList list; for (int i = 0; i < size; ++i) { list.append(UDUP(pl_components[i])); } GFFreeMemory(pl_components); return list; } inline auto QStringListToCharArray(const QStringList& list) -> char** { char** char_array = static_cast(GFAllocateMemory(list.size() * sizeof(char*))); int index = 0; for (const QString& item : list) { QByteArray value = item.toUtf8(); char_array[index] = static_cast(GFAllocateMemory(value.size() + 1)); std::strcpy(char_array[index], value.constData()); index++; } return char_array; } template inline auto ArrayToQList(T** pl_components, int size) -> QList { if (pl_components == nullptr || size <= 0) { return QList(); } QList list; for (int i = 0; i < size; ++i) { list.append(*pl_components[i]); GFFreeMemory(pl_components[i]); } GFFreeMemory(pl_components); return list; } template inline auto QListToArray(const QList& list) -> T** { T** array = static_cast(GFAllocateMemory(list.size() * sizeof(T*))); int index = 0; for (const T& item : list) { auto mem = static_cast(GFAllocateMemory(sizeof(T))); array[index] = new (mem) T(item); index++; } return array; } inline auto ConvertQVariantToVoidPtr(const QVariant& variant) -> void* { void* mem = GFAllocateMemory(sizeof(QVariant)); auto* variant_ptr = new (mem) QVariant(variant); return static_cast(variant_ptr); } inline auto ConvertVoidPtrToQVariant(void* ptr) -> QVariant { if (ptr == nullptr) return {}; auto* variant_ptr = static_cast(ptr); QVariant variant = *variant_ptr; GFFreeMemory(variant_ptr); return variant; } #define Q_VARIANT_Q_OBJECT_FACTORY_DECLARE(name) \ auto name(void* data_raw_ptr) -> void*; #define Q_VARIANT_Q_OBJECT_FACTORY_DEFINE(name, func) \ auto name(void* data_raw_ptr) -> void* { \ auto data = ConvertVoidPtrToQVariant(data_raw_ptr); \ return func(data); \ } #define GUI_OBJECT(factory, data) \ GFUICreateGUIObject(factory, ConvertQVariantToVoidPtr(data))