amuse/driver/amuseconv.cpp

184 lines
5.4 KiB
C++
Raw Normal View History

#include "amuse/amuse.hpp"
#include "athena/FileReader.hpp"
#include "athena/DNAYaml.hpp"
#include "logvisor/logvisor.hpp"
2017-12-29 07:57:22 +00:00
#include <cstdio>
#include <cstring>
2016-06-22 22:15:53 +00:00
static logvisor::Module Log("amuseconv");
2018-12-08 05:20:09 +00:00
enum ConvType { ConvN64, ConvGCN, ConvPC };
static void ReportConvType(ConvType tp) {
switch (tp) {
case ConvN64:
2019-07-20 04:23:25 +00:00
Log.report(logvisor::Info, fmt(_SYS_STR("using N64 format")));
2018-12-08 05:20:09 +00:00
break;
case ConvPC:
2019-07-20 04:23:25 +00:00
Log.report(logvisor::Info, fmt(_SYS_STR("using PC format")));
2018-12-08 05:20:09 +00:00
break;
case ConvGCN:
default:
2019-07-20 04:23:25 +00:00
Log.report(logvisor::Info, fmt(_SYS_STR("using GameCube format")));
2018-12-08 05:20:09 +00:00
break;
}
2016-06-22 21:43:45 +00:00
}
2018-12-08 05:20:09 +00:00
static bool BuildAudioGroup(amuse::SystemStringView groupBase, amuse::SystemStringView targetPath) { return true; }
2018-12-08 05:20:09 +00:00
static bool ExtractAudioGroup(amuse::SystemStringView inPath, amuse::SystemStringView targetPath) {
amuse::ContainerRegistry::Type type;
auto groups = amuse::ContainerRegistry::LoadContainer(inPath.data(), type);
2018-12-08 05:20:09 +00:00
if (groups.size()) {
2019-07-20 04:23:25 +00:00
Log.report(logvisor::Info, fmt(_SYS_STR("Found '{}'")), amuse::ContainerRegistry::TypeToName(type));
2018-12-08 05:20:09 +00:00
amuse::Mkdir(targetPath.data(), 0755);
2019-07-20 04:23:25 +00:00
Log.report(logvisor::Info, fmt(_SYS_STR("Established directory at {}")), targetPath);
2018-12-08 05:20:09 +00:00
for (auto& group : groups) {
2019-07-20 04:23:25 +00:00
Log.report(logvisor::Info, fmt(_SYS_STR("Extracting {}")), group.first);
2018-12-08 05:20:09 +00:00
}
}
auto songs = amuse::ContainerRegistry::LoadSongs(inPath.data());
amuse::SystemString songsDir = amuse::SystemString(targetPath) + _SYS_STR("/midifiles");
bool madeDir = false;
for (auto& pair : songs) {
if (!madeDir) {
amuse::Mkdir(targetPath.data(), 0755);
amuse::Mkdir(songsDir.c_str(), 0755);
madeDir = true;
}
2018-12-08 05:20:09 +00:00
amuse::SystemString songPath = songsDir + _SYS_STR('/') + pair.first + _SYS_STR(".mid");
FILE* fp = amuse::FOpen(songPath.c_str(), _SYS_STR("wb"));
if (fp) {
2019-07-20 04:23:25 +00:00
Log.report(logvisor::Info, fmt(_SYS_STR("Extracting {}")), pair.first);
2018-12-08 05:20:09 +00:00
int extractedVersion;
bool isBig;
std::vector<uint8_t> mid = amuse::SongConverter::SongToMIDI(pair.second.m_data.get(), extractedVersion, isBig);
fwrite(mid.data(), 1, mid.size(), fp);
fclose(fp);
}
2018-12-08 05:20:09 +00:00
}
2018-12-08 05:20:09 +00:00
return true;
}
2018-12-08 05:20:09 +00:00
static bool BuildSNG(amuse::SystemStringView inPath, amuse::SystemStringView targetPath, int version, bool big) {
FILE* fp = amuse::FOpen(inPath.data(), _SYS_STR("rb"));
if (!fp)
return false;
fseek(fp, 0, SEEK_END);
long sz = ftell(fp);
fseek(fp, 0, SEEK_SET);
std::vector<uint8_t> data(sz, 0);
fread(&data[0], 1, sz, fp);
fclose(fp);
std::vector<uint8_t> out = amuse::SongConverter::MIDIToSong(data, version, big);
if (out.empty())
return false;
fp = amuse::FOpen(targetPath.data(), _SYS_STR("wb"));
fwrite(out.data(), 1, out.size(), fp);
fclose(fp);
return true;
}
2018-12-08 05:20:09 +00:00
static bool ExtractSNG(amuse::SystemStringView inPath, amuse::SystemStringView targetPath) {
FILE* fp = amuse::FOpen(inPath.data(), _SYS_STR("rb"));
if (!fp)
return false;
fseek(fp, 0, SEEK_END);
long sz = ftell(fp);
fseek(fp, 0, SEEK_SET);
std::vector<uint8_t> data(sz, 0);
fread(&data[0], 1, sz, fp);
fclose(fp);
int extractedVersion;
bool isBig;
std::vector<uint8_t> out = amuse::SongConverter::SongToMIDI(data.data(), extractedVersion, isBig);
if (out.empty())
return false;
fp = amuse::FOpen(targetPath.data(), _SYS_STR("wb"));
fwrite(out.data(), 1, out.size(), fp);
fclose(fp);
return true;
}
#if _WIN32
int wmain(int argc, const amuse::SystemChar** argv)
#else
int main(int argc, const amuse::SystemChar** argv)
#endif
{
2018-12-08 05:20:09 +00:00
logvisor::RegisterConsoleLogger();
2018-12-08 05:20:09 +00:00
if (argc < 3) {
2019-07-20 04:23:25 +00:00
fmt::print(fmt("Usage: amuseconv <in-file> <out-file> [n64|pc|gcn]\n"));
2018-12-08 05:20:09 +00:00
return 0;
}
ConvType type = ConvGCN;
if (argc >= 4) {
if (!amuse::CompareCaseInsensitive(argv[3], _SYS_STR("n64")))
type = ConvN64;
else if (!amuse::CompareCaseInsensitive(argv[3], _SYS_STR("gcn")))
type = ConvGCN;
else if (!amuse::CompareCaseInsensitive(argv[3], _SYS_STR("pc")))
type = ConvPC;
else {
2019-07-20 04:23:25 +00:00
Log.report(logvisor::Error, fmt(_SYS_STR("unrecognized format: {}")), argv[3]);
2018-12-08 05:20:09 +00:00
return 1;
}
2018-12-08 05:20:09 +00:00
}
bool good = false;
FILE* fin = amuse::FOpen(argv[1], _SYS_STR("rb"));
if (fin) {
fclose(fin);
amuse::SystemString barePath(argv[1]);
size_t dotPos = barePath.rfind(_SYS_STR('.'));
const amuse::SystemChar* dot = barePath.c_str() + dotPos;
if (dotPos != amuse::SystemString::npos) {
if (!amuse::CompareCaseInsensitive(dot, _SYS_STR(".mid")) ||
!amuse::CompareCaseInsensitive(dot, _SYS_STR(".midi"))) {
ReportConvType(type);
good = BuildSNG(barePath, argv[2], 1, true);
} else if (!amuse::CompareCaseInsensitive(dot, _SYS_STR(".son")) ||
!amuse::CompareCaseInsensitive(dot, _SYS_STR(".sng"))) {
good = ExtractSNG(argv[1], argv[2]);
} else {
good = ExtractAudioGroup(argv[1], argv[2]);
}
}
2018-12-08 05:20:09 +00:00
} else {
amuse::Sstat theStat;
if (!amuse::Stat(argv[1], &theStat) && S_ISDIR(theStat.st_mode)) {
amuse::SystemString projectPath(argv[1]);
projectPath += _SYS_STR("/project.yaml");
fin = amuse::FOpen(projectPath.c_str(), _SYS_STR("rb"));
if (fin) {
fclose(fin);
2018-12-08 05:20:09 +00:00
ReportConvType(type);
good = BuildAudioGroup(argv[1], argv[2]);
}
}
2018-12-08 05:20:09 +00:00
}
2018-12-08 05:20:09 +00:00
if (!good) {
2019-07-20 04:23:25 +00:00
Log.report(logvisor::Error, fmt(_SYS_STR("unable to convert {} to {}")), argv[1], argv[2]);
2018-12-08 05:20:09 +00:00
return 1;
}
2018-12-08 05:20:09 +00:00
return 0;
}