WiiUPluginLoaderBackend/source/plugin/PluginMetaInformationFactory.cpp

145 lines
5.8 KiB
C++
Raw Normal View History

/****************************************************************************
* Copyright (C) 2018-2020 Maschell
*
* This program 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.
*
* This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
****************************************************************************/
2022-02-04 16:25:44 +01:00
#include "PluginMetaInformationFactory.h"
#include "elfio/elfio.hpp"
#include "fs/FSUtils.h"
#include "utils/logger.h"
#include "utils/wiiu_zlib.hpp"
2021-12-15 17:09:30 +01:00
#include <memory>
std::optional<std::unique_ptr<PluginMetaInformation>> PluginMetaInformationFactory::loadPlugin(const std::shared_ptr<PluginData> &pluginData, PluginParseErrors &error) {
if (!pluginData->buffer) {
error = PLUGIN_PARSE_ERROR_BUFFER_EMPTY;
DEBUG_FUNCTION_LINE_ERR("Buffer is empty");
return {};
}
ELFIO::elfio reader(new wiiu_zlib);
if (!reader.load(reinterpret_cast<const char *>(pluginData->buffer.get()), pluginData->length)) {
error = PLUGIN_PARSE_ERROR_ELFIO_PARSE_FAILED;
2022-04-22 22:55:53 +02:00
DEBUG_FUNCTION_LINE_ERR("Can't process PluginData in elfio");
return {};
}
return loadPlugin(reader, error);
}
2020-05-28 20:51:31 +02:00
std::optional<std::unique_ptr<PluginMetaInformation>> PluginMetaInformationFactory::loadPlugin(const std::string &filePath, PluginParseErrors &error) {
ELFIO::elfio reader(new wiiu_zlib);
uint8_t *buffer = nullptr;
uint32_t length = 0;
if (FSUtils::LoadFileToMem(filePath.c_str(), &buffer, &length) < 0) {
2022-04-22 22:55:53 +02:00
DEBUG_FUNCTION_LINE_ERR("Failed to load file to memory");
error = PLUGIN_PARSE_ERROR_IO_ERROR;
return {};
}
if (!reader.load(reinterpret_cast<const char *>(buffer), length)) {
error = PLUGIN_PARSE_ERROR_ELFIO_PARSE_FAILED;
2022-04-22 22:55:53 +02:00
DEBUG_FUNCTION_LINE_ERR("Can't process PluginData in elfio");
return {};
}
auto res = loadPlugin(reader, error);
free(buffer);
return res;
}
std::optional<std::unique_ptr<PluginMetaInformation>> PluginMetaInformationFactory::loadPlugin(char *buffer, size_t size, PluginParseErrors &error) {
ELFIO::elfio reader(new wiiu_zlib);
if (!reader.load(reinterpret_cast<const char *>(buffer), size)) {
error = PLUGIN_PARSE_ERROR_ELFIO_PARSE_FAILED;
2022-04-22 22:55:53 +02:00
DEBUG_FUNCTION_LINE_ERR("Can't find or process ELF file");
return std::nullopt;
}
return loadPlugin(reader, error);
}
std::optional<std::unique_ptr<PluginMetaInformation>> PluginMetaInformationFactory::loadPlugin(const ELFIO::elfio &reader, PluginParseErrors &error) {
size_t pluginSize = 0;
auto pluginInfo = std::unique_ptr<PluginMetaInformation>(new PluginMetaInformation);
uint32_t sec_num = reader.sections.size();
2020-05-03 12:30:15 +02:00
for (uint32_t i = 0; i < sec_num; ++i) {
ELFIO::section *psec = reader.sections[i];
// Calculate total size:
if ((psec->get_type() == ELFIO::SHT_PROGBITS || psec->get_type() == ELFIO::SHT_NOBITS) && (psec->get_flags() & ELFIO::SHF_ALLOC)) {
uint32_t sectionSize = psec->get_size();
2022-02-04 16:25:44 +01:00
auto address = (uint32_t) psec->get_address();
2020-05-03 12:30:15 +02:00
if ((address >= 0x02000000) && address < 0x10000000) {
pluginSize += sectionSize;
2020-05-03 12:30:15 +02:00
} else if ((address >= 0x10000000) && address < 0xC0000000) {
pluginSize += sectionSize;
}
}
// Get meta information and check WUPS version:
if (psec->get_name() == ".wups.meta") {
2020-05-03 12:30:15 +02:00
const void *sectionData = psec->get_data();
2022-02-04 16:25:44 +01:00
uint32_t sectionSize = psec->get_size();
2020-05-03 12:30:15 +02:00
char *curEntry = (char *) sectionData;
while ((uint32_t) curEntry < (uint32_t) sectionData + sectionSize) {
if (*curEntry == '\0') {
curEntry++;
continue;
}
auto firstFound = std::string(curEntry).find_first_of('=');
2020-05-03 12:30:15 +02:00
if (firstFound != std::string::npos) {
curEntry[firstFound] = '\0';
std::string key(curEntry);
std::string value(curEntry + firstFound + 1);
2021-09-25 14:26:18 +02:00
if (key == "name") {
2021-12-15 17:09:30 +01:00
pluginInfo->setName(value);
2021-09-25 14:26:18 +02:00
} else if (key == "author") {
2021-12-15 17:09:30 +01:00
pluginInfo->setAuthor(value);
2021-09-25 14:26:18 +02:00
} else if (key == "version") {
2021-12-15 17:09:30 +01:00
pluginInfo->setVersion(value);
2021-09-25 14:26:18 +02:00
} else if (key == "license") {
2021-12-15 17:09:30 +01:00
pluginInfo->setLicense(value);
2021-09-25 14:26:18 +02:00
} else if (key == "buildtimestamp") {
2021-12-15 17:09:30 +01:00
pluginInfo->setBuildTimestamp(value);
2021-09-25 14:26:18 +02:00
} else if (key == "description") {
2021-12-15 17:09:30 +01:00
pluginInfo->setDescription(value);
2021-10-01 17:25:48 +02:00
} else if (key == "storage_id") {
2021-12-15 17:09:30 +01:00
pluginInfo->setStorageId(value);
2021-09-25 14:26:18 +02:00
} else if (key == "wups") {
if (value != "0.7.1") {
error = PLUGIN_PARSE_ERROR_INCOMPATIBLE_VERSION;
2022-04-22 22:55:53 +02:00
DEBUG_FUNCTION_LINE_ERR("Warning: Ignoring plugin - Unsupported WUPS version: %s.", value.c_str());
return std::nullopt;
}
}
}
curEntry += strlen(curEntry) + 1;
}
}
}
2021-12-15 17:09:30 +01:00
pluginInfo->setSize(pluginSize);
error = PLUGIN_PARSE_ERROR_NONE;
return pluginInfo;
}