#include "ExportCore.hpp" #include "Database.hpp" #include "DataTypes.hpp" #include "Utilities.hpp" #include namespace VSW::Materializer::ExportEnvironment { struct ExportContext { ExportContext(CKContext* ctx, const YYCC::yycc_u8string_view& db_path) : db(db_path), cache(), reporter(ctx) {} Database::EnvironmentDatabase db; DataTypes::Environment::DataCache cache; Utilities::EnhancedReporter reporter; }; static void IterateParameterOperation(ExportContext& expctx, CKParameterManager* param_mgr) { // prepare variables std::vector op_list; CKGUID guid; // get the count of all supported paramter operations and iterate it. int count = param_mgr->GetParameterOperationCount(); int listCount = 0, cacheListCount = 0; for (int i = 0; i < count; i++) { // fill the shared data part. expctx.cache.op.op_code = i; guid = param_mgr->OperationCodeToGuid(i); Utilities::CopyGuid(expctx.cache.op.op_guid, guid); expctx.cache.op.op_name = param_mgr->OperationCodeToName(i); // get all sub-operation of this parameter operation. // each sub-operation can have different in out parameter type // and they are just grouped into the same parameter operation name. int op_list_count = param_mgr->GetAvailableOperationsDesc(guid, nullptr, nullptr, nullptr, nullptr); op_list.resize(static_cast(op_list_count)); param_mgr->GetAvailableOperationsDesc(guid, nullptr, nullptr, nullptr, op_list.data()); for (const auto& op : op_list) { Utilities::CopyGuid(expctx.cache.op.in1_guid, op.P1Guid); Utilities::CopyGuid(expctx.cache.op.in2_guid, op.P2Guid); Utilities::CopyGuid(expctx.cache.op.out_guid, op.ResGuid); expctx.cache.op.func_ptr = Utilities::RelativeAddress(op.Fct); expctx.db.Write(expctx.cache.op); } } } static void IterateParameter(ExportContext& expctx, CKParameterManager* param_mgr) { // prepare variables CKParameterTypeDesc* desc = nullptr; // get the count of all paramter types and iterate it. int count = param_mgr->GetParameterTypesCount(); for (int i = 0; i < count; i++) { desc = param_mgr->GetParameterTypeDescription(i); expctx.cache.param.index = desc->Index; Utilities::CopyGuid(expctx.cache.param.guid, desc->Guid); Utilities::CopyGuid(expctx.cache.param.derived_from, desc->DerivedFrom); expctx.cache.param.type_name = desc->TypeName.CStr(); expctx.cache.param.default_size = desc->DefaultSize; expctx.cache.param.func_CreateDefault = Utilities::RelativeAddress(desc->CreateDefaultFunction); expctx.cache.param.func_Delete = Utilities::RelativeAddress(desc->DeleteFunction); expctx.cache.param.func_SaveLoad = Utilities::RelativeAddress(desc->SaveLoadFunction); expctx.cache.param.func_Check = Utilities::RelativeAddress(desc->CheckFunction); expctx.cache.param.func_Copy = Utilities::RelativeAddress(desc->CopyFunction); expctx.cache.param.func_String = Utilities::RelativeAddress(desc->StringFunction); expctx.cache.param.func_UICreator = Utilities::RelativeAddress(desc->UICreatorFunction); CKPluginEntry* plugin_entry = desc->CreatorDll; if (plugin_entry != nullptr) { expctx.cache.param.creator_dll_index = plugin_entry->m_PluginDllIndex; expctx.cache.param.creator_plugin_index = plugin_entry->m_PositionInDll; } else { expctx.cache.param.creator_dll_index = -1; expctx.cache.param.creator_plugin_index = -1; } expctx.cache.param.dw_param = desc->dwParam; expctx.cache.param.dw_flags = desc->dwFlags; expctx.cache.param.cid = desc->Cid; Utilities::CopyGuid(expctx.cache.param.saver_manager, desc->Saver_Manager); expctx.db.Write(expctx.cache.param); } } static void IterateAttribute(ExportContext& expctx, CKAttributeManager* attr_mgr) { // get the count of all attributes and iterate it. int count = attr_mgr->GetAttributeCount(); for (int i = 0; i < count; i++) { expctx.cache.attr.index = i; expctx.cache.attr.name = attr_mgr->GetAttributeNameByType(i); expctx.cache.attr.category_index = attr_mgr->GetAttributeCategoryIndex(i); Utilities::CopyCKString(expctx.cache.attr.category_name, attr_mgr->GetAttributeCategory(i)); expctx.cache.attr.flags = attr_mgr->GetAttributeFlags(i); expctx.cache.attr.param_index = attr_mgr->GetAttributeParameterType(i); expctx.cache.attr.compatible_classid = attr_mgr->GetAttributeCompatibleClassId(i); Utilities::CopyCKString(expctx.cache.attr.default_value, attr_mgr->GetAttributeDefaultValue(i)); expctx.db.Write(expctx.cache.attr); } } static void IteratePlugin(ExportContext& expctx, CKPluginManager* plugin_mgr) { // prepare variables CK_PLUGIN_TYPE plugin_type; // get category count and iterate them int category_count = plugin_mgr->GetCategoryCount(); for (int i = 0; i < category_count; ++i) { // get category name and set it expctx.cache.plugin.category = plugin_mgr->GetCategoryName(i); // iterate plugin within this category int plugin_count = plugin_mgr->GetPluginCount(i); for (int j = 0; j < plugin_count; j++) { CKPluginEntry* plugin_entry = plugin_mgr->GetPluginInfo(i, j); CKPluginInfo* plugin_info = &(plugin_entry->m_PluginInfo); CKPluginDll* plugin_dll = plugin_mgr->GetPluginDllInfo(plugin_entry->m_PluginDllIndex); // dll infomation (name + position in dll) expctx.cache.plugin.dll_name = plugin_dll->m_DllFileName.CStr(); expctx.cache.plugin.position_in_dll = plugin_entry->m_PositionInDll; // plugin info Utilities::CopyGuid(expctx.cache.plugin.guid, plugin_info->m_GUID); expctx.cache.plugin.desc = plugin_info->m_Description.CStr(); expctx.cache.plugin.author = plugin_info->m_Author.CStr(); expctx.cache.plugin.summary = plugin_info->m_Summary.CStr(); expctx.cache.plugin.version = plugin_info->m_Version; plugin_type = plugin_info->m_Type; expctx.cache.plugin.type = plugin_type; expctx.cache.plugin.func_init = Utilities::RelativeAddress(plugin_info->m_InitInstanceFct); expctx.cache.plugin.func_exit = Utilities::RelativeAddress(plugin_info->m_ExitInstanceFct); // extra fields according to plugin type // first reset these specific fields expctx.cache.plugin.reader_fct.clear(); // then try to fetch these specific fields switch (plugin_type) { case CKPLUGIN_BITMAP_READER: case CKPLUGIN_SOUND_READER: case CKPLUGIN_MODEL_READER: case CKPLUGIN_MOVIE_READER: { // Reader specific expctx.cache.plugin.reader_fct = Utilities::RelativeAddress(plugin_entry->m_ReadersInfo->m_GetReaderFct); expctx.cache.plugin.reader_opt_count = plugin_entry->m_ReadersInfo->m_OptionCount; expctx.cache.plugin.reader_flags = plugin_entry->m_ReadersInfo->m_ReaderFlags; Utilities::CopyGuid(expctx.cache.plugin.reader_setting_param_guid, plugin_entry->m_ReadersInfo->m_SettingsParameterGuid); expctx.cache.plugin.reader_file_ext = static_cast(plugin_info->m_Extension); break; } case CKPLUGIN_BEHAVIOR_DLL: { // Behavior specific std::vector guids; for (int i = 0; i < plugin_entry->m_BehaviorsInfo->m_BehaviorsGUID.Size(); ++i) { int64_t guid_cache; Utilities::CopyGuid(guid_cache, plugin_entry->m_BehaviorsInfo->m_BehaviorsGUID[i]); guids.emplace_back(YYCC::ParserHelper::ToString(guid_cache)); } expctx.cache.plugin.behavior_guids = YYCC::EncodingHelper::ToOrdinaryView( YYCC::StringHelper::Join(guids, YYCC_U8(", ")) ); break; } case CKPLUGIN_MANAGER_DLL: case CKPLUGIN_RENDERENGINE_DLL: { // Render engine and manager specific // active info expctx.cache.plugin.manager_active = plugin_entry->m_Active; break; } } expctx.db.Write(expctx.cache.plugin); } } } #if !defined(VIRTOOLS_21) static void IterateVariable(ExportContext& expctx, CKVariableManager* var_mgr) { // prepare variables CKVariableManager::Variable* varobj = nullptr; const char* var_name; CKVariableManager::Variable::Type var_type; const char* var_representation; int int_cache; float float_cache; XString xstring_cache; for (CKVariableManager::Iterator it = var_mgr->GetVariableIterator(); !it.End(); it++) { // get variable varobj = it.GetVariable(); // variable name var_name = it.GetName(); expctx.cache.variable.name = var_name; // variable description Utilities::CopyCKString(expctx.cache.variable.desciption, varobj->GetDescription()); // variable flags expctx.cache.variable.flags = varobj->GetFlags(); // variable type var_type = varobj->GetType(); expctx.cache.variable.type = var_type; // variable representation // Representation need a special treatment. // Because it is not a name. // So we should record it as empty string if it is nullptr, instead of default var_representation = varobj->GetRepresentation(); Utilities::CopyCKString(expctx.cache.variable.representation, var_representation, ""); // We output variable stored value in different way // according to its type. switch (var_type) { case CKVariableManager::Variable::Type::INT: var_mgr->GetValue(var_name, &int_cache); expctx.cache.variable.data = YYCC::EncodingHelper::ToOrdinaryView(YYCC::ParserHelper::ToString(int_cache)); break; case CKVariableManager::Variable::Type::FLOAT: var_mgr->GetValue(var_name, &float_cache); expctx.cache.variable.data = YYCC::EncodingHelper::ToOrdinaryView(YYCC::ParserHelper::ToString(float_cache)); break; case CKVariableManager::Variable::Type::STRING: var_mgr->GetValue(var_name, xstring_cache); expctx.cache.variable.data = xstring_cache.CStr(); break; default: throw std::runtime_error("invalid variable type!"); } expctx.db.Write(expctx.cache.variable); } } #endif void Export(CKContext* ctx, const YYCC::yycc_u8string_view& db_path, UINT code_page) { // create database and data cache in context ExportContext expctx(ctx, db_path); // export environment one by one IterateParameterOperation(expctx, ctx->GetParameterManager()); IterateParameter(expctx, ctx->GetParameterManager()); IterateAttribute(expctx, ctx->GetAttributeManager()); IteratePlugin(expctx, CKGetPluginManager()); #if !defined(VIRTOOLS_21) IterateVariable(expctx, ctx->GetVariableManager()); #endif } }