metaforce/DataSpec/SpecBase.cpp

406 lines
13 KiB
C++
Raw Normal View History

2015-11-27 14:21:19 -08:00
#if _WIN32
#define _CRT_RAND_S
#include <stdlib.h>
#endif
2015-07-01 16:50:39 -07:00
#include "SpecBase.hpp"
#include "Blender/BlenderSupport.hpp"
2016-09-18 16:47:48 -07:00
#include "hecl/Blender/BlenderConnection.hpp"
2015-11-09 18:07:15 -08:00
#include "DNACommon/DNACommon.hpp"
#include "DNACommon/TXTR.hpp"
2015-07-01 16:50:39 -07:00
2015-11-21 21:16:40 -08:00
#include <time.h>
2016-02-13 01:02:47 -08:00
namespace DataSpec
2015-07-01 16:50:39 -07:00
{
2016-03-04 15:04:53 -08:00
static logvisor::Module Log("urde::SpecBase");
2016-03-04 15:04:53 -08:00
static const hecl::SystemChar* MomErr[] =
2015-11-21 21:16:40 -08:00
{
_S("Your metroid is in another castle"),
_S("HECL is experiencing a PTSD attack"),
_S("Unable to freeze metroids"),
_S("Ridley ate your homework"),
2017-02-04 22:20:31 -08:00
_S("Expected 0 maternal symbolisms, found 2147483647"),
2015-11-21 21:16:40 -08:00
_S("Contradictive narratives unsupported"),
_S("Wiimote profile \"NES + Zapper\" not recognized"),
_S("Unable to find Waldo"),
2016-02-20 00:53:06 -08:00
_S("Expected Ridley, found furby"),
_S("Adam has not authorized this, please do not bug the developers"),
_S("Lady returned objection"),
_S("Unterminated plot thread 'Deleter' detected")
2015-11-21 21:16:40 -08:00
};
constexpr uint32_t MomErrCount = std::extent<decltype(MomErr)>::value;
SpecBase::SpecBase(const hecl::Database::DataSpecEntry* specEntry, hecl::Database::Project& project, bool pc)
: hecl::Database::IDataSpec(specEntry), m_project(project), m_pc(pc),
m_masterShader(project.getProjectWorkingPath(), ".hecl/RetroMasterShader.blend")
{
DataSpec::UniqueIDBridge::setThreadProject(m_project);
}
2015-11-09 18:07:15 -08:00
static const hecl::SystemString regNONE = _S("");
static const hecl::SystemString regE = _S("NTSC");
static const hecl::SystemString regJ = _S("NTSC-J");
static const hecl::SystemString regP = _S("PAL");
void SpecBase::setThreadProject()
{
UniqueIDBridge::setThreadProject(m_project);
}
2015-10-03 22:08:56 -07:00
bool SpecBase::canExtract(const ExtractPassInfo& info, std::vector<ExtractReport>& reps)
2015-07-01 16:50:39 -07:00
{
2016-03-04 15:04:53 -08:00
m_disc = nod::OpenDiscFromImage(info.srcpath.c_str(), m_isWii);
2015-07-15 18:57:34 -07:00
if (!m_disc)
2015-07-02 13:41:40 -07:00
return false;
2016-01-19 19:18:30 -08:00
const char* gameID = m_disc->getHeader().m_gameID;
2015-07-02 13:41:40 -07:00
2015-11-21 21:16:40 -08:00
if (!memcmp(gameID, "R3O", 3))
{
unsigned int t = time(nullptr);
2015-11-27 14:21:19 -08:00
#if _WIN32
rand_s(&t);
2016-02-20 00:53:06 -08:00
int r = t % MomErrCount;
2015-11-27 14:21:19 -08:00
#else
2016-02-20 00:53:06 -08:00
int r = rand_r(&t) % MomErrCount;
2015-11-27 14:21:19 -08:00
#endif
2016-03-04 15:04:53 -08:00
Log.report(logvisor::Fatal, MomErr[r]);
2015-11-21 21:16:40 -08:00
}
m_standalone = true;
if (m_isWii && !memcmp(gameID, "R3M", 3))
m_standalone = false;
2015-07-01 16:50:39 -07:00
if (m_standalone && !checkStandaloneID(gameID))
2015-07-09 22:28:08 -07:00
return false;
2016-01-19 19:18:30 -08:00
char region = m_disc->getHeader().m_gameID[3];
2016-03-04 15:04:53 -08:00
const hecl::SystemString* regstr = &regNONE;
2015-07-12 11:07:58 -07:00
switch (region)
{
case 'E':
regstr = &regE;
break;
case 'J':
regstr = &regJ;
break;
case 'P':
regstr = &regP;
break;
}
if (m_standalone)
2015-08-05 14:46:07 -07:00
return checkFromStandaloneDisc(*m_disc, *regstr, info.extractArgs, reps);
2015-07-09 22:28:08 -07:00
else
2015-08-05 14:46:07 -07:00
return checkFromTrilogyDisc(*m_disc, *regstr, info.extractArgs, reps);
2015-07-01 16:50:39 -07:00
}
2015-09-29 23:23:40 -07:00
void SpecBase::doExtract(const ExtractPassInfo& info, FProgress progress)
2015-07-01 16:50:39 -07:00
{
2016-04-09 21:49:02 -07:00
DataSpec::g_curSpec.reset(this);
2015-08-05 14:46:07 -07:00
if (!Blender::BuildMasterShader(m_masterShader))
2016-03-04 15:04:53 -08:00
Log.report(logvisor::Fatal, "Unable to build master shader blend");
if (m_isWii)
{
/* Extract root files for repacking later */
2016-03-04 16:03:41 -08:00
hecl::ProjectPath outDir(m_project.getProjectWorkingPath(), _S("out"));
outDir.makeDirChain(true);
2016-03-04 15:04:53 -08:00
nod::ExtractionContext ctx = {true, info.force, nullptr};
2015-09-28 20:49:31 -07:00
if (!m_standalone)
{
2015-09-02 15:00:40 -07:00
progress(_S("Trilogy Files"), _S(""), 1, 0.0);
2016-03-04 15:04:53 -08:00
nod::Partition* data = m_disc->getDataPartition();
const nod::Node& root = data->getFSTRoot();
for (const nod::Node& child : root)
if (child.getKind() == nod::Node::Kind::File)
child.extractToDirectory(outDir.getAbsolutePath(), ctx);
2015-09-02 15:00:40 -07:00
progress(_S("Trilogy Files"), _S(""), 1, 1.0);
}
}
2015-08-05 14:46:07 -07:00
extractFromDisc(*m_disc, info.force, progress);
2015-07-01 16:50:39 -07:00
}
2016-09-18 16:47:48 -07:00
static bool IsPathAudioGroup(const hecl::ProjectPath& path)
{
if (path.getPathType() != hecl::ProjectPath::Type::Glob ||
!path.getWithExtension(_S(".pool"), true).isFile() ||
!path.getWithExtension(_S(".proj"), true).isFile() ||
!path.getWithExtension(_S(".sdir"), true).isFile() ||
!path.getWithExtension(_S(".samp"), true).isFile())
return false;
return true;
}
static bool IsPathSong(const hecl::ProjectPath& path)
{
if (path.getPathType() != hecl::ProjectPath::Type::Glob ||
!path.getWithExtension(_S(".mid"), true).isFile() ||
!path.getWithExtension(_S(".yaml"), true).isFile())
return false;
return true;
}
bool SpecBase::canCook(const hecl::ProjectPath& path, hecl::BlenderToken& btok)
2015-07-01 16:50:39 -07:00
{
2015-10-03 22:08:56 -07:00
if (!checkPathPrefix(path))
return false;
2016-09-23 11:56:42 -07:00
hecl::ProjectPath asBlend;
if (path.getPathType() == hecl::ProjectPath::Type::Glob)
asBlend = path.getWithExtension(_S(".blend"), true);
else
asBlend = path;
if (hecl::IsPathBlend(asBlend))
2015-09-30 17:40:21 -07:00
{
hecl::BlenderConnection& conn = btok.getBlenderConnection();
2016-09-23 11:56:42 -07:00
if (!conn.openBlend(asBlend))
2015-09-30 17:40:21 -07:00
return false;
2016-03-04 15:04:53 -08:00
if (conn.getBlendType() != hecl::BlenderConnection::BlendType::None)
2015-09-30 17:40:21 -07:00
return true;
}
2016-03-04 15:04:53 -08:00
else if (hecl::IsPathPNG(path))
2015-09-30 17:40:21 -07:00
{
return true;
}
2016-03-04 15:04:53 -08:00
else if (hecl::IsPathYAML(path))
2015-09-30 17:40:21 -07:00
{
2016-08-21 20:47:48 -07:00
athena::io::FileReader reader(path.getAbsolutePath());
bool retval = validateYAMLDNAType(reader);
2015-09-30 17:40:21 -07:00
return retval;
}
2016-09-18 16:47:48 -07:00
else if (IsPathAudioGroup(path))
{
return true;
}
else if (IsPathSong(path))
{
return true;
}
2015-07-22 12:05:18 -07:00
return false;
2015-07-01 16:50:39 -07:00
}
const hecl::Database::DataSpecEntry* SpecBase::overrideDataSpec(const hecl::ProjectPath& path,
const hecl::Database::DataSpecEntry* oldEntry,
2016-10-01 16:20:20 -07:00
hecl::BlenderToken& btok) const
{
if (!checkPathPrefix(path))
return nullptr;
2016-09-23 11:56:42 -07:00
hecl::ProjectPath asBlend;
if (path.getPathType() == hecl::ProjectPath::Type::Glob)
asBlend = path.getWithExtension(_S(".blend"), true);
else
asBlend = path;
if (hecl::IsPathBlend(asBlend))
{
if (hecl::StringUtils::EndsWith(path.getAuxInfo(), _S(".CSKR")) ||
hecl::StringUtils::EndsWith(path.getAuxInfo(), _S(".ANIM")))
2016-08-31 16:31:12 -07:00
return oldEntry;
hecl::BlenderConnection& conn = btok.getBlenderConnection();
2016-09-23 11:56:42 -07:00
if (!conn.openBlend(asBlend))
{
Log.report(logvisor::Error, _S("unable to cook '%s'"),
path.getAbsolutePath().c_str());
return nullptr;
}
hecl::BlenderConnection::BlendType type = conn.getBlendType();
if (type == hecl::BlenderConnection::BlendType::Mesh ||
type == hecl::BlenderConnection::BlendType::Area)
return oldEntry;
}
else if (hecl::IsPathPNG(path))
{
return oldEntry;
}
return &getOriginalSpec();
}
2016-03-04 15:04:53 -08:00
void SpecBase::doCook(const hecl::ProjectPath& path, const hecl::ProjectPath& cookedPath,
bool fast, hecl::BlenderToken& btok, FCookProgress progress)
2015-07-01 16:50:39 -07:00
{
2016-12-24 17:36:42 -08:00
cookedPath.makeDirChain(false);
2016-04-09 21:49:02 -07:00
DataSpec::g_curSpec.reset(this);
2016-09-23 11:56:42 -07:00
hecl::ProjectPath asBlend;
if (path.getPathType() == hecl::ProjectPath::Type::Glob)
asBlend = path.getWithExtension(_S(".blend"), true);
else
asBlend = path;
if (hecl::IsPathBlend(asBlend))
2015-10-03 22:08:56 -07:00
{
hecl::BlenderConnection& conn = btok.getBlenderConnection();
2016-09-23 11:56:42 -07:00
if (!conn.openBlend(asBlend))
2015-10-03 22:08:56 -07:00
return;
2015-10-06 18:17:17 -07:00
switch (conn.getBlendType())
{
2016-03-04 15:04:53 -08:00
case hecl::BlenderConnection::BlendType::Mesh:
2015-10-03 22:08:56 -07:00
{
2016-03-04 15:04:53 -08:00
hecl::BlenderConnection::DataStream ds = conn.beginData();
2016-04-09 21:49:02 -07:00
cookMesh(cookedPath, path, ds, fast, btok, progress);
2015-10-06 18:17:17 -07:00
break;
2015-10-03 22:08:56 -07:00
}
2016-03-04 15:04:53 -08:00
case hecl::BlenderConnection::BlendType::Actor:
2015-10-06 18:17:17 -07:00
{
2016-03-04 15:04:53 -08:00
hecl::BlenderConnection::DataStream ds = conn.beginData();
2016-04-09 21:49:02 -07:00
cookActor(cookedPath, path, ds, fast, btok, progress);
2015-10-06 18:17:17 -07:00
break;
}
2016-03-04 15:04:53 -08:00
case hecl::BlenderConnection::BlendType::Area:
2015-10-06 18:17:17 -07:00
{
2016-03-04 15:04:53 -08:00
hecl::BlenderConnection::DataStream ds = conn.beginData();
2016-04-09 21:49:02 -07:00
cookArea(cookedPath, path, ds, fast, btok, progress);
2015-10-06 18:17:17 -07:00
break;
}
case hecl::BlenderConnection::BlendType::World:
{
hecl::BlenderConnection::DataStream ds = conn.beginData();
cookWorld(cookedPath, path, ds, fast, btok, progress);
break;
}
case hecl::BlenderConnection::BlendType::Frame:
{
hecl::BlenderConnection::DataStream ds = conn.beginData();
cookGuiFrame(cookedPath, path, ds, btok, progress);
break;
}
2017-03-19 22:09:53 -07:00
case hecl::BlenderConnection::BlendType::MapArea:
{
hecl::BlenderConnection::DataStream ds = conn.beginData();
cookMapArea(cookedPath, path, ds, btok, progress);
break;
}
case hecl::BlenderConnection::BlendType::MapUniverse:
{
hecl::BlenderConnection::DataStream ds = conn.beginData();
cookMapUniverse(cookedPath, path, ds, btok, progress);
break;
}
2015-10-06 18:17:17 -07:00
default: break;
}
}
else if (hecl::IsPathPNG(path))
{
if (m_pc)
TXTR::CookPC(path, cookedPath);
else
TXTR::Cook(path, cookedPath);
}
2016-03-04 15:04:53 -08:00
else if (hecl::IsPathYAML(path))
2015-10-06 18:17:17 -07:00
{
2016-08-21 20:47:48 -07:00
athena::io::FileReader reader(path.getAbsolutePath());
cookYAML(cookedPath, path, reader, progress);
2015-10-03 22:08:56 -07:00
}
2016-09-18 16:47:48 -07:00
else if (IsPathAudioGroup(path))
{
cookAudioGroup(cookedPath, path, progress);
}
else if (IsPathSong(path))
{
cookSong(cookedPath, path, progress);
}
2015-07-01 16:50:39 -07:00
}
2016-10-02 15:41:36 -07:00
void SpecBase::flattenDependencies(const hecl::ProjectPath& path,
std::vector<hecl::ProjectPath>& pathsOut,
hecl::BlenderToken& btok)
2015-07-01 16:50:39 -07:00
{
2016-10-02 15:41:36 -07:00
DataSpec::g_curSpec.reset(this);
hecl::ProjectPath asBlend;
if (path.getPathType() == hecl::ProjectPath::Type::Glob)
asBlend = path.getWithExtension(_S(".blend"), true);
else
asBlend = path;
if (hecl::IsPathBlend(asBlend))
{
hecl::BlenderConnection& conn = btok.getBlenderConnection();
if (!conn.openBlend(asBlend))
return;
switch (conn.getBlendType())
{
case hecl::BlenderConnection::BlendType::Mesh:
{
hecl::BlenderConnection::DataStream ds = conn.beginData();
std::vector<hecl::ProjectPath> texs = ds.getTextures();
for (const hecl::ProjectPath& tex : texs)
pathsOut.push_back(tex);
break;
}
case hecl::BlenderConnection::BlendType::Actor:
{
hecl::BlenderConnection::DataStream ds = conn.beginData();
2016-10-07 20:41:08 -07:00
hecl::BlenderConnection::DataStream::Actor actor = ds.compileActorCharacterOnly();
2016-10-02 15:41:36 -07:00
for (auto& sub : actor.subtypes)
{
if (sub.armature >= 0)
{
pathsOut.push_back(sub.mesh);
hecl::SystemStringView chSysName(sub.name);
pathsOut.push_back(path.ensureAuxInfo(chSysName.sys_str() + _S(".CSKR")));
const auto& arm = actor.armatures[sub.armature];
hecl::SystemStringView armSysName(arm.name);
pathsOut.push_back(path.ensureAuxInfo(armSysName.sys_str() + _S(".CINF")));
if (sub.overlayMeshes.size())
{
pathsOut.push_back(sub.overlayMeshes[0].second);
pathsOut.push_back(path.ensureAuxInfo(chSysName.sys_str() + _S(".over.CSKR")));
}
}
}
break;
}
case hecl::BlenderConnection::BlendType::Area:
{
hecl::BlenderConnection::DataStream ds = conn.beginData();
std::vector<hecl::ProjectPath> texs = ds.getTextures();
for (const hecl::ProjectPath& tex : texs)
pathsOut.push_back(tex);
break;
}
default: break;
}
}
else if (hecl::IsPathYAML(path))
{
athena::io::FileReader reader(path.getAbsolutePath());
flattenDependenciesYAML(reader, pathsOut);
}
pathsOut.push_back(path);
}
void SpecBase::flattenDependencies(const UniqueID32& id, std::vector<hecl::ProjectPath>& pathsOut)
{
hecl::ProjectPath path = UniqueIDBridge::TranslatePakIdToPath(id);
if (path)
2016-12-24 17:36:42 -08:00
flattenDependencies(path, pathsOut, *g_ThreadBlenderToken.get());
2016-10-02 15:41:36 -07:00
}
void SpecBase::flattenDependencies(const UniqueID64& id, std::vector<hecl::ProjectPath>& pathsOut)
{
hecl::ProjectPath path = UniqueIDBridge::TranslatePakIdToPath(id);
if (path)
2016-12-24 17:36:42 -08:00
flattenDependencies(path, pathsOut, *g_ThreadBlenderToken.get());
2015-07-01 16:50:39 -07:00
}
2016-10-02 15:41:36 -07:00
bool SpecBase::canPackage(const PackagePassInfo& info)
2015-07-01 16:50:39 -07:00
{
2016-10-02 15:41:36 -07:00
return false;
2015-07-01 16:50:39 -07:00
}
2015-08-05 14:46:07 -07:00
void SpecBase::doPackage(const PackagePassInfo& info)
2015-07-01 16:50:39 -07:00
{
}
}