#include "dynos.cpp.h" extern void ModelTextureAdd(char* id, char* data, int w, int h); // // Pointers // static void *GetPointerFromData(GfxData *aGfxData, const String &aPtrName, u32 aPtrData) { // Lights for (auto& _Node : aGfxData->mLights) { if (_Node->mName == aPtrName) { if (aPtrData == 1) { return (void *) &_Node->mData->l[0]; } if (aPtrData == 2) { return (void *) &_Node->mData->a; } sys_fatal("Unknown Light type: %u", aPtrData); } } // Textures for (auto& _Node : aGfxData->mTextures) { if (_Node->mName == aPtrName) { return (void *) _Node; } } // Display lists for (auto &_Node : aGfxData->mDisplayLists) { if (_Node->mName == aPtrName) { return (void *) _Node->mData; } } // Geo layouts for (auto &_Node : aGfxData->mGeoLayouts) { if (_Node->mName == aPtrName) { return (void *) _Node->mData; } } // Vertices for (auto &_Node : aGfxData->mVertices) { if (_Node->mName == aPtrName) { return (void *) (_Node->mData + aPtrData); } } // Error sys_fatal("Pointer not found: %s", aPtrName.begin()); return NULL; } static void *ReadPointer(FILE *aFile, GfxData *aGfxData, u32 aValue) { // FUNC if (aValue == FUNCTION_CODE) { s32 _GeoFunctionIndex = ReadBytes(aFile); return DynOS_Geo_GetFunctionPointerFromIndex(_GeoFunctionIndex); } // PNTR if (aValue == POINTER_CODE) { String _PtrName; _PtrName.Read(aFile); u32 _PtrData = ReadBytes(aFile); return GetPointerFromData(aGfxData, _PtrName, _PtrData); } // Not a pointer return NULL; } // // Read binary // static void LoadLightData(FILE *aFile, GfxData *aGfxData) { DataNode *_Node = New>(); // Name _Node->mName.Read(aFile); // Data _Node->mData = New(); *_Node->mData = ReadBytes(aFile); // Append aGfxData->mLights.Add(_Node); } static void LoadTextureData(FILE *aFile, GfxData *aGfxData) { DataNode *_Node = New>(); // Name _Node->mName.Read(aFile); // Data _Node->mData = New(); _Node->mData->mUploaded = false; _Node->mData->mPngData.Read(aFile); if (!_Node->mData->mPngData.Empty()) { u8 *_RawData = pngutils_read_png_from_memory(_Node->mData->mPngData.begin(), _Node->mData->mPngData.Count(), &_Node->mData->mRawWidth, &_Node->mData->mRawHeight, NULL, 4); _Node->mData->mRawFormat = G_IM_FMT_RGBA; _Node->mData->mRawSize = G_IM_SIZ_32b; _Node->mData->mRawData = Array(_RawData, _RawData + (_Node->mData->mRawWidth * _Node->mData->mRawHeight * 4)); ModelTextureAdd((std::string("gfx/") + _Node->mName.begin()).data(), (char*)_RawData, _Node->mData->mRawWidth, _Node->mData->mRawHeight); free(_RawData); } else { // Probably a palette _Node->mData->mRawData = Array(); _Node->mData->mRawWidth = 0; _Node->mData->mRawHeight = 0; _Node->mData->mRawFormat = 0; _Node->mData->mRawSize = 0; } // Append aGfxData->mTextures.Add(_Node); } static void LoadVertexData(FILE *aFile, GfxData *aGfxData) { DataNode *_Node = New>(); // Name _Node->mName.Read(aFile); // Data _Node->mSize = ReadBytes(aFile); _Node->mData = New(_Node->mSize); for (u32 i = 0; i != _Node->mSize; ++i) { _Node->mData[i].n.ob[0] = ReadBytes(aFile); _Node->mData[i].n.ob[1] = ReadBytes(aFile); _Node->mData[i].n.ob[2] = ReadBytes(aFile); _Node->mData[i].n.flag = ReadBytes(aFile); _Node->mData[i].n.tc[0] = ReadBytes(aFile); _Node->mData[i].n.tc[1] = ReadBytes(aFile); _Node->mData[i].n.n[0] = ReadBytes (aFile); _Node->mData[i].n.n[1] = ReadBytes (aFile); _Node->mData[i].n.n[2] = ReadBytes (aFile); _Node->mData[i].n.a = ReadBytes(aFile); } // Append aGfxData->mVertices.Add(_Node); } static void LoadDisplayListData(FILE *aFile, GfxData *aGfxData) { DataNode *_Node = New>(); // Name _Node->mName.Read(aFile); // Data _Node->mSize = ReadBytes(aFile); _Node->mData = New(_Node->mSize); for (u32 i = 0; i != _Node->mSize; ++i) { u32 _WordsW0 = ReadBytes(aFile); u32 _WordsW1 = ReadBytes(aFile); void *_Ptr = ReadPointer(aFile, aGfxData, _WordsW1); if (_Ptr) { _Node->mData[i].words.w0 = (uintptr_t) _WordsW0; _Node->mData[i].words.w1 = (uintptr_t) _Ptr; } else { _Node->mData[i].words.w0 = (uintptr_t) _WordsW0; _Node->mData[i].words.w1 = (uintptr_t) _WordsW1; } } // Append aGfxData->mDisplayLists.Add(_Node); } static void LoadGeoLayoutData(FILE *aFile, GfxData *aGfxData) { DataNode *_Node = New>(); // Name _Node->mName.Read(aFile); // Data _Node->mSize = ReadBytes(aFile); _Node->mData = New(_Node->mSize); for (u32 i = 0; i != _Node->mSize; ++i) { u32 _Value = ReadBytes(aFile); void *_Ptr = ReadPointer(aFile, aGfxData, _Value); if (_Ptr) { _Node->mData[i] = (uintptr_t) _Ptr; } else { _Node->mData[i] = (uintptr_t) _Value; } } // Append aGfxData->mGeoLayouts.Add(_Node); } // For retro-compatibility static void LoadGfxDynCmd(FILE *aFile, GfxData *aGfxData) { Gfx *_Data = NULL; String _DisplayListName; _DisplayListName.Read(aFile); for (auto& _DisplayList : aGfxData->mDisplayLists) { if (_DisplayList->mName == _DisplayListName) { _Data = _DisplayList->mData; break; } } if (!_Data) { sys_fatal("Display list not found: %s", _DisplayListName.begin()); } ReadBytes(aFile); ReadBytes(aFile); } static void LoadAnimationData(FILE *aFile, GfxData *aGfxData) { DataNode *_Node = New>(); // Name _Node->mName.Read(aFile); // Data _Node->mData = New(); _Node->mData->mFlags = ReadBytes(aFile); _Node->mData->mUnk02 = ReadBytes(aFile); _Node->mData->mUnk04 = ReadBytes(aFile); _Node->mData->mUnk06 = ReadBytes(aFile); _Node->mData->mUnk08 = ReadBytes(aFile); _Node->mData->mUnk0A.second = ReadBytes(aFile); _Node->mData->mLength = ReadBytes(aFile); _Node->mData->mValues.second.Read(aFile); _Node->mData->mIndex.second.Read(aFile); // Append aGfxData->mAnimations.Add(_Node); } static void LoadAnimationTable(FILE *aFile, GfxData *aGfxData) { void *_AnimationPtr = NULL; // Data String _AnimationName; _AnimationName.Read(aFile); if (_AnimationName != "NULL") { for (auto &_AnimData : aGfxData->mAnimations) { if (_AnimData->mName == _AnimationName) { _AnimationPtr = (void *) _AnimData->mData; break; } } if (!_AnimationPtr) { sys_fatal("Animation not found: %s", _AnimationName.begin()); } } // Append aGfxData->mAnimationTable.Add({ "", _AnimationPtr }); } // // Load from binary // GfxData *DynOS_Gfx_LoadFromBinary(const SysPath &aPackFolder, const char *aActorName) { struct DynosGfxDataCache { SysPath mPackFolder; Array> mGfxData; }; static Array sDynosGfxDataCache; // Look for pack in cache DynosGfxDataCache *_Pack = NULL; for (s32 i = 0; i != sDynosGfxDataCache.Count(); ++i) { if (sDynosGfxDataCache[i]->mPackFolder == aPackFolder) { _Pack = sDynosGfxDataCache[i]; break; } } // Look for actor in pack if (_Pack) { for (s32 i = 0; i != _Pack->mGfxData.Count(); ++i) { if (_Pack->mGfxData[i].first == aActorName) { // Perfectly valid, aActorName comes from static RO data, so its address never changes during execution return _Pack->mGfxData[i].second; } } } // Load data from binary file GfxData *_GfxData = NULL; SysPath _Filename = fstring("%s/%s.bin", aPackFolder.begin(), aActorName); FILE *_File = fopen(_Filename.c_str(), "rb"); if (_File) { _GfxData = New(); for (bool _Done = false; !_Done;) { switch (ReadBytes(_File)) { case DATA_TYPE_LIGHT: LoadLightData (_File, _GfxData); break; case DATA_TYPE_TEXTURE: LoadTextureData (_File, _GfxData); break; case DATA_TYPE_VERTEX: LoadVertexData (_File, _GfxData); break; case DATA_TYPE_DISPLAY_LIST: LoadDisplayListData(_File, _GfxData); break; case DATA_TYPE_GEO_LAYOUT: LoadGeoLayoutData (_File, _GfxData); break; case DATA_TYPE_ANIMATION: LoadAnimationData (_File, _GfxData); break; case DATA_TYPE_ANIMATION_TABLE: LoadAnimationTable (_File, _GfxData); break; case DATA_TYPE_GFXDYNCMD: LoadGfxDynCmd (_File, _GfxData); break; default: _Done = true; break; } } fclose(_File); } // Add data to cache, even if not loaded if (_Pack) { _Pack->mGfxData.Add({ aActorName, _GfxData }); } else { _Pack = New(); _Pack->mPackFolder = aPackFolder; _Pack->mGfxData.Add({ aActorName, _GfxData }); sDynosGfxDataCache.Add(_Pack); } return _GfxData; }