mirror of
https://github.com/AxioDL/metaforce.git
synced 2025-10-24 15:30:23 +00:00
301 lines
9.9 KiB
C++
301 lines
9.9 KiB
C++
#include <athena/FileWriter.hpp>
|
|
#include <lzokay.hpp>
|
|
#include "MREA.hpp"
|
|
#include "../DNAMP1/MREA.hpp"
|
|
#include "DataSpec/DNACommon/EGMC.hpp"
|
|
#include "DeafBabe.hpp"
|
|
#include "hecl/Blender/Connection.hpp"
|
|
|
|
namespace DataSpec {
|
|
extern hecl::Database::DataSpecEntry SpecEntMP2ORIG;
|
|
|
|
namespace DNAMP2 {
|
|
|
|
void MREA::StreamReader::nextBlock() {
|
|
if (m_nextBlk >= m_blkCount)
|
|
Log.report(logvisor::Fatal, fmt("MREA stream overrun"));
|
|
|
|
BlockInfo& info = m_blockInfos[m_nextBlk++];
|
|
|
|
/* Reallocate read buffer if needed */
|
|
if (info.bufSize > m_compBufSz) {
|
|
m_compBufSz = info.bufSize;
|
|
m_compBuf.reset(new atUint8[m_compBufSz]);
|
|
}
|
|
|
|
/* Reallocate decompress buffer if needed */
|
|
if (info.decompSize > m_decompBufSz) {
|
|
m_decompBufSz = info.decompSize;
|
|
m_decompBuf.reset(new atUint8[m_decompBufSz]);
|
|
}
|
|
|
|
if (info.compSize == 0) {
|
|
/* Read uncompressed block */
|
|
m_source.readUBytesToBuf(m_decompBuf.get(), info.decompSize);
|
|
} else {
|
|
/* Read compressed segments */
|
|
atUint32 blockStart = ROUND_UP_32(info.compSize) - info.compSize;
|
|
m_source.seek(blockStart);
|
|
atUint32 rem = info.decompSize;
|
|
atUint8* bufCur = m_decompBuf.get();
|
|
while (rem) {
|
|
atInt16 chunkSz = m_source.readInt16Big();
|
|
if (chunkSz < 0) {
|
|
chunkSz = -chunkSz;
|
|
m_source.readUBytesToBuf(bufCur, chunkSz);
|
|
bufCur += chunkSz;
|
|
rem -= chunkSz;
|
|
} else {
|
|
m_source.readUBytesToBuf(m_compBuf.get(), chunkSz);
|
|
size_t dsz;
|
|
lzokay::decompress(m_compBuf.get(), chunkSz, bufCur, rem, dsz);
|
|
bufCur += dsz;
|
|
rem -= dsz;
|
|
}
|
|
}
|
|
}
|
|
|
|
m_posInBlk = 0;
|
|
m_blkSz = info.decompSize;
|
|
}
|
|
|
|
MREA::StreamReader::StreamReader(athena::io::IStreamReader& source, atUint32 blkCount)
|
|
: m_compBufSz(0x4120)
|
|
, m_compBuf(new atUint8[0x4120])
|
|
, m_decompBufSz(0x4120)
|
|
, m_decompBuf(new atUint8[0x4120])
|
|
, m_source(source)
|
|
, m_blkCount(blkCount) {
|
|
m_blockInfos.reserve(blkCount);
|
|
for (atUint32 i = 0; i < blkCount; ++i) {
|
|
m_blockInfos.emplace_back();
|
|
BlockInfo& info = m_blockInfos.back();
|
|
info.read(source);
|
|
m_totalDecompLen += info.decompSize;
|
|
}
|
|
source.seekAlign32();
|
|
m_blkBase = source.position();
|
|
nextBlock();
|
|
}
|
|
|
|
void MREA::StreamReader::seek(atInt64 diff, athena::SeekOrigin whence) {
|
|
atUint64 target = diff;
|
|
if (whence == athena::SeekOrigin::Current) {
|
|
target = m_pos + diff;
|
|
} else if (whence == athena::SeekOrigin::End) {
|
|
target = m_totalDecompLen - diff;
|
|
}
|
|
|
|
if (target >= m_totalDecompLen)
|
|
Log.report(logvisor::Fatal, fmt("MREA stream seek overrun"));
|
|
|
|
/* Determine which block contains position */
|
|
atUint32 dAccum = 0;
|
|
atUint32 cAccum = 0;
|
|
atUint32 bIdx = 0;
|
|
for (BlockInfo& info : m_blockInfos) {
|
|
atUint32 newAccum = dAccum + info.decompSize;
|
|
if (newAccum > target)
|
|
break;
|
|
dAccum = newAccum;
|
|
++bIdx;
|
|
if (info.compSize)
|
|
cAccum += ROUND_UP_32(info.compSize);
|
|
else
|
|
cAccum += info.decompSize;
|
|
}
|
|
|
|
/* Seek source if needed */
|
|
if (bIdx != m_nextBlk - 1) {
|
|
m_source.seek(m_blkBase + cAccum, athena::SeekOrigin::Begin);
|
|
m_nextBlk = bIdx;
|
|
nextBlock();
|
|
}
|
|
|
|
m_pos = target;
|
|
m_posInBlk = target - dAccum;
|
|
}
|
|
|
|
atUint64 MREA::StreamReader::readUBytesToBuf(void* buf, atUint64 len) {
|
|
atUint8* bufCur = reinterpret_cast<atUint8*>(buf);
|
|
atUint64 rem = len;
|
|
while (rem) {
|
|
atUint64 lRem = rem;
|
|
atUint64 blkRem = m_blkSz - m_posInBlk;
|
|
if (lRem > blkRem)
|
|
lRem = blkRem;
|
|
memcpy(bufCur, &m_decompBuf[m_posInBlk], lRem);
|
|
bufCur += lRem;
|
|
rem -= lRem;
|
|
m_posInBlk += lRem;
|
|
m_pos += lRem;
|
|
if (rem)
|
|
nextBlock();
|
|
}
|
|
return len;
|
|
}
|
|
|
|
void MREA::StreamReader::writeDecompInfos(athena::io::IStreamWriter& writer) const {
|
|
for (const BlockInfo& info : m_blockInfos) {
|
|
BlockInfo modInfo = info;
|
|
modInfo.compSize = 0;
|
|
modInfo.write(writer);
|
|
}
|
|
}
|
|
|
|
bool MREA::Extract(const SpecBase& dataSpec, PAKEntryReadStream& rs, const hecl::ProjectPath& outPath,
|
|
PAKRouter<PAKBridge>& pakRouter, const DNAMP2::PAK::Entry& entry, bool force,
|
|
hecl::blender::Token& btok, std::function<void(const hecl::SystemChar*)>) {
|
|
using RigPair = std::pair<std::pair<UniqueID32, CSKR*>, std::pair<UniqueID32, CINF*>>;
|
|
RigPair dummy = {};
|
|
|
|
if (!force && outPath.isFile())
|
|
return true;
|
|
|
|
/* Do extract */
|
|
Header head;
|
|
head.read(rs);
|
|
rs.seekAlign32();
|
|
|
|
/* MREA decompression stream */
|
|
StreamReader drs(rs, head.compressedBlockCount);
|
|
hecl::ProjectPath decompPath = outPath.getCookedPath(SpecEntMP2ORIG).getWithExtension(_SYS_STR(".decomp"));
|
|
decompPath.makeDirChain(false);
|
|
athena::io::FileWriter mreaDecompOut(decompPath.getAbsolutePath());
|
|
head.write(mreaDecompOut);
|
|
mreaDecompOut.seekAlign32();
|
|
drs.writeDecompInfos(mreaDecompOut);
|
|
mreaDecompOut.seekAlign32();
|
|
atUint64 decompLen = drs.length();
|
|
mreaDecompOut.writeBytes(drs.readBytes(decompLen).get(), decompLen);
|
|
mreaDecompOut.close();
|
|
drs.seek(0, athena::SeekOrigin::Begin);
|
|
|
|
/* Start up blender connection */
|
|
hecl::blender::Connection& conn = btok.getBlenderConnection();
|
|
if (!conn.createBlend(outPath, hecl::blender::BlendType::Area))
|
|
return false;
|
|
|
|
/* Calculate offset to EGMC section */
|
|
atUint64 egmcOffset = 0;
|
|
for (unsigned i = 0; i < head.egmcSecIdx; i++)
|
|
egmcOffset += head.secSizes[i];
|
|
|
|
/* Load EGMC if possible so we can assign meshes to scanIds */
|
|
drs.seek(egmcOffset, athena::SeekOrigin::Begin);
|
|
UniqueID32 egmcId(drs);
|
|
DNACommon::EGMC egmc;
|
|
pakRouter.lookupAndReadDNA(egmcId, egmc);
|
|
|
|
drs.seek(0, athena::SeekOrigin::Begin);
|
|
|
|
/* Open Py Stream and read sections */
|
|
hecl::blender::PyOutStream os = conn.beginPythonOut(true);
|
|
os.format(fmt(
|
|
"import bpy\n"
|
|
"import bmesh\n"
|
|
"from mathutils import Vector\n"
|
|
"\n"
|
|
"bpy.context.scene.name = '{}'\n"),
|
|
pakRouter.getBestEntryName(entry, false));
|
|
DNACMDL::InitGeomBlenderContext(os, dataSpec.getMasterShaderPath());
|
|
MaterialSet::RegisterMaterialProps(os);
|
|
os << "# Clear Scene\n"
|
|
"if len(bpy.data.collections):\n"
|
|
" bpy.data.collections.remove(bpy.data.collections[0])\n"
|
|
"\n"
|
|
"bpy.types.Light.retro_layer = bpy.props.IntProperty(name='Retro: Light Layer')\n"
|
|
"bpy.types.Light.retro_origtype = bpy.props.IntProperty(name='Retro: Original Type')\n"
|
|
"bpy.types.Object.retro_disable_enviro_visor = bpy.props.BoolProperty(name='Retro: Disable in Combat/Scan "
|
|
"Visor')\n"
|
|
"bpy.types.Object.retro_disable_thermal_visor = bpy.props.BoolProperty(name='Retro: Disable in Thermal "
|
|
"Visor')\n"
|
|
"bpy.types.Object.retro_disable_xray_visor = bpy.props.BoolProperty(name='Retro: Disable in X-Ray Visor')\n"
|
|
"bpy.types.Object.retro_thermal_level = bpy.props.EnumProperty(items=[('COOL', 'Cool', 'Cool Temperature'),"
|
|
"('HOT', 'Hot', 'Hot Temperature'),"
|
|
"('WARM', 'Warm', 'Warm Temperature')],"
|
|
"name='Retro: Thermal Visor Level')\n"
|
|
"\n";
|
|
|
|
/* One shared material set for all meshes */
|
|
os << "# Materials\n"
|
|
"materials = []\n"
|
|
"\n";
|
|
MaterialSet matSet;
|
|
atUint64 secStart = drs.position();
|
|
matSet.read(drs);
|
|
matSet.readToBlender(os, pakRouter, entry, 0);
|
|
drs.seek(secStart + head.secSizes[0], athena::SeekOrigin::Begin);
|
|
std::vector<DNACMDL::VertexAttributes> vertAttribs;
|
|
DNACMDL::GetVertexAttributes(matSet, vertAttribs);
|
|
|
|
/* Read meshes */
|
|
atUint32 curSec = 1;
|
|
for (atUint32 m = 0; m < head.meshCount; ++m) {
|
|
MeshHeader mHeader;
|
|
secStart = drs.position();
|
|
mHeader.read(drs);
|
|
drs.seek(secStart + head.secSizes[curSec++], athena::SeekOrigin::Begin);
|
|
curSec += DNACMDL::ReadGeomSectionsToBlender<PAKRouter<PAKBridge>, MaterialSet, RigPair, DNACMDL::SurfaceHeader_2>(
|
|
os, drs, pakRouter, entry, dummy, true, true, vertAttribs, m, head.secCount, 0, &head.secSizes[curSec]);
|
|
os.format(fmt(
|
|
"obj.retro_disable_enviro_visor = {}\n"
|
|
"obj.retro_disable_thermal_visor = {}\n"
|
|
"obj.retro_disable_xray_visor = {}\n"
|
|
"obj.retro_thermal_level = '{}'\n"),
|
|
mHeader.visorFlags.disableEnviro() ? "True" : "False", mHeader.visorFlags.disableThermal() ? "True" : "False",
|
|
mHeader.visorFlags.disableXray() ? "True" : "False", mHeader.visorFlags.thermalLevelStr());
|
|
|
|
/* Seek through AROT-relation sections */
|
|
drs.seek(head.secSizes[curSec++], athena::SeekOrigin::Current);
|
|
drs.seek(head.secSizes[curSec++], athena::SeekOrigin::Current);
|
|
}
|
|
|
|
/* Skip AROT */
|
|
drs.seek(head.secSizes[curSec++], athena::SeekOrigin::Current);
|
|
|
|
/* Skip BVH */
|
|
drs.seek(head.secSizes[curSec++], athena::SeekOrigin::Current);
|
|
|
|
/* Skip Bitmap */
|
|
drs.seek(head.secSizes[curSec++], athena::SeekOrigin::Current);
|
|
|
|
/* Skip SCLY (for now) */
|
|
for (atUint32 l = 0; l < head.sclyLayerCount; ++l) {
|
|
drs.seek(head.secSizes[curSec++], athena::SeekOrigin::Current);
|
|
}
|
|
|
|
/* Skip SCGN (for now) */
|
|
drs.seek(head.secSizes[curSec++], athena::SeekOrigin::Current);
|
|
|
|
/* Read collision meshes */
|
|
DeafBabe collision;
|
|
secStart = drs.position();
|
|
collision.read(drs);
|
|
DeafBabe::BlenderInit(os);
|
|
collision.sendToBlender(os);
|
|
drs.seek(secStart + head.secSizes[curSec++], athena::SeekOrigin::Begin);
|
|
|
|
/* Skip unknown section */
|
|
drs.seek(head.secSizes[curSec++], athena::SeekOrigin::Current);
|
|
|
|
/* Read BABEDEAD Lights as Cycles emissives */
|
|
secStart = drs.position();
|
|
DNAMP1::MREA::ReadBabeDeadToBlender_1_2(os, drs);
|
|
drs.seek(secStart + head.secSizes[curSec++], athena::SeekOrigin::Begin);
|
|
|
|
/* Origins to center of mass */
|
|
os << "bpy.context.view_layer.layer_collection.children['Collision'].hide_viewport = False\n"
|
|
"bpy.ops.object.select_by_type(type='MESH')\n"
|
|
"bpy.ops.object.origin_set(type='ORIGIN_CENTER_OF_MASS')\n"
|
|
"bpy.ops.object.select_all(action='DESELECT')\n"
|
|
"bpy.context.view_layer.layer_collection.children['Collision'].hide_viewport = True\n";
|
|
|
|
os.centerView();
|
|
os.close();
|
|
return conn.saveBlend();
|
|
}
|
|
|
|
} // namespace DNAMP2
|
|
} // namespace DataSpec
|