2021-12-03 15:40:19 -06:00
|
|
|
// Copyright 2021 Dolphin Emulator Project
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
|
|
|
|
#include "DolphinTool/ConvertCommand.h"
|
2022-03-12 09:48:12 +01:00
|
|
|
|
2023-06-14 23:27:14 -05:00
|
|
|
#include <cstdlib>
|
2022-03-12 09:48:12 +01:00
|
|
|
#include <iostream>
|
|
|
|
#include <limits>
|
|
|
|
#include <optional>
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2021-12-03 15:40:19 -06:00
|
|
|
|
|
|
|
#include <OptionParser.h>
|
2023-06-16 18:59:30 -05:00
|
|
|
#include <fmt/format.h>
|
|
|
|
#include <fmt/ostream.h>
|
2021-12-03 15:40:19 -06:00
|
|
|
|
2022-03-12 09:48:12 +01:00
|
|
|
#include "Common/CommonTypes.h"
|
|
|
|
#include "DiscIO/Blob.h"
|
|
|
|
#include "DiscIO/DiscUtils.h"
|
|
|
|
#include "DiscIO/ScrubbedBlob.h"
|
|
|
|
#include "DiscIO/Volume.h"
|
|
|
|
#include "DiscIO/VolumeDisc.h"
|
|
|
|
#include "DiscIO/WIABlob.h"
|
|
|
|
#include "UICommon/UICommon.h"
|
|
|
|
|
2021-12-03 15:40:19 -06:00
|
|
|
namespace DolphinTool
|
|
|
|
{
|
2023-06-14 18:33:11 -05:00
|
|
|
static std::optional<DiscIO::WIARVZCompressionType>
|
|
|
|
ParseCompressionTypeString(const std::string& compression_str)
|
|
|
|
{
|
|
|
|
if (compression_str == "none")
|
|
|
|
return DiscIO::WIARVZCompressionType::None;
|
|
|
|
else if (compression_str == "purge")
|
|
|
|
return DiscIO::WIARVZCompressionType::Purge;
|
|
|
|
else if (compression_str == "bzip2")
|
|
|
|
return DiscIO::WIARVZCompressionType::Bzip2;
|
|
|
|
else if (compression_str == "lzma")
|
|
|
|
return DiscIO::WIARVZCompressionType::LZMA;
|
|
|
|
else if (compression_str == "lzma2")
|
|
|
|
return DiscIO::WIARVZCompressionType::LZMA2;
|
|
|
|
else if (compression_str == "zstd")
|
|
|
|
return DiscIO::WIARVZCompressionType::Zstd;
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::optional<DiscIO::BlobType> ParseFormatString(const std::string& format_str)
|
|
|
|
{
|
|
|
|
if (format_str == "iso")
|
|
|
|
return DiscIO::BlobType::PLAIN;
|
|
|
|
else if (format_str == "gcz")
|
|
|
|
return DiscIO::BlobType::GCZ;
|
|
|
|
else if (format_str == "wia")
|
|
|
|
return DiscIO::BlobType::WIA;
|
|
|
|
else if (format_str == "rvz")
|
|
|
|
return DiscIO::BlobType::RVZ;
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ConvertCommand(const std::vector<std::string>& args)
|
2021-12-03 15:40:19 -06:00
|
|
|
{
|
2022-03-08 01:57:08 -06:00
|
|
|
optparse::OptionParser parser;
|
2021-12-03 15:40:19 -06:00
|
|
|
|
2022-03-08 01:57:08 -06:00
|
|
|
parser.usage("usage: convert [options]... [FILE]...");
|
2021-12-03 15:40:19 -06:00
|
|
|
|
2022-03-08 01:57:08 -06:00
|
|
|
parser.add_option("-u", "--user")
|
2023-06-16 19:33:38 -05:00
|
|
|
.type("string")
|
2022-01-02 02:07:37 -06:00
|
|
|
.action("store")
|
2022-05-02 12:19:01 +01:00
|
|
|
.help("User folder path, required for temporary processing files. "
|
2023-06-16 19:33:38 -05:00
|
|
|
"Will be automatically created if this option is not set.")
|
|
|
|
.set_default("");
|
2022-01-02 02:07:37 -06:00
|
|
|
|
2022-03-08 01:57:08 -06:00
|
|
|
parser.add_option("-i", "--input")
|
2021-12-03 15:40:19 -06:00
|
|
|
.type("string")
|
|
|
|
.action("store")
|
|
|
|
.help("Path to disc image FILE.")
|
|
|
|
.metavar("FILE");
|
|
|
|
|
2022-03-08 01:57:08 -06:00
|
|
|
parser.add_option("-o", "--output")
|
2021-12-03 15:40:19 -06:00
|
|
|
.type("string")
|
|
|
|
.action("store")
|
|
|
|
.help("Path to the destination FILE.")
|
|
|
|
.metavar("FILE");
|
|
|
|
|
2022-03-08 01:57:08 -06:00
|
|
|
parser.add_option("-f", "--format")
|
2021-12-03 15:40:19 -06:00
|
|
|
.type("string")
|
|
|
|
.action("store")
|
|
|
|
.help("Container format to use. Default is RVZ. [%choices]")
|
|
|
|
.choices({"iso", "gcz", "wia", "rvz"});
|
|
|
|
|
2022-03-08 01:57:08 -06:00
|
|
|
parser.add_option("-s", "--scrub")
|
2021-12-03 15:40:19 -06:00
|
|
|
.action("store_true")
|
|
|
|
.help("Scrub junk data as part of conversion.");
|
|
|
|
|
2022-03-08 01:57:08 -06:00
|
|
|
parser.add_option("-b", "--block_size")
|
2021-12-03 15:40:19 -06:00
|
|
|
.type("int")
|
|
|
|
.action("store")
|
|
|
|
.help("Block size for GCZ/WIA/RVZ formats, as an integer. Suggested value for RVZ: 131072 "
|
|
|
|
"(128 KiB)");
|
|
|
|
|
2022-03-08 01:57:08 -06:00
|
|
|
parser.add_option("-c", "--compression")
|
2021-12-03 15:40:19 -06:00
|
|
|
.type("string")
|
|
|
|
.action("store")
|
|
|
|
.help("Compression method to use when converting to WIA/RVZ. Suggested value for RVZ: zstd "
|
|
|
|
"[%choices]")
|
2024-02-13 11:44:41 -08:00
|
|
|
.choices({"none", "zstd", "bzip2", "lzma", "lzma2"});
|
2021-12-03 15:40:19 -06:00
|
|
|
|
2022-03-08 01:57:08 -06:00
|
|
|
parser.add_option("-l", "--compression_level")
|
2021-12-03 15:40:19 -06:00
|
|
|
.type("int")
|
|
|
|
.action("store")
|
|
|
|
.help("Level of compression for the selected method. Ignored if 'none'. Suggested value for "
|
|
|
|
"zstd: 5");
|
|
|
|
|
2022-03-08 01:57:08 -06:00
|
|
|
const optparse::Values& options = parser.parse_args(args);
|
2021-12-03 15:40:19 -06:00
|
|
|
|
2022-01-02 02:07:37 -06: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
|
2023-06-16 19:33:38 -05:00
|
|
|
UICommon::SetUserDirectory(options["user"]);
|
2022-01-02 02:07:37 -06:00
|
|
|
UICommon::Init();
|
|
|
|
|
2021-12-03 15:40:19 -06:00
|
|
|
// Validate options
|
|
|
|
|
|
|
|
// --input
|
2023-06-16 19:33:38 -05:00
|
|
|
if (!options.is_set("input"))
|
2021-12-03 15:40:19 -06:00
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: No input set\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
2023-06-16 19:33:38 -05:00
|
|
|
const std::string& input_file_path = options["input"];
|
2021-12-03 15:40:19 -06:00
|
|
|
|
|
|
|
// --output
|
2023-06-16 19:33:38 -05:00
|
|
|
if (!options.is_set("output"))
|
2021-12-03 15:40:19 -06:00
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: No output set\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
2023-06-16 19:33:38 -05:00
|
|
|
const std::string& output_file_path = options["output"];
|
2021-12-03 15:40:19 -06:00
|
|
|
|
|
|
|
// --format
|
2023-06-16 19:33:38 -05:00
|
|
|
const std::optional<DiscIO::BlobType> format_o = ParseFormatString(options["format"]);
|
2021-12-03 15:40:19 -06:00
|
|
|
if (!format_o.has_value())
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: No output format set\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
const DiscIO::BlobType format = format_o.value();
|
|
|
|
|
2022-03-12 09:48:12 +01:00
|
|
|
// Open the blob reader
|
|
|
|
std::unique_ptr<DiscIO::BlobReader> blob_reader = DiscIO::CreateBlobReader(input_file_path);
|
|
|
|
if (!blob_reader)
|
2021-12-03 15:40:19 -06:00
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: The input file could not be opened.\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// --scrub
|
|
|
|
const bool scrub = static_cast<bool>(options.get("scrub"));
|
|
|
|
|
2022-03-12 09:48:12 +01:00
|
|
|
// Open the volume
|
|
|
|
std::unique_ptr<DiscIO::Volume> volume = DiscIO::CreateDisc(input_file_path);
|
|
|
|
if (!volume)
|
2021-12-03 15:40:19 -06:00
|
|
|
{
|
2022-03-12 09:48:12 +01:00
|
|
|
if (scrub)
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: Scrubbing is only supported for GC/Wii disc images.\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2022-03-12 09:48:12 +01:00
|
|
|
}
|
|
|
|
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr,
|
|
|
|
"Warning: The input file is not a GC/Wii disc image. Continuing anyway.\n");
|
2022-03-12 09:48:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (scrub)
|
|
|
|
{
|
|
|
|
if (volume->IsDatelDisc())
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: Scrubbing a Datel disc is not supported.\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2022-03-12 09:48:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
blob_reader = DiscIO::ScrubbedBlob::Create(input_file_path);
|
|
|
|
|
|
|
|
if (!blob_reader)
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: Unable to process disc image. Try again without --scrub.\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2022-03-12 09:48:12 +01:00
|
|
|
}
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (scrub && format == DiscIO::BlobType::RVZ)
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Warning: Scrubbing an RVZ container does not offer significant space "
|
|
|
|
"advantages. Continuing anyway.\n");
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (scrub && format == DiscIO::BlobType::PLAIN)
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Warning: Scrubbing does not save space when converting to ISO unless "
|
|
|
|
"using external compression. Continuing anyway.\n");
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
2022-03-12 09:48:12 +01:00
|
|
|
if (!scrub && format == DiscIO::BlobType::GCZ && volume &&
|
2021-12-03 15:40:19 -06:00
|
|
|
volume->GetVolumeType() == DiscIO::Platform::WiiDisc && !volume->IsDatelDisc())
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Warning: Converting Wii disc images to GCZ without scrubbing may not "
|
|
|
|
"offer space advantages over ISO. Continuing anyway.\n");
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
2022-03-12 09:48:12 +01:00
|
|
|
if (volume && volume->IsNKit())
|
2021-12-03 15:40:19 -06:00
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr,
|
|
|
|
"Warning: Converting an NKit file, output will still be NKit! Continuing anyway.\n");
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// --block_size
|
|
|
|
std::optional<int> block_size_o;
|
|
|
|
if (options.is_set("block_size"))
|
|
|
|
block_size_o = static_cast<int>(options.get("block_size"));
|
|
|
|
|
|
|
|
if (format == DiscIO::BlobType::GCZ || format == DiscIO::BlobType::WIA ||
|
|
|
|
format == DiscIO::BlobType::RVZ)
|
|
|
|
{
|
|
|
|
if (!block_size_o.has_value())
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: Block size must be set for GCZ/RVZ/WIA\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!DiscIO::IsDiscImageBlockSizeValid(block_size_o.value(), format))
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: Block size is not valid for this format\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (block_size_o.value() < DiscIO::PREFERRED_MIN_BLOCK_SIZE ||
|
|
|
|
block_size_o.value() > DiscIO::PREFERRED_MAX_BLOCK_SIZE)
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr,
|
|
|
|
"Warning: Block size is not ideal for performance. Continuing anyway.\n");
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
2022-03-12 09:48:12 +01:00
|
|
|
if (format == DiscIO::BlobType::GCZ && volume &&
|
2022-08-01 11:53:30 +02:00
|
|
|
!DiscIO::IsGCZBlockSizeLegacyCompatible(block_size_o.value(), volume->GetDataSize()))
|
2021-12-03 15:40:19 -06:00
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr,
|
|
|
|
"Warning: For GCZs to be compatible with Dolphin < 5.0-11893, the file size "
|
|
|
|
"must be an integer multiple of the block size and must not be an integer "
|
|
|
|
"multiple of the block size multiplied by 32. Continuing anyway.\n");
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// --compress, --compress_level
|
|
|
|
std::optional<DiscIO::WIARVZCompressionType> compression_o =
|
2023-06-16 19:33:38 -05:00
|
|
|
ParseCompressionTypeString(options["compression"]);
|
2021-12-03 15:40:19 -06:00
|
|
|
|
|
|
|
std::optional<int> compression_level_o;
|
|
|
|
if (options.is_set("compression_level"))
|
|
|
|
compression_level_o = static_cast<int>(options.get("compression_level"));
|
|
|
|
|
|
|
|
if (format == DiscIO::BlobType::WIA || format == DiscIO::BlobType::RVZ)
|
|
|
|
{
|
|
|
|
if (!compression_o.has_value())
|
|
|
|
{
|
2024-08-11 22:15:56 +01:00
|
|
|
fmt::print(std::cerr, "Error: Compression method must be set for WIA or RVZ\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if ((format == DiscIO::BlobType::WIA &&
|
|
|
|
compression_o.value() == DiscIO::WIARVZCompressionType::Zstd) ||
|
|
|
|
(format == DiscIO::BlobType::RVZ &&
|
|
|
|
compression_o.value() == DiscIO::WIARVZCompressionType::Purge))
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: Compression type is not supported for the container format\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (compression_o.value() == DiscIO::WIARVZCompressionType::None)
|
|
|
|
{
|
|
|
|
compression_level_o = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!compression_level_o.has_value())
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr,
|
|
|
|
"Error: Compression level must be set when compression type is not 'none'\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
2022-02-24 04:51:52 -06:00
|
|
|
const std::pair<int, int> range =
|
|
|
|
DiscIO::GetAllowedCompressionLevels(compression_o.value(), false);
|
2021-12-03 15:40:19 -06:00
|
|
|
if (compression_level_o.value() < range.first || compression_level_o.value() > range.second)
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: Compression level not in acceptable range\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Perform the conversion
|
|
|
|
const auto NOOP_STATUS_CALLBACK = [](const std::string& text, float percent) { return true; };
|
|
|
|
|
|
|
|
bool success = false;
|
|
|
|
|
|
|
|
switch (format)
|
|
|
|
{
|
|
|
|
case DiscIO::BlobType::PLAIN:
|
2022-03-12 09:48:12 +01:00
|
|
|
{
|
2021-12-03 15:40:19 -06:00
|
|
|
success = DiscIO::ConvertToPlain(blob_reader.get(), input_file_path, output_file_path,
|
|
|
|
NOOP_STATUS_CALLBACK);
|
|
|
|
break;
|
2022-03-12 09:48:12 +01:00
|
|
|
}
|
2021-12-03 15:40:19 -06:00
|
|
|
|
|
|
|
case DiscIO::BlobType::GCZ:
|
2022-03-12 09:48:12 +01:00
|
|
|
{
|
|
|
|
u32 sub_type = std::numeric_limits<u32>::max();
|
|
|
|
if (volume)
|
|
|
|
{
|
|
|
|
if (volume->GetVolumeType() == DiscIO::Platform::GameCubeDisc)
|
|
|
|
sub_type = 0;
|
|
|
|
else if (volume->GetVolumeType() == DiscIO::Platform::WiiDisc)
|
|
|
|
sub_type = 1;
|
|
|
|
}
|
|
|
|
success = DiscIO::ConvertToGCZ(blob_reader.get(), input_file_path, output_file_path, sub_type,
|
2021-12-03 15:40:19 -06:00
|
|
|
block_size_o.value(), NOOP_STATUS_CALLBACK);
|
|
|
|
break;
|
2022-03-12 09:48:12 +01:00
|
|
|
}
|
2021-12-03 15:40:19 -06:00
|
|
|
|
|
|
|
case DiscIO::BlobType::WIA:
|
|
|
|
case DiscIO::BlobType::RVZ:
|
2022-03-12 09:48:12 +01:00
|
|
|
{
|
2021-12-03 15:40:19 -06:00
|
|
|
success = DiscIO::ConvertToWIAOrRVZ(blob_reader.get(), input_file_path, output_file_path,
|
|
|
|
format == DiscIO::BlobType::RVZ, compression_o.value(),
|
|
|
|
compression_level_o.value(), block_size_o.value(),
|
|
|
|
NOOP_STATUS_CALLBACK);
|
|
|
|
break;
|
2022-03-12 09:48:12 +01:00
|
|
|
}
|
2021-12-03 15:40:19 -06:00
|
|
|
|
|
|
|
default:
|
2022-03-12 09:48:12 +01:00
|
|
|
{
|
2021-12-03 15:40:19 -06:00
|
|
|
ASSERT(false);
|
|
|
|
break;
|
|
|
|
}
|
2022-03-12 09:48:12 +01:00
|
|
|
}
|
2021-12-03 15:40:19 -06:00
|
|
|
|
|
|
|
if (!success)
|
|
|
|
{
|
2023-06-16 18:59:30 -05:00
|
|
|
fmt::print(std::cerr, "Error: Conversion failed\n");
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_FAILURE;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
|
2023-06-14 23:27:14 -05:00
|
|
|
return EXIT_SUCCESS;
|
2021-12-03 15:40:19 -06:00
|
|
|
}
|
|
|
|
} // namespace DolphinTool
|