[Merge] mapimport into develop

This commit is contained in:
momo5502 2016-12-26 02:30:48 +01:00
commit f7a81f7af0
6 changed files with 332 additions and 131 deletions

View File

@ -1,5 +1,7 @@
#include <STDInclude.hpp> #include <STDInclude.hpp>
#define IW4X_IMG_VERSION "0"
namespace Assets namespace Assets
{ {
void IGfxImage::load(Game::XAssetHeader* header, std::string name, Components::ZoneBuilder::Zone* builder) void IGfxImage::load(Game::XAssetHeader* header, std::string name, Components::ZoneBuilder::Zone* builder)
@ -19,13 +21,31 @@ namespace Assets
image->category = 0; image->category = 0;
image->cardMemory = 0; image->cardMemory = 0;
image->loadDef = builder->getAllocator()->allocate<Game::GfxImageLoadDef>(); const char* tempName = image->name;
if (!image->loadDef) if (tempName[0] == '*') tempName++;
Components::FileSystem::File imageFile(fmt::sprintf("images/%s.iw4xImage", tempName));
if (imageFile.exists())
{ {
Components::Logger::Error("Failed to allocate GfxImageLoadDef structure!"); Utils::Stream::Reader reader(builder->getAllocator(), imageFile.getBuffer());
return;
if (reader.read<__int64>() != *reinterpret_cast<__int64*>("IW4xImg" IW4X_IMG_VERSION))
{
Components::Logger::Error(0, "Reading image '%s' failed, header is invalid!", name.data());
} }
AssertSize(Game::MapType, 1);
image->mapType = reader.read<Game::MapType>();
image->semantic = reader.read<char>();
image->category = reader.read<char>();
image->dataLen1 = reader.read<int>();
image->dataLen2 = image->dataLen1;
image->loadDef = reinterpret_cast<Game::GfxImageLoadDef*>(reader.readArray<char>(image->dataLen1 + 16));
}
else
{
char nameBuffer[MAX_PATH] = { 0 }; char nameBuffer[MAX_PATH] = { 0 };
Components::Materials::FormatImagePath(nameBuffer, sizeof(nameBuffer), 0, 0, name.data()); Components::Materials::FormatImagePath(nameBuffer, sizeof(nameBuffer), 0, 0, name.data());
Components::FileSystem::File iwi(nameBuffer); Components::FileSystem::File iwi(nameBuffer);
@ -44,7 +64,7 @@ namespace Assets
image->dataLen1 = iwiHeader->fileSizeForPicmip[0] - 32; image->dataLen1 = iwiHeader->fileSizeForPicmip[0] - 32;
image->dataLen2 = iwiHeader->fileSizeForPicmip[0] - 32; image->dataLen2 = iwiHeader->fileSizeForPicmip[0] - 32;
if (std::memcmp(iwiHeader->tag, "IWi", 3)) if (std::memcmp(iwiHeader->tag, "IWi", 3) && iwiHeader->version == 8)
{ {
Components::Logger::Error("Image is not a valid IWi!"); Components::Logger::Error("Image is not a valid IWi!");
return; return;
@ -86,6 +106,7 @@ namespace Assets
break; break;
} }
} }
}
header->image = image; header->image = image;
} }

View File

@ -1,51 +1,185 @@
#include <STDInclude.hpp> #include <STDInclude.hpp>
#define IW4X_GFXMAP_VERSION 1
namespace Assets namespace Assets
{ {
void IGfxWorld::load(Game::XAssetHeader* /*header*/, std::string name, Components::ZoneBuilder::Zone* /*builder*/) void IGfxWorld::load(Game::XAssetHeader* header, std::string name, Components::ZoneBuilder::Zone* builder)
{ {
Game::GfxWorld* map = Game::DB_FindXAssetHeader(Game::XAssetType::ASSET_TYPE_GFX_MAP, name.data()).gfxWorld; if (name != "maps/iw4_credits.d3dbsp") return;
if (map) return; Game::GfxWorld* map = Game::DB_FindXAssetHeader(Game::XAssetType::ASSET_TYPE_GFX_MAP, "maps/iw4_credits.d3dbsp").gfxWorld;
if (!map) return;
header->gfxWorld = map;
Components::Logger::Error("Missing GfxMap %s... you can't make them yet you idiot.", name.data()); map->name = "maps/mp/mp_toujane.d3dbsp";
map->baseName = "mp_toujane";
Game::Material* basemat = /*Game::DB_FindXAssetHeader(Game::XAssetType::ASSET_TYPE_MATERIAL, "white").material;*/map->dpvs.surfaces[0].material;
if (!basemat) return;
for (unsigned int i = 0; i < map->dpvs.staticSurfaceCount; ++i)
{
if (map->dpvs.surfaces[i].material)
{
map->dpvs.surfaces[i].material = basemat;
}
}
//return;
Components::FileSystem::File mapFile(fmt::sprintf("gfxworld/%s.iw4xGfxWorld", /*map->baseName*/"mp_waw_toujane_night"));
if (mapFile.exists())
{
Utils::Stream::Reader reader(builder->getAllocator(), mapFile.getBuffer());
if (reader.read<__int64>() != *reinterpret_cast<__int64*>("IW4xGfxW"))
{
Components::Logger::Error(0, "Reading gfxworld '%s' failed, header is invalid!", name.data());
}
int version = reader.read<int>();
if (version != IW4X_GFXMAP_VERSION)
{
Components::Logger::Error(0, "Reading gfxworld '%s' failed, expected version is %d, but it was %d!", name.data(), IW4X_GFXMAP_VERSION, version);
}
std::string _name = reader.readString(); // Name
std::string bname = reader.readString(); // Basename
map->nodeCount = reader.read<int>();
map->planeCount = reader.read<int>();
map->bounds.midPoint[0] = reader.read<float>();
map->bounds.midPoint[1] = reader.read<float>();
map->bounds.midPoint[2] = reader.read<float>();
map->bounds.halfSize[0] = reader.read<float>();
map->bounds.halfSize[1] = reader.read<float>();
map->bounds.halfSize[2] = reader.read<float>();
map->dpvsPlanes = reader.read<Game::GfxWorldDpvsPlanes>();
map->dpvsPlanes.planes = reader.readArray<Game::cplane_t>(map->planeCount);
map->dpvsPlanes.nodes = reader.readArray<unsigned short>(map->nodeCount);
map->dpvsPlanes.sceneEntCellBits = reinterpret_cast<unsigned int*>(reader.readArray<char>(map->dpvsPlanes.cellCount << 11));
map->aabbTreeCounts = reader.readArray<Game::GfxCellTreeCount>(map->dpvsPlanes.cellCount);
//map->aabbTrees = reader.readArray<Game::GfxCellTree>(map->dpvsPlanes.cellCount);
map->aabbTrees = builder->getAllocator()->allocateArray<Game::GfxCellTree>(map->dpvsPlanes.cellCount);
for (int i = 0; i < map->dpvsPlanes.cellCount; ++i)
{
map->aabbTrees[i].aabbTree = reader.readArray<Game::GfxAabbTree>(map->aabbTreeCounts[i].aabbTreeCount);
}
map->cells = reader.readArray<Game::GfxCell>(map->dpvsPlanes.cellCount);
for (int i = 0; i < map->dpvsPlanes.cellCount; ++i)
{
Game::GfxCell* cell = &map->cells[i];
cell->portals = reader.readArray<Game::GfxPortal>(cell->portalCount);
for (int j = 0; j < cell->portalCount; ++j)
{
if (cell->portals[j].vertices)
{
cell->portals[j].vertices = reader.readArray<Game::vec3_t>(cell->portals[j].vertexCount & 0xFF);
}
}
}
//return;
Game::GfxWorldVertex* originalVerts = map->draw.vd.vertices;
map->draw.indexCount = reader.read<int>();
map->draw.indices = reader.readArray<unsigned short>(map->draw.indexCount);
map->draw.vertexCount = reader.read<unsigned int>();
map->draw.vd.vertices = reader.readArray<Game::GfxWorldVertex>(map->draw.vertexCount);
map->draw.vertexLayerDataSize = reader.read<unsigned int>();
map->draw.vld.data = reader.readArray<char>(map->draw.vertexLayerDataSize);
for (unsigned int i = 0; i < map->draw.vertexCount; ++i)
{
map->draw.vd.vertices[i].lmapCoord[0] = originalVerts[i % 3].lmapCoord[0];
map->draw.vd.vertices[i].lmapCoord[1] = originalVerts[i % 3].lmapCoord[1];
map->draw.vd.vertices[i].texCoord[0] = originalVerts[i % 3].texCoord[0];
map->draw.vd.vertices[i].texCoord[1] = originalVerts[i % 3].texCoord[1];
map->draw.vd.vertices[i].texCoord[2] = originalVerts[i % 3].texCoord[2];
}
for (int i = 0; i < 8; ++i)
{
(&map->dpvs.staticSurfaceCount)[i] = reader.read<unsigned int>();
}
map->surfaceCount = map->dpvs.staticSurfaceCount;
map->dpvs.sortedSurfIndex = reader.readArray<unsigned short>(map->dpvs.staticSurfaceCount + map->dpvs.staticSurfaceCountNoDecal);
//map->dpvs.surfacesBounds = reader.readArray<Game::GfxSurfaceBounds>(map->dpvs.staticSurfaceCount);
map->dpvs.surfaces = builder->getAllocator()->allocateArray<Game::GfxSurface>(map->dpvs.staticSurfaceCount);
for (unsigned int i = 0; i < map->dpvs.staticSurfaceCount; ++i)
{
//Game::GfxSurface* surf = reader.readArray<Game::GfxSurface>(1);
//std::memcpy(&map->dpvs.surfaces[i], surf, sizeof(Game::GfxSurface));
map->dpvs.surfaces[i] = reader.read<Game::GfxSurface>();
if (map->dpvs.surfaces[i].material)
{
std::string matname = reader.readString(); // Name
map->dpvs.surfaces[i].material = basemat;
}
}
}
map->draw.reflectionImages = 0;
map->draw.reflectionProbeCount = 0;
map->draw.reflectionProbes = 0;
map->draw.reflectionProbeTextures = 0;
//Components::Logger::Error("Missing GfxMap %s... you can't make them yet you idiot.", name.data());
} }
void IGfxWorld::mark(Game::XAssetHeader header, Components::ZoneBuilder::Zone* builder) void IGfxWorld::mark(Game::XAssetHeader header, Components::ZoneBuilder::Zone* builder)
{ {
Game::GfxWorld* asset = header.gfxWorld; Game::GfxWorld* asset = header.gfxWorld;
if (asset->worldDraw.reflectionImages) if (asset->draw.reflectionImages)
{ {
for (unsigned int i = 0; i < asset->worldDraw.reflectionProbeCount; ++i) for (unsigned int i = 0; i < asset->draw.reflectionProbeCount; ++i)
{ {
builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->worldDraw.reflectionImages[i]); builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->draw.reflectionImages[i]);
} }
} }
if (asset->worldDraw.lightmaps) if (asset->draw.lightmaps)
{ {
for (int i = 0; i < asset->worldDraw.lightmapCount; ++i) for (int i = 0; i < asset->draw.lightmapCount; ++i)
{ {
if (asset->worldDraw.lightmaps[i].primary) if (asset->draw.lightmaps[i].primary)
{ {
builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->worldDraw.lightmaps[i].primary); builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->draw.lightmaps[i].primary);
} }
if (asset->worldDraw.lightmaps[i].secondary) if (asset->draw.lightmaps[i].secondary)
{ {
builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->worldDraw.lightmaps[i].secondary); builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->draw.lightmaps[i].secondary);
} }
} }
} }
if (asset->worldDraw.skyImage) if (asset->draw.skyImage)
{ {
builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->worldDraw.skyImage); builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->draw.skyImage);
} }
if (asset->worldDraw.outdoorImage) if (asset->draw.outdoorImage)
{ {
builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->worldDraw.outdoorImage); builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->draw.outdoorImage);
} }
if (asset->sun.spriteMaterial) if (asset->sun.spriteMaterial)
@ -60,7 +194,7 @@ namespace Assets
if (asset->skies) if (asset->skies)
{ {
for (unsigned int i = 0; i < asset->skyCount; ++i) for (int i = 0; i < asset->skyCount; ++i)
{ {
if (asset->skies[i].skyImage) if (asset->skies[i].skyImage)
{ {
@ -80,14 +214,14 @@ namespace Assets
} }
} }
if (asset->unknownImage) if (asset->outdoorImage)
{ {
builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->unknownImage); builder->loadAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->outdoorImage);
} }
if (asset->dpvs.surfaces) if (asset->dpvs.surfaces)
{ {
for (int i = 0; i < asset->dpvsSurfaceCount; ++i) for (unsigned int i = 0; i < asset->surfaceCount; ++i)
{ {
if (asset->dpvs.surfaces[i].material) if (asset->dpvs.surfaces[i].material)
{ {
@ -398,7 +532,7 @@ namespace Assets
if (asset->sortedSurfIndex) if (asset->sortedSurfIndex)
{ {
buffer->align(Utils::Stream::ALIGN_2); buffer->align(Utils::Stream::ALIGN_2);
buffer->saveArray(asset->sortedSurfIndex, asset->staticSurfaceCount + asset->litSurfsBegin); buffer->saveArray(asset->sortedSurfIndex, asset->staticSurfaceCount + asset->staticSurfaceCountNoDecal);
Utils::Stream::ClearPointer(&dest->sortedSurfIndex); Utils::Stream::ClearPointer(&dest->sortedSurfIndex);
} }
@ -421,9 +555,9 @@ namespace Assets
buffer->align(Utils::Stream::ALIGN_4); buffer->align(Utils::Stream::ALIGN_4);
Game::GfxSurface* destSurfaceTable = buffer->dest<Game::GfxSurface>(); Game::GfxSurface* destSurfaceTable = buffer->dest<Game::GfxSurface>();
buffer->saveArray(asset->surfaces, world->dpvsSurfaceCount); buffer->saveArray(asset->surfaces, world->surfaceCount);
for (int i = 0; i < world->dpvsSurfaceCount; ++i) for (unsigned int i = 0; i < world->surfaceCount; ++i)
{ {
Game::GfxSurface* surface = &asset->surfaces[i]; Game::GfxSurface* surface = &asset->surfaces[i];
Game::GfxSurface* destSurface = &destSurfaceTable[i]; Game::GfxSurface* destSurface = &destSurfaceTable[i];
@ -444,7 +578,7 @@ namespace Assets
SaveLogEnter("GfxSurfaceBounds"); SaveLogEnter("GfxSurfaceBounds");
buffer->align(Utils::Stream::ALIGN_4); buffer->align(Utils::Stream::ALIGN_4);
buffer->saveArray(asset->surfacesBounds, world->dpvsSurfaceCount); buffer->saveArray(asset->surfacesBounds, world->surfaceCount);
Utils::Stream::ClearPointer(&dest->surfacesBounds); Utils::Stream::ClearPointer(&dest->surfacesBounds);
SaveLogExit(); SaveLogExit();
@ -482,7 +616,7 @@ namespace Assets
SaveLogEnter("GfxDrawSurf"); SaveLogEnter("GfxDrawSurf");
buffer->align(Utils::Stream::ALIGN_4); buffer->align(Utils::Stream::ALIGN_4);
buffer->saveArray(asset->surfaceMaterials, world->dpvsSurfaceCount); buffer->saveArray(asset->surfaceMaterials, world->surfaceCount);
Utils::Stream::ClearPointer(&dest->surfaceMaterials); Utils::Stream::ClearPointer(&dest->surfaceMaterials);
SaveLogExit(); SaveLogExit();
@ -579,7 +713,7 @@ namespace Assets
Game::GfxSky* destSkyTable = buffer->dest<Game::GfxSky>(); Game::GfxSky* destSkyTable = buffer->dest<Game::GfxSky>();
buffer->saveArray(asset->skies, asset->skyCount); buffer->saveArray(asset->skies, asset->skyCount);
for (unsigned int i = 0; i < asset->skyCount; ++i) for (int i = 0; i < asset->skyCount; ++i)
{ {
Game::GfxSky* destSky = &destSkyTable[i]; Game::GfxSky* destSky = &destSkyTable[i];
Game::GfxSky* sky = &asset->skies[i]; Game::GfxSky* sky = &asset->skies[i];
@ -732,7 +866,7 @@ namespace Assets
SaveLogExit(); SaveLogExit();
} }
this->saveGfxWorldDraw(&asset->worldDraw, &dest->worldDraw, builder); this->saveGfxWorldDraw(&asset->draw, &dest->draw, builder);
this->saveGfxLightGrid(&asset->lightGrid, &dest->lightGrid, builder); this->saveGfxLightGrid(&asset->lightGrid, &dest->lightGrid, builder);
if (asset->models) if (asset->models)
@ -773,25 +907,25 @@ namespace Assets
this->savesunflare_t(&asset->sun, &dest->sun, builder); this->savesunflare_t(&asset->sun, &dest->sun, builder);
if (asset->unknownImage) if (asset->outdoorImage)
{ {
dest->unknownImage = builder->saveSubAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->unknownImage).image; dest->outdoorImage = builder->saveSubAsset(Game::XAssetType::ASSET_TYPE_IMAGE, asset->outdoorImage).image;
} }
buffer->pushBlock(Game::XFILE_BLOCK_RUNTIME); buffer->pushBlock(Game::XFILE_BLOCK_RUNTIME);
if (asset->cellCasterBits[0]) if (asset->cellCasterBits)
{ {
buffer->align(Utils::Stream::ALIGN_4); buffer->align(Utils::Stream::ALIGN_4);
buffer->saveArray(asset->cellCasterBits[0], cellCount * ((cellCount + 31) >> 5)); buffer->saveArray(asset->cellCasterBits, cellCount * ((cellCount + 31) >> 5));
Utils::Stream::ClearPointer(&dest->cellCasterBits[0]); Utils::Stream::ClearPointer(&dest->cellCasterBits);
} }
if (asset->cellCasterBits[1]) if (asset->cellHasSunLitSurfsBits)
{ {
buffer->align(Utils::Stream::ALIGN_4); buffer->align(Utils::Stream::ALIGN_4);
buffer->saveArray(asset->cellCasterBits[1], ((cellCount + 31) >> 5)); buffer->saveArray(asset->cellHasSunLitSurfsBits, ((cellCount + 31) >> 5));
Utils::Stream::ClearPointer(&dest->cellCasterBits[1]); Utils::Stream::ClearPointer(&dest->cellHasSunLitSurfsBits);
} }
if (asset->sceneDynModel) if (asset->sceneDynModel)
@ -815,29 +949,29 @@ namespace Assets
if (asset->primaryLightEntityShadowVis) if (asset->primaryLightEntityShadowVis)
{ {
buffer->align(Utils::Stream::ALIGN_4); buffer->align(Utils::Stream::ALIGN_4);
buffer->save(asset->primaryLightEntityShadowVis, 1, (asset->unkCount2 + 0x1FFFF - asset->unkCount1) << 15); buffer->save(asset->primaryLightEntityShadowVis, 1, (asset->primaryLightCount + 0x1FFFF - asset->lastSunPrimaryLightIndex) << 15);
Utils::Stream::ClearPointer(&dest->primaryLightEntityShadowVis); Utils::Stream::ClearPointer(&dest->primaryLightEntityShadowVis);
} }
if (asset->primaryLightDynEntShadowVis[0]) if (asset->primaryLightDynEntShadowVis[0])
{ {
buffer->align(Utils::Stream::ALIGN_4); buffer->align(Utils::Stream::ALIGN_4);
buffer->saveArray(asset->primaryLightDynEntShadowVis[0], asset->dpvsDyn.dynEntClientCount[0] * (asset->unkCount2 - 1 - asset->unkCount1)); buffer->saveArray(asset->primaryLightDynEntShadowVis[0], asset->dpvsDyn.dynEntClientCount[0] * (asset->primaryLightCount - 1 - asset->lastSunPrimaryLightIndex));
Utils::Stream::ClearPointer(&dest->primaryLightDynEntShadowVis[0]); Utils::Stream::ClearPointer(&dest->primaryLightDynEntShadowVis[0]);
} }
if (asset->primaryLightDynEntShadowVis[1]) if (asset->primaryLightDynEntShadowVis[1])
{ {
buffer->align(Utils::Stream::ALIGN_4); buffer->align(Utils::Stream::ALIGN_4);
buffer->saveArray(asset->primaryLightDynEntShadowVis[1], asset->dpvsDyn.dynEntClientCount[1] * (asset->unkCount2 - 1 - asset->unkCount1)); buffer->saveArray(asset->primaryLightDynEntShadowVis[1], asset->dpvsDyn.dynEntClientCount[1] * (asset->primaryLightCount - 1 - asset->lastSunPrimaryLightIndex));
Utils::Stream::ClearPointer(&dest->primaryLightDynEntShadowVis[1]); Utils::Stream::ClearPointer(&dest->primaryLightDynEntShadowVis[1]);
} }
if (asset->primaryLightForModelDynEnt) if (asset->nonSunPrimaryLightForModelDynEnt)
{ {
// no align cause char // no align cause char
buffer->saveArray(asset->primaryLightForModelDynEnt, asset->dpvsDyn.dynEntClientCount[0]); buffer->saveArray(asset->nonSunPrimaryLightForModelDynEnt, asset->dpvsDyn.dynEntClientCount[0]);
Utils::Stream::ClearPointer(&dest->primaryLightForModelDynEnt); Utils::Stream::ClearPointer(&dest->nonSunPrimaryLightForModelDynEnt);
} }
buffer->popBlock(); buffer->popBlock();
@ -849,9 +983,9 @@ namespace Assets
buffer->align(Utils::Stream::ALIGN_4); buffer->align(Utils::Stream::ALIGN_4);
Game::GfxShadowGeometry* destShadowGeometryTable = buffer->dest<Game::GfxShadowGeometry>(); Game::GfxShadowGeometry* destShadowGeometryTable = buffer->dest<Game::GfxShadowGeometry>();
buffer->saveArray(asset->shadowGeom, asset->unkCount2); buffer->saveArray(asset->shadowGeom, asset->primaryLightCount);
for (int i = 0; i < asset->unkCount2; ++i) for (unsigned int i = 0; i < asset->primaryLightCount; ++i)
{ {
Game::GfxShadowGeometry* destShadowGeometry = &destShadowGeometryTable[i]; Game::GfxShadowGeometry* destShadowGeometry = &destShadowGeometryTable[i];
Game::GfxShadowGeometry* shadowGeometry = &asset->shadowGeom[i]; Game::GfxShadowGeometry* shadowGeometry = &asset->shadowGeom[i];
@ -882,9 +1016,9 @@ namespace Assets
buffer->align(Utils::Stream::ALIGN_4); buffer->align(Utils::Stream::ALIGN_4);
Game::GfxLightRegion* destLightRegionTable = buffer->dest<Game::GfxLightRegion>(); Game::GfxLightRegion* destLightRegionTable = buffer->dest<Game::GfxLightRegion>();
buffer->saveArray(asset->lightRegion, asset->unkCount2); buffer->saveArray(asset->lightRegion, asset->primaryLightCount);
for (int i = 0; i < asset->unkCount2; ++i) for (unsigned int i = 0; i < asset->primaryLightCount; ++i)
{ {
Game::GfxLightRegion* destLightRegion = &destLightRegionTable[i]; Game::GfxLightRegion* destLightRegion = &destLightRegionTable[i];
Game::GfxLightRegion* lightRegion = &asset->lightRegion[i]; Game::GfxLightRegion* lightRegion = &asset->lightRegion[i];
@ -929,12 +1063,12 @@ namespace Assets
this->saveGfxWorldDpvsStatic(asset, &asset->dpvs, &dest->dpvs, asset->dpvsPlanes.cellCount, builder); this->saveGfxWorldDpvsStatic(asset, &asset->dpvs, &dest->dpvs, asset->dpvsPlanes.cellCount, builder);
this->saveGfxWorldDpvsDynamic(&asset->dpvsDyn, &dest->dpvsDyn, builder); this->saveGfxWorldDpvsDynamic(&asset->dpvsDyn, &dest->dpvsDyn, builder);
if (asset->heroOnlyLight) if (asset->heroOnlyLights)
{ {
// no assert cause we use save manually here AssertSize(Game::GfxHeroOnlyLight, 56);
buffer->align(Utils::Stream::ALIGN_4); buffer->align(Utils::Stream::ALIGN_4);
buffer->save(asset->heroOnlyLight, 56, asset->heroOnlyLightCount); buffer->saveArray(asset->heroOnlyLights, asset->heroOnlyLightCount);
Utils::Stream::ClearPointer(&dest->heroOnlyLight); Utils::Stream::ClearPointer(&dest->heroOnlyLights);
} }
buffer->popBlock(); buffer->popBlock();

View File

@ -306,11 +306,11 @@ namespace Components
map.append(fmt::sprintf("mtllib %s.mtl\n\n", world->baseName)); map.append(fmt::sprintf("mtllib %s.mtl\n\n", world->baseName));
Logger::Print("Writing vertices...\n"); Logger::Print("Writing vertices...\n");
for (unsigned int i = 0; i < world->worldDraw.vertexCount; ++i) for (unsigned int i = 0; i < world->draw.vertexCount; ++i)
{ {
float x = world->worldDraw.vd.vertices[i].xyz[1]; float x = world->draw.vd.vertices[i].xyz[1];
float y = world->worldDraw.vd.vertices[i].xyz[2]; float y = world->draw.vd.vertices[i].xyz[2];
float z = world->worldDraw.vd.vertices[i].xyz[0]; float z = world->draw.vd.vertices[i].xyz[0];
map.append(fmt::sprintf("v %.6f %.6f %.6f\n", x, y, z)); map.append(fmt::sprintf("v %.6f %.6f %.6f\n", x, y, z));
} }
@ -318,9 +318,9 @@ namespace Components
map.append("\n"); map.append("\n");
Logger::Print("Writing texture coordinates...\n"); Logger::Print("Writing texture coordinates...\n");
for (unsigned int i = 0; i < world->worldDraw.vertexCount; ++i) for (unsigned int i = 0; i < world->draw.vertexCount; ++i)
{ {
map.append(fmt::sprintf("vt %.6f %.6f\n", world->worldDraw.vd.vertices[i].texCoord[0], -world->worldDraw.vd.vertices[i].texCoord[1])); map.append(fmt::sprintf("vt %.6f %.6f\n", world->draw.vd.vertices[i].texCoord[0], -world->draw.vd.vertices[i].texCoord[1]));
} }
Logger::Print("Searching materials...\n"); Logger::Print("Searching materials...\n");
@ -395,9 +395,9 @@ namespace Components
int indexOffset = world->dpvs.surfaces[i].tris.baseIndex; int indexOffset = world->dpvs.surfaces[i].tris.baseIndex;
for (unsigned short j = 0; j < world->dpvs.surfaces[i].tris.triCount; ++j) for (unsigned short j = 0; j < world->dpvs.surfaces[i].tris.triCount; ++j)
{ {
int a = world->worldDraw.indices[indexOffset + j * 3 + 0] + vertOffset; int a = world->draw.indices[indexOffset + j * 3 + 0] + vertOffset;
int b = world->worldDraw.indices[indexOffset + j * 3 + 1] + vertOffset; int b = world->draw.indices[indexOffset + j * 3 + 1] + vertOffset;
int c = world->worldDraw.indices[indexOffset + j * 3 + 2] + vertOffset; int c = world->draw.indices[indexOffset + j * 3 + 2] + vertOffset;
map.append(fmt::sprintf("f %d/%d %d/%d %d/%d\n", a, a, b, b, c, c)); map.append(fmt::sprintf("f %d/%d %d/%d %d/%d\n", a, a, b, b, c, c));
} }

View File

@ -1602,6 +1602,7 @@ namespace Game
float dist; float dist;
char type; char type;
char signbits; char signbits;
char pad[2];
}; };
struct cbrushside_t struct cbrushside_t
@ -2797,6 +2798,7 @@ namespace Game
{ {
unsigned int packed; unsigned int packed;
char array[4]; char array[4];
unsigned char uArray[4];
}; };
union PackedUnitVec union PackedUnitVec
@ -2967,9 +2969,16 @@ namespace Game
{ {
unsigned int smodelCount; unsigned int smodelCount;
unsigned int staticSurfaceCount; unsigned int staticSurfaceCount;
unsigned int staticSurfaceCountNoDecal;
unsigned int litSurfsBegin; unsigned int litSurfsBegin;
unsigned int litSurfsEnd; unsigned int litSurfsEnd;
char unknown1[0x20]; unsigned int decalSurfsBegin;
unsigned int decalSurfsEnd;
unsigned int emissiveSurfsBegin;
unsigned int emissiveSurfsEnd;
unsigned int smodelVisDataCount;
unsigned int surfaceVisDataCount;
int surfStuff;
int sunShadowCount; int sunShadowCount;
char *smodelVisData[3]; char *smodelVisData[3];
char *surfaceVisData[3]; char *surfaceVisData[3];
@ -3009,8 +3018,8 @@ namespace Game
struct GfxPortalWritable struct GfxPortalWritable
{ {
bool isQueued; char isQueued;
bool isAncestor; char isAncestor;
char recursionDepth; char recursionDepth;
char hullPointCount; char hullPointCount;
float(*hullPoints)[2]; float(*hullPoints)[2];
@ -3020,6 +3029,7 @@ namespace Game
{ {
float coeffs[4]; float coeffs[4];
char side[3]; char side[3];
char pad;
}; };
struct GfxPortal struct GfxPortal
@ -3198,48 +3208,56 @@ namespace Game
int skySamplerState; int skySamplerState;
}; };
struct GfxHeroOnlyLight
{
char pad[56];
};
struct GfxWorld struct GfxWorld
{ {
const char *name; const char *name;
const char *baseName; const char *baseName;
int planeCount; int planeCount;
int nodeCount; int nodeCount;
int dpvsSurfaceCount; unsigned int surfaceCount;
unsigned int skyCount; int skyCount;
GfxSky *skies; GfxSky *skies;
int unkCount1; unsigned int lastSunPrimaryLightIndex;
int unkCount2; unsigned int primaryLightCount;
char unknown1[16]; unsigned int sortKeyLitDecal;
GfxWorldDpvsPlanes dpvsPlanes; //The following rely on the count in this unsigned int sortKeyEffectDecal;
unsigned int sortKeyEffectAuto;
unsigned int sortKeyDistortion;
GfxWorldDpvsPlanes dpvsPlanes;
GfxCellTreeCount *aabbTreeCounts; GfxCellTreeCount *aabbTreeCounts;
GfxCellTree *aabbTrees; GfxCellTree *aabbTrees;
GfxCell *cells; GfxCell *cells;
GfxWorldDraw worldDraw; GfxWorldDraw draw;
GfxLightGrid lightGrid; GfxLightGrid lightGrid;
int modelCount; int modelCount;
GfxBrushModel *models; GfxBrushModel *models;
float mins[3]; Bounds bounds;
float maxs[3];
unsigned int checksum; unsigned int checksum;
int materialMemoryCount; int materialMemoryCount;
MaterialMemory *materialMemory; MaterialMemory *materialMemory;
sunflare_t sun; sunflare_t sun;
char pad[64]; float outdoorLookupMatrix[4][4];
GfxImage* unknownImage; GfxImage *outdoorImage;
unsigned int *cellCasterBits[2]; unsigned int *cellCasterBits;
unsigned int *cellHasSunLitSurfsBits;
GfxSceneDynModel *sceneDynModel; GfxSceneDynModel *sceneDynModel;
GfxSceneDynBrush *sceneDynBrush; GfxSceneDynBrush *sceneDynBrush;
unsigned int *primaryLightEntityShadowVis; unsigned int *primaryLightEntityShadowVis;
unsigned int *primaryLightDynEntShadowVis[2]; unsigned int *primaryLightDynEntShadowVis[2];
char *primaryLightForModelDynEnt; char *nonSunPrimaryLightForModelDynEnt;
GfxShadowGeometry *shadowGeom; GfxShadowGeometry *shadowGeom;
GfxLightRegion *lightRegion; GfxLightRegion *lightRegion;
GfxWorldDpvsStatic dpvs; GfxWorldDpvsStatic dpvs;
GfxWorldDpvsDynamic dpvsDyn; GfxWorldDpvsDynamic dpvsDyn;
int pad2; unsigned int mapVtxChecksum;
unsigned int heroOnlyLightCount; unsigned int heroOnlyLightCount;
char * heroOnlyLight; GfxHeroOnlyLight *heroOnlyLights;
int unknown5; char fogTypesAllowed;
}; };
#pragma pack(pop) #pragma pack(pop)

View File

@ -59,6 +59,29 @@ namespace Utils
} }
} }
void* Stream::Reader::readPointer()
{
void* pointer = this->read<void*>();
if (!this->hasPointer(pointer))
{
this->pointerMap[pointer] = 0;
}
return pointer;
}
void Stream::Reader::mapPointer(void* oldPointer, void* newPointer)
{
if (this->hasPointer(oldPointer))
{
this->pointerMap[oldPointer] = newPointer;
}
}
bool Stream::Reader::hasPointer(void* pointer)
{
return this->pointerMap.find(pointer) != this->pointerMap.end();
}
Stream::Stream() : criticalSectionState(0) Stream::Stream() : criticalSectionState(0)
{ {
memset(this->blockSize, 0, sizeof(this->blockSize)); memset(this->blockSize, 0, sizeof(this->blockSize));

View File

@ -49,9 +49,14 @@ namespace Utils
bool end(); bool end();
void seek(unsigned int position); void seek(unsigned int position);
void* readPointer();
void mapPointer(void* oldPointer, void* newPointer);
bool hasPointer(void* pointer);
private: private:
unsigned int position; unsigned int position;
std::string buffer; std::string buffer;
std::map<void*, void*> pointerMap;
Utils::Memory::Allocator* allocator; Utils::Memory::Allocator* allocator;
}; };