#include #include #include "logvisor/logvisor.hpp" #include "nod/nod.hpp" static void printHelp() { fprintf(stderr, "Usage:\n" " nodtool extract [-f] []\n" " nodtool makegcn []\n" " nodtool makewii(sl|dl) []\n"); } #if NOD_UCS2 #ifdef strcasecmp #undef strcasecmp #endif #define strcasecmp _wcsicmp #define PRISize "Iu" int wmain(int argc, wchar_t* argv[]) #else #define PRISize "zu" int main(int argc, char* argv[]) #endif { if (argc < 3 || (!strcasecmp(argv[1], _S("makegcn")) && argc < 7) || (!strcasecmp(argv[1], _S("makewiisl")) && argc < 8) || (!strcasecmp(argv[1], _S("makewiidl")) && argc < 8)) { printHelp(); return 1; } /* Enable logging to console */ logvisor::RegisterConsoleLogger(); nod::ExtractionContext ctx = { true, true, [&](const std::string& str){ fprintf(stderr, "%s\n", str.c_str()); }}; const nod::SystemChar* inDir = nullptr; const nod::SystemChar* outDir = _S("."); for (int a=2 ; a disc = nod::OpenDiscFromImage(inDir, isWii); if (!disc) return 1; nod::Mkdir(outDir, 0755); if (isWii) static_cast(*disc).writeOutDataPartitionHeader( (nod::SystemString(outDir) + _S("/partition_head.bin")).c_str()); nod::Partition* dataPart = disc->getDataPartition(); if (!dataPart) return 1; if (!dataPart->extractToDirectory(outDir, ctx)) return 1; } else if (!strcasecmp(argv[1], _S("makegcn"))) { #if NOD_UCS2 if (wcslen(argv[2]) < 6) nod::LogModule.report(logvisor::Fatal, _S("game-id is not at least 6 characters")); #else if (strlen(argv[2]) < 6) nod::LogModule.report(logvisor::Fatal, _S("game-id is not at least 6 characters")); #endif /* Pre-validate paths */ nod::Sstat theStat; if (nod::Stat(argv[4], &theStat) || !S_ISDIR(theStat.st_mode)) nod::LogModule.report(logvisor::Fatal, _S("unable to stat %s as directory"), argv[4]); if (nod::Stat(argv[5], &theStat) || !S_ISREG(theStat.st_mode)) nod::LogModule.report(logvisor::Fatal, _S("unable to stat %s as file"), argv[5]); if (nod::Stat(argv[6], &theStat) || !S_ISREG(theStat.st_mode)) nod::LogModule.report(logvisor::Fatal, "unable to stat %s as file", argv[6]); nod::SystemString gameIdSys(argv[2]); nod::SystemUTF8View gameId(gameIdSys); nod::SystemString gameTitleSys(argv[3]); nod::SystemUTF8View gameTitle(gameTitleSys); size_t lastIdx = -1; auto progFunc = [&](size_t idx, const nod::SystemString& name, size_t bytes) { if (idx != lastIdx) { lastIdx = idx; printf("\n"); } if (bytes != -1) nod::Printf(_S("\r%s %" PRISize " B"), name.c_str(), bytes); else nod::Printf(_S("\r%s"), name.c_str()); fflush(stdout); }; bool ret; if (argc < 8) { nod::SystemString outPath(argv[4]); outPath.append(_S(".iso")); nod::DiscBuilderGCN b(outPath.c_str(), gameId.utf8_str().c_str(), gameTitle.utf8_str().c_str(), 0x0003EB60, progFunc); ret = b.buildFromDirectory(argv[4], argv[5], argv[6]); } else { nod::DiscBuilderGCN b(argv[7], gameId.utf8_str().c_str(), gameTitle.utf8_str().c_str(), 0x0003EB60, progFunc); ret = b.buildFromDirectory(argv[4], argv[5], argv[6]); } printf("\n"); if (!ret) return 1; } else if (!strcasecmp(argv[1], _S("makewiisl")) || !strcasecmp(argv[1], _S("makewiidl"))) { #if NOD_UCS2 if (wcslen(argv[2]) < 6) nod::LogModule.report(logvisor::Fatal, _S("game-id is not at least 6 characters")); #else if (strlen(argv[2]) < 6) nod::LogModule.report(logvisor::Fatal, _S("game-id is not at least 6 characters")); #endif /* Pre-validate paths */ nod::Sstat theStat; if (nod::Stat(argv[4], &theStat) || !S_ISDIR(theStat.st_mode)) nod::LogModule.report(logvisor::Fatal, _S("unable to stat %s as directory"), argv[4]); if (nod::Stat(argv[5], &theStat) || !S_ISREG(theStat.st_mode)) nod::LogModule.report(logvisor::Fatal, _S("unable to stat %s as file"), argv[5]); if (nod::Stat(argv[6], &theStat) || !S_ISREG(theStat.st_mode)) nod::LogModule.report(logvisor::Fatal, _S("unable to stat %s as file"), argv[6]); if (nod::Stat(argv[7], &theStat) || !S_ISREG(theStat.st_mode)) nod::LogModule.report(logvisor::Fatal, _S("unable to stat %s as file"), argv[7]); nod::SystemString gameIdSys(argv[2]); nod::SystemUTF8View gameId(gameIdSys); nod::SystemString gameTitleSys(argv[3]); nod::SystemUTF8View gameTitle(gameTitleSys); size_t lastIdx = -1; auto progFunc = [&](size_t idx, const nod::SystemString& name, size_t bytes) { if (idx != lastIdx) { lastIdx = idx; printf("\n"); } if (bytes != -1) nod::Printf(_S("\r%s %" PRISize " B"), name.c_str(), bytes); else nod::Printf(_S("\r%s"), name.c_str()); fflush(stdout); }; bool dual = (argv[1][7] == _S('d') || argv[1][7] == _S('D')); bool ret; if (argc < 9) { nod::SystemString outPath(argv[4]); outPath.append(_S(".iso")); nod::DiscBuilderWii b(outPath.c_str(), gameId.utf8_str().c_str(), gameTitle.utf8_str().c_str(), dual, progFunc); ret = b.buildFromDirectory(argv[4], argv[5], argv[6], argv[7]); } else { nod::DiscBuilderWii b(argv[8], gameId.utf8_str().c_str(), gameTitle.utf8_str().c_str(), dual, progFunc); ret = b.buildFromDirectory(argv[4], argv[5], argv[6], argv[7]); } printf("\n"); if (!ret) return 1; } else { printHelp(); return 1; } return 0; }