2021-12-03 22:40:19 +01:00
|
|
|
// Copyright 2021 Dolphin Emulator Project
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
|
|
|
|
#include "DolphinTool/VerifyCommand.h"
|
2023-06-15 01:33:11 +02:00
|
|
|
|
2023-06-15 06:27:14 +02:00
|
|
|
#include <cstdlib>
|
2023-06-15 01:33:11 +02:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2021-12-03 22:40:19 +01:00
|
|
|
|
|
|
|
#include <OptionParser.h>
|
|
|
|
|
2023-06-15 01:33:11 +02:00
|
|
|
#include "Common/StringUtil.h"
|
|
|
|
#include "DiscIO/VolumeDisc.h"
|
|
|
|
#include "DiscIO/VolumeVerifier.h"
|
|
|
|
#include "UICommon/UICommon.h"
|
|
|
|
|
2021-12-03 22:40:19 +01:00
|
|
|
namespace DolphinTool
|
|
|
|
{
|
2023-06-15 01:33:11 +02:00
|
|
|
static std::string HashToHexString(const std::vector<u8>& hash)
|
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << std::hex;
|
|
|
|
for (int i = 0; i < static_cast<int>(hash.size()); ++i)
|
|
|
|
{
|
|
|
|
ss << std::setw(2) << std::setfill('0') << static_cast<int>(hash[i]);
|
|
|
|
}
|
|
|
|
return ss.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void PrintFullReport(const DiscIO::VolumeVerifier::Result& result)
|
|
|
|
{
|
|
|
|
if (!result.hashes.crc32.empty())
|
|
|
|
std::cout << "CRC32: " << HashToHexString(result.hashes.crc32) << std::endl;
|
|
|
|
else
|
|
|
|
std::cout << "CRC32 not computed" << std::endl;
|
|
|
|
|
|
|
|
if (!result.hashes.md5.empty())
|
|
|
|
std::cout << "MD5: " << HashToHexString(result.hashes.md5) << std::endl;
|
|
|
|
else
|
|
|
|
std::cout << "MD5 not computed" << std::endl;
|
|
|
|
|
|
|
|
if (!result.hashes.sha1.empty())
|
|
|
|
std::cout << "SHA1: " << HashToHexString(result.hashes.sha1) << std::endl;
|
|
|
|
else
|
|
|
|
std::cout << "SHA1 not computed" << std::endl;
|
|
|
|
|
|
|
|
std::cout << "Problems Found: " << (result.problems.empty() ? "No" : "Yes") << std::endl;
|
|
|
|
|
|
|
|
for (const auto& problem : result.problems)
|
|
|
|
{
|
|
|
|
std::cout << std::endl << "Severity: ";
|
|
|
|
switch (problem.severity)
|
|
|
|
{
|
|
|
|
case DiscIO::VolumeVerifier::Severity::Low:
|
|
|
|
std::cout << "Low";
|
|
|
|
break;
|
|
|
|
case DiscIO::VolumeVerifier::Severity::Medium:
|
|
|
|
std::cout << "Medium";
|
|
|
|
break;
|
|
|
|
case DiscIO::VolumeVerifier::Severity::High:
|
|
|
|
std::cout << "High";
|
|
|
|
break;
|
|
|
|
case DiscIO::VolumeVerifier::Severity::None:
|
|
|
|
std::cout << "None";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ASSERT(false);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
|
|
|
std::cout << "Summary: " << problem.text << std::endl << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int VerifyCommand(const std::vector<std::string>& args)
|
2021-12-03 22:40:19 +01:00
|
|
|
{
|
2023-06-14 22:14:03 +02:00
|
|
|
optparse::OptionParser parser;
|
2021-12-03 22:40:19 +01:00
|
|
|
|
2023-06-14 22:14:03 +02:00
|
|
|
parser.usage("usage: verify [options]...");
|
2021-12-03 22:40:19 +01:00
|
|
|
|
2023-06-14 22:14:03 +02:00
|
|
|
parser.add_option("-u", "--user")
|
2022-01-02 09:07:37 +01:00
|
|
|
.action("store")
|
2022-05-02 13:19:01 +02:00
|
|
|
.help("User folder path, required for temporary processing files. "
|
2022-01-02 09:07:37 +01:00
|
|
|
"Will be automatically created if this option is not set.");
|
|
|
|
|
2023-06-14 22:14:03 +02:00
|
|
|
parser.add_option("-i", "--input")
|
2021-12-03 22:40:19 +01:00
|
|
|
.type("string")
|
|
|
|
.action("store")
|
|
|
|
.help("Path to disc image FILE.")
|
|
|
|
.metavar("FILE");
|
|
|
|
|
2023-06-14 22:14:03 +02:00
|
|
|
parser.add_option("-a", "--algorithm")
|
2021-12-03 22:40:19 +01:00
|
|
|
.type("string")
|
|
|
|
.action("store")
|
|
|
|
.help("Optional. Compute and print the digest using the selected algorithm, then exit. "
|
|
|
|
"[%choices]")
|
|
|
|
.choices({"crc32", "md5", "sha1"});
|
|
|
|
|
2023-06-14 22:14:03 +02:00
|
|
|
const optparse::Values& options = parser.parse_args(args);
|
2021-12-03 22:40:19 +01:00
|
|
|
|
2022-01-02 09:07:37 +01:00
|
|
|
// Initialize the dolphin user directory, required for temporary processing files
|
|
|
|
// If this is not set, destructive file operations could occur due to path confusion
|
|
|
|
std::string user_directory;
|
|
|
|
if (options.is_set("user"))
|
|
|
|
user_directory = static_cast<const char*>(options.get("user"));
|
|
|
|
|
|
|
|
UICommon::SetUserDirectory(user_directory);
|
|
|
|
UICommon::Init();
|
|
|
|
|
2021-12-03 22:40:19 +01:00
|
|
|
// Validate options
|
|
|
|
const std::string input_file_path = static_cast<const char*>(options.get("input"));
|
|
|
|
if (input_file_path.empty())
|
|
|
|
{
|
|
|
|
std::cerr << "Error: No input set" << std::endl;
|
2023-06-15 06:27:14 +02:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 22:40:19 +01:00
|
|
|
}
|
|
|
|
|
2022-07-28 19:31:16 +02:00
|
|
|
DiscIO::Hashes<bool> hashes_to_calculate{};
|
2023-06-14 22:14:03 +02:00
|
|
|
const bool algorithm_is_set = options.is_set("algorithm");
|
|
|
|
if (!algorithm_is_set)
|
2022-07-28 19:31:16 +02:00
|
|
|
{
|
|
|
|
hashes_to_calculate = DiscIO::VolumeVerifier::GetDefaultHashesToCalculate();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-06-14 22:14:03 +02:00
|
|
|
const std::string algorithm = static_cast<const char*>(options.get("algorithm"));
|
2022-07-28 19:31:16 +02:00
|
|
|
if (algorithm == "crc32")
|
|
|
|
hashes_to_calculate.crc32 = true;
|
|
|
|
else if (algorithm == "md5")
|
|
|
|
hashes_to_calculate.md5 = true;
|
|
|
|
else if (algorithm == "sha1")
|
|
|
|
hashes_to_calculate.sha1 = true;
|
|
|
|
}
|
2021-12-03 22:40:19 +01:00
|
|
|
|
2022-07-28 19:31:16 +02:00
|
|
|
if (!hashes_to_calculate.crc32 && !hashes_to_calculate.md5 && !hashes_to_calculate.sha1)
|
2021-12-03 22:40:19 +01:00
|
|
|
{
|
|
|
|
// optparse should protect from this
|
|
|
|
std::cerr << "Error: No algorithms selected for the operation" << std::endl;
|
2023-06-15 06:27:14 +02:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 22:40:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Open the volume
|
2023-06-14 22:14:03 +02:00
|
|
|
const std::unique_ptr<DiscIO::VolumeDisc> volume = DiscIO::CreateDisc(input_file_path);
|
2021-12-03 22:40:19 +01:00
|
|
|
if (!volume)
|
|
|
|
{
|
|
|
|
std::cerr << "Error: Unable to open disc image" << std::endl;
|
2023-06-15 06:27:14 +02:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 22:40:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Verify the volume
|
2023-06-14 22:14:03 +02:00
|
|
|
DiscIO::VolumeVerifier verifier(*volume, false, hashes_to_calculate);
|
|
|
|
verifier.Start();
|
|
|
|
while (verifier.GetBytesProcessed() != verifier.GetTotalBytes())
|
2021-12-03 22:40:19 +01:00
|
|
|
{
|
2023-06-14 22:14:03 +02:00
|
|
|
verifier.Process();
|
2021-12-03 22:40:19 +01:00
|
|
|
}
|
2023-06-14 22:14:03 +02:00
|
|
|
verifier.Finish();
|
|
|
|
const DiscIO::VolumeVerifier::Result& result = verifier.GetResult();
|
2021-12-03 22:40:19 +01:00
|
|
|
|
2023-06-14 22:14:03 +02:00
|
|
|
// Print the report
|
|
|
|
if (!algorithm_is_set)
|
2021-12-03 22:40:19 +01:00
|
|
|
{
|
|
|
|
PrintFullReport(result);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-06-14 22:14:03 +02:00
|
|
|
if (hashes_to_calculate.crc32 && !result.hashes.crc32.empty())
|
|
|
|
std::cout << HashToHexString(result.hashes.crc32) << std::endl;
|
|
|
|
else if (hashes_to_calculate.md5 && !result.hashes.md5.empty())
|
|
|
|
std::cout << HashToHexString(result.hashes.md5) << std::endl;
|
|
|
|
else if (hashes_to_calculate.sha1 && !result.hashes.sha1.empty())
|
|
|
|
std::cout << HashToHexString(result.hashes.sha1) << std::endl;
|
2021-12-03 22:40:19 +01:00
|
|
|
else
|
|
|
|
{
|
2022-07-28 19:31:16 +02:00
|
|
|
std::cerr << "Error: No hash computed" << std::endl;
|
2023-06-15 06:27:14 +02:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 22:40:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-15 06:27:14 +02:00
|
|
|
return EXIT_SUCCESS;
|
2021-12-03 22:40:19 +01:00
|
|
|
}
|
|
|
|
} // namespace DolphinTool
|