index buffer generation using final vertex data (#1816)
This results a smaller output file for some meshes.
This commit is contained in:
parent
45fb76865e
commit
72d0a0c4fa
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -11,8 +11,6 @@
|
|||||||
|
|
||||||
#include <tinystl/allocator.h>
|
#include <tinystl/allocator.h>
|
||||||
#include <tinystl/string.h>
|
#include <tinystl/string.h>
|
||||||
#include <tinystl/unordered_map.h>
|
|
||||||
#include <tinystl/unordered_set.h>
|
|
||||||
#include <tinystl/vector.h>
|
#include <tinystl/vector.h>
|
||||||
namespace stl = tinystl;
|
namespace stl = tinystl;
|
||||||
|
|
||||||
@ -63,15 +61,12 @@ struct Index3
|
|||||||
int32_t m_position;
|
int32_t m_position;
|
||||||
int32_t m_texcoord;
|
int32_t m_texcoord;
|
||||||
int32_t m_normal;
|
int32_t m_normal;
|
||||||
int32_t m_vertexIndex;
|
|
||||||
int32_t m_vbc; // Barycentric ID. Holds eigher 0, 1 or 2.
|
int32_t m_vbc; // Barycentric ID. Holds eigher 0, 1 or 2.
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef stl::unordered_map<uint64_t, Index3> Index3Map;
|
|
||||||
|
|
||||||
struct TriIndices
|
struct TriIndices
|
||||||
{
|
{
|
||||||
uint64_t m_index[3];
|
Index3 m_index[3];
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef stl::vector<TriIndices> TriangleArray;
|
typedef stl::vector<TriIndices> TriangleArray;
|
||||||
@ -498,7 +493,6 @@ int main(int _argc, const char* _argv[])
|
|||||||
Vec3Array positions;
|
Vec3Array positions;
|
||||||
Vec3Array normals;
|
Vec3Array normals;
|
||||||
Vec3Array texcoords;
|
Vec3Array texcoords;
|
||||||
Index3Map indexMap;
|
|
||||||
TriangleArray triangles;
|
TriangleArray triangles;
|
||||||
GroupArray groups;
|
GroupArray groups;
|
||||||
|
|
||||||
@ -539,7 +533,6 @@ int main(int _argc, const char* _argv[])
|
|||||||
Index3 index;
|
Index3 index;
|
||||||
index.m_texcoord = -1;
|
index.m_texcoord = -1;
|
||||||
index.m_normal = -1;
|
index.m_normal = -1;
|
||||||
index.m_vertexIndex = -1;
|
|
||||||
if (hasBc)
|
if (hasBc)
|
||||||
{
|
{
|
||||||
index.m_vbc = edge < 3 ? edge : (1+(edge+1) )&1;
|
index.m_vbc = edge < 3 ? edge : (1+(edge+1) )&1;
|
||||||
@ -583,29 +576,10 @@ int main(int _argc, const char* _argv[])
|
|||||||
index.m_position = (pos < 0) ? pos + numPositions : pos - 1;
|
index.m_position = (pos < 0) ? pos + numPositions : pos - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
const uint64_t hash0 = uint64_t(index.m_position)<< 0;
|
|
||||||
const uint64_t hash1 = uint64_t(index.m_texcoord)<<20;
|
|
||||||
const uint64_t hash2 = uint64_t(index.m_normal )<<40;
|
|
||||||
const uint64_t hash3 = uint64_t(index.m_vbc )<<60;
|
|
||||||
const uint64_t hash = hash0^hash1^hash2^hash3;
|
|
||||||
|
|
||||||
stl::pair<Index3Map::iterator, bool> result = indexMap.insert(stl::make_pair(hash, index) );
|
|
||||||
if (!result.second)
|
|
||||||
{
|
|
||||||
Index3& oldIndex = result.first->second;
|
|
||||||
BX_UNUSED(oldIndex);
|
|
||||||
BX_CHECK(true
|
|
||||||
&& oldIndex.m_position == index.m_position
|
|
||||||
&& oldIndex.m_texcoord == index.m_texcoord
|
|
||||||
&& oldIndex.m_normal == index.m_normal
|
|
||||||
, "Hash collision!"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (edge)
|
switch (edge)
|
||||||
{
|
{
|
||||||
case 0: case 1: case 2:
|
case 0: case 1: case 2:
|
||||||
triangle.m_index[edge] = hash;
|
triangle.m_index[edge] = index;
|
||||||
if (2 == edge)
|
if (2 == edge)
|
||||||
{
|
{
|
||||||
if (ccw)
|
if (ccw)
|
||||||
@ -620,12 +594,12 @@ int main(int _argc, const char* _argv[])
|
|||||||
if (ccw)
|
if (ccw)
|
||||||
{
|
{
|
||||||
triangle.m_index[2] = triangle.m_index[1];
|
triangle.m_index[2] = triangle.m_index[1];
|
||||||
triangle.m_index[1] = hash;
|
triangle.m_index[1] = index;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
triangle.m_index[1] = triangle.m_index[2];
|
triangle.m_index[1] = triangle.m_index[2];
|
||||||
triangle.m_index[2] = hash;
|
triangle.m_index[2] = index;
|
||||||
}
|
}
|
||||||
|
|
||||||
triangles.push_back(triangle);
|
triangles.push_back(triangle);
|
||||||
@ -770,38 +744,50 @@ int main(int _argc, const char* _argv[])
|
|||||||
bool hasNormal;
|
bool hasNormal;
|
||||||
bool hasTexcoord;
|
bool hasTexcoord;
|
||||||
{
|
{
|
||||||
Index3Map::const_iterator it = indexMap.begin();
|
TriangleArray::const_iterator it = triangles.begin();
|
||||||
hasNormal = -1 != it->second.m_normal;
|
hasNormal = -1 != it->m_index[0].m_normal;
|
||||||
hasTexcoord = -1 != it->second.m_texcoord;
|
hasTexcoord = -1 != it->m_index[0].m_texcoord;
|
||||||
|
|
||||||
if (!hasTexcoord)
|
if (!hasTexcoord)
|
||||||
{
|
{
|
||||||
for (Index3Map::iterator jt = indexMap.begin(), jtEnd = indexMap.end(); jt != jtEnd && !hasTexcoord; ++jt)
|
for (TriangleArray::iterator jt = triangles.begin(), jtEnd = triangles.end(); jt != jtEnd && !hasTexcoord; ++jt)
|
||||||
{
|
{
|
||||||
hasTexcoord |= -1 != jt->second.m_texcoord;
|
for (uint32_t i = 0; i < 3; ++i)
|
||||||
|
{
|
||||||
|
hasTexcoord |= -1 != jt->m_index[i].m_texcoord;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hasTexcoord)
|
if (hasTexcoord)
|
||||||
{
|
{
|
||||||
for (Index3Map::iterator jt = indexMap.begin(), jtEnd = indexMap.end(); jt != jtEnd; ++jt)
|
for (TriangleArray::iterator jt = triangles.begin(), jtEnd = triangles.end(); jt != jtEnd; ++jt)
|
||||||
{
|
{
|
||||||
jt->second.m_texcoord = -1 == jt->second.m_texcoord ? 0 : jt->second.m_texcoord;
|
for (uint32_t i = 0; i < 3; ++i)
|
||||||
|
{
|
||||||
|
jt->m_index[i].m_texcoord = -1 == jt->m_index[i].m_texcoord ? 0 : jt->m_index[i].m_texcoord;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!hasNormal)
|
if (!hasNormal)
|
||||||
{
|
{
|
||||||
for (Index3Map::iterator jt = indexMap.begin(), jtEnd = indexMap.end(); jt != jtEnd && !hasNormal; ++jt)
|
for (TriangleArray::iterator jt = triangles.begin(), jtEnd = triangles.end(); jt != jtEnd && !hasNormal; ++jt)
|
||||||
{
|
{
|
||||||
hasNormal |= -1 != jt->second.m_normal;
|
for (uint32_t i = 0; i < 3; ++i)
|
||||||
|
{
|
||||||
|
hasNormal |= -1 != jt->m_index[i].m_normal;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hasNormal)
|
if (hasNormal)
|
||||||
{
|
{
|
||||||
for (Index3Map::iterator jt = indexMap.begin(), jtEnd = indexMap.end(); jt != jtEnd; ++jt)
|
for (TriangleArray::iterator jt = triangles.begin(), jtEnd = triangles.end(); jt != jtEnd; ++jt)
|
||||||
{
|
{
|
||||||
jt->second.m_normal = -1 == jt->second.m_normal ? 0 : jt->second.m_normal;
|
for (uint32_t i = 0; i < 3; ++i)
|
||||||
|
{
|
||||||
|
jt->m_index[i].m_normal = -1 == jt->m_index[i].m_normal ? 0 : jt->m_index[i].m_normal;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -876,6 +862,11 @@ int main(int _argc, const char* _argv[])
|
|||||||
uint8_t* vertices = vertexData;
|
uint8_t* vertices = vertexData;
|
||||||
uint16_t* indices = indexData;
|
uint16_t* indices = indexData;
|
||||||
|
|
||||||
|
const uint32_t tableSize = 65536 * 2;
|
||||||
|
const uint32_t hashmod = tableSize - 1;
|
||||||
|
uint32_t* table = new uint32_t[tableSize];
|
||||||
|
bx::memSet(table, 0xff, tableSize * sizeof(uint32_t));
|
||||||
|
|
||||||
stl::string material = groups.begin()->m_material;
|
stl::string material = groups.begin()->m_material;
|
||||||
|
|
||||||
PrimitiveArray primitives;
|
PrimitiveArray primitives;
|
||||||
@ -943,10 +934,7 @@ int main(int _argc, const char* _argv[])
|
|||||||
);
|
);
|
||||||
primitives.clear();
|
primitives.clear();
|
||||||
|
|
||||||
for (Index3Map::iterator indexIt = indexMap.begin(); indexIt != indexMap.end(); ++indexIt)
|
bx::memSet(table, 0xff, tableSize * sizeof(uint32_t));
|
||||||
{
|
|
||||||
indexIt->second.m_vertexIndex = -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
++writtenPrimitives;
|
++writtenPrimitives;
|
||||||
writtenVertices += numVertices;
|
writtenVertices += numVertices;
|
||||||
@ -968,11 +956,7 @@ int main(int _argc, const char* _argv[])
|
|||||||
TriIndices& triangle = triangles[tri];
|
TriIndices& triangle = triangles[tri];
|
||||||
for (uint32_t edge = 0; edge < 3; ++edge)
|
for (uint32_t edge = 0; edge < 3; ++edge)
|
||||||
{
|
{
|
||||||
uint64_t hash = triangle.m_index[edge];
|
Index3& index = triangle.m_index[edge];
|
||||||
Index3& index = indexMap[hash];
|
|
||||||
if (index.m_vertexIndex == -1)
|
|
||||||
{
|
|
||||||
index.m_vertexIndex = numVertices++;
|
|
||||||
|
|
||||||
float* position = (float*)(vertices + positionOffset);
|
float* position = (float*)(vertices + positionOffset);
|
||||||
bx::memCopy(position, &positions[index.m_position], 3*sizeof(float) );
|
bx::memCopy(position, &positions[index.m_position], 3*sizeof(float) );
|
||||||
@ -1011,13 +995,42 @@ int main(int _argc, const char* _argv[])
|
|||||||
{
|
{
|
||||||
float normal[4];
|
float normal[4];
|
||||||
bx::store(normal, bx::normalize(bx::load<bx::Vec3>(&normals[index.m_normal]) ) );
|
bx::store(normal, bx::normalize(bx::load<bx::Vec3>(&normals[index.m_normal]) ) );
|
||||||
|
normal[3] = 0.0f;
|
||||||
bgfx::vertexPack(normal, true, bgfx::Attrib::Normal, decl, vertices);
|
bgfx::vertexPack(normal, true, bgfx::Attrib::Normal, decl, vertices);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uint32_t hash = bx::hash<bx::HashMurmur2A>(vertices, stride);
|
||||||
|
size_t bucket = hash & hashmod;
|
||||||
|
uint32_t vertexIndex = UINT32_MAX;
|
||||||
|
|
||||||
|
for (size_t probe = 0; probe <= hashmod; ++probe)
|
||||||
|
{
|
||||||
|
uint32_t& item = table[bucket];
|
||||||
|
|
||||||
|
if (item == ~0u)
|
||||||
|
{
|
||||||
vertices += stride;
|
vertices += stride;
|
||||||
|
item = numVertices++;
|
||||||
|
vertexIndex = item;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
*indices++ = (uint16_t)index.m_vertexIndex;
|
if (0 == bx::memCmp(vertexData + item * stride, vertices, stride))
|
||||||
|
{
|
||||||
|
vertexIndex = item;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
bucket = (bucket + probe + 1) & hashmod;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( vertexIndex == UINT32_MAX )
|
||||||
|
{
|
||||||
|
bx::printf("hash table insert failed");
|
||||||
|
exit(bx::kExitFailure);
|
||||||
|
}
|
||||||
|
|
||||||
|
*indices++ = (uint16_t)vertexIndex;
|
||||||
++numIndices;
|
++numIndices;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1045,6 +1058,7 @@ int main(int _argc, const char* _argv[])
|
|||||||
bx::printf("size: %d\n", uint32_t(bx::seek(&writer) ) );
|
bx::printf("size: %d\n", uint32_t(bx::seek(&writer) ) );
|
||||||
bx::close(&writer);
|
bx::close(&writer);
|
||||||
|
|
||||||
|
delete [] table;
|
||||||
delete [] indexData;
|
delete [] indexData;
|
||||||
delete [] vertexData;
|
delete [] vertexData;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user