// NeL - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 of the // License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "std3d.h" #include "nel/3d/stripifier.h" // For now, don't use NVidia stripifier. //#include "nv_tri_strip_objects.h" #include #include using namespace std; #ifdef DEBUG_NEW #define new DEBUG_NEW #endif namespace NL3D { // *************************************************************************** CStripifier::CStripifier() { } // *************************************************************************** /* NVidia(tm) 's method get better performance (8ms on 50K faces meshe, instead of 8.9ms), but precomputing is much slower (1'40 instead of 0'?? :) ). */ /*void CStripifier::optimizeTriangles(const CIndexBuffer &in, CIndexBuffer &out, uint cacheSize) { NvStripifier stripifier; WordVec inIndices; NvStripInfoVec outStrips; sint i; // prepare inIndices. inIndices.resize(in.getNumTri()*3); for(i=0;i< (sint)inIndices.size(); i++) { inIndices[i]= in.getPtr()[i]; } // build strips. stripifier.Stripify(inIndices, cacheSize, outStrips); // build triangles from strips, and release memory out.setNumTri(0); out.reserveTri(in.getNumTri()); for(i= 0;i< (sint)outStrips.size(); i++) { NvStripInfo *stripInfo= outStrips[i]; // build triangle from the strip. for(uint j= 0;j< stripInfo->m_faces.size(); j++) { NvFaceInfo *face= stripInfo->m_faces[j]; out.addTri(face->m_v0, face->m_v1, face->m_v2); // delete this face. delete face; } // delete this strip. // Unref first the edges touched by this strip. NvEdgeInfo *edgeInfo = stripInfo->m_startInfo.m_startEdge; while(edgeInfo) { NvEdgeInfo *edgeInfoNext= edgeInfo->m_nextV1; edgeInfo->Unref(); edgeInfo= edgeInfoNext; } // delete delete stripInfo; } }*/ // *************************************************************************** struct CVertexCache { CVertexCache(sint cacheSize, sint nbVerts) { _VertexInCache.resize(nbVerts, 0); _Cache.resize(cacheSize, 0xFFFFFFFF); } void touchVertex(uint vert) { if(isVertexInCache(vert)) { // do nothing ?????? depends of vcache implementation } else { // pop front uint removed= _Cache.front(); if(removed!=0xFFFFFFFF) _VertexInCache[removed]= 0; _Cache.pop_front(); // push_back _VertexInCache[vert]= 3; _Cache.push_back(vert); } } bool isVertexInCache(uint vert) { return _VertexInCache[vert]==3; } // return which vertex is at which place in the cache. 0xFFFFFFFF if the entry is empty uint getVertexInCache(uint vertIdInCache) { return _Cache[vertIdInCache]; } void tempTouchVertex(uint vert, bool inCache) { if( _VertexInCache[vert]&1 ) { if(inCache) _VertexInCache[vert]|= 2; else _VertexInCache[vert]&= 1; } } private: // 0 if not in the cache vector _VertexInCache; deque _Cache; }; // *************************************************************************** struct COrderFace { sint v[3]; bool Inserted; void insertInPB(CIndexBuffer &out, CVertexCache &vertexCache) { uint index = out.getNumIndexes (); out.setNumIndexes (index+3); CIndexBufferReadWrite ibaWrite; out.lock (ibaWrite); ibaWrite.setTri(index, v[0], v[1], v[2]); vertexCache.touchVertex(v[0]); vertexCache.touchVertex(v[1]); vertexCache.touchVertex(v[2]); Inserted= true; } sint countCacheMiss(CVertexCache &vertexCache) { sint ret=0 ; if(!vertexCache.isVertexInCache(v[0])) ret++; if(!vertexCache.isVertexInCache(v[1])) ret++; if(!vertexCache.isVertexInCache(v[2])) ret++; return ret; } }; // *************************************************************************** struct CCornerNode { // next in the list CCornerNode *Next; // corner == tuple face/vertex. uint FaceId; uint VertexId; }; // *************************************************************************** void CStripifier::optimizeTriangles(const CIndexBuffer &in, CIndexBuffer &out, uint cacheSize) { vector inFaces; sint i; sint numTris= in.getNumIndexes()/3; // TestYoyo: All the same tri => perfect vertex caching... /*out.setNumTri(numTris); for(i=0;i< numTris; i++) { uint32 v0= *(in.getPtr()+0); uint32 v1= *(in.getPtr()+1); uint32 v2= *(in.getPtr()+2); out.setTri(i, v0, v1, v2); } return;*/ // prepare inIndices. //-------------------- { CIndexBufferRead ibaRead; in.lock (ibaRead); inFaces.resize(numTris); if (ibaRead.getFormat() == CIndexBuffer::Indices32) { for(i=0;i< numTris; i++) { const uint32 *ibaPtr = (const uint32 *) ibaRead.getPtr(); inFaces[i].v[0]= ibaPtr[i*3 + 0]; inFaces[i].v[1]= ibaPtr[i*3 + 1]; inFaces[i].v[2]= ibaPtr[i*3 + 2]; inFaces[i].Inserted= false; } } else { nlassert(ibaRead.getFormat() == CIndexBuffer::Indices16); for(i=0;i< numTris; i++) { const uint16 *ibaPtr = (const uint16 *) ibaRead.getPtr(); inFaces[i].v[0]= ibaPtr[i*3 + 0]; inFaces[i].v[1]= ibaPtr[i*3 + 1]; inFaces[i].v[2]= ibaPtr[i*3 + 2]; inFaces[i].Inserted= false; } } } // build our cache, and compute max number of vertices. //-------------------- int numVerts=0; for (i = 0; i < numTris; i++) { numVerts= max(numVerts, (int)inFaces[i].v[0]); numVerts= max(numVerts, (int)inFaces[i].v[1]); numVerts= max(numVerts, (int)inFaces[i].v[2]); } numVerts++; CVertexCache vertexCache(cacheSize, numVerts); // Compute vertex connectivity. //-------------------- vector vertexConnectivity; vector cornerAllocator; cornerAllocator.resize(numTris * 3); vertexConnectivity.resize(numVerts, NULL); // For all triangles. for (i = 0; i < numTris; i++) { COrderFace *ordFace= &inFaces[i]; // For each corner, allocate and fill for(sint j=0; j<3;j++) { sint vertexId= ordFace->v[j]; // allocate a corner CCornerNode *corner= &cornerAllocator[i*3 + j]; // fill it. corner->FaceId= i; corner->VertexId= vertexId; // Link it to the vertex list of faces. corner->Next= vertexConnectivity[vertexId]; vertexConnectivity[vertexId]= corner; } } // build output optimized triangles //-------------------- out.setFormat(in.getFormat()); out.setNumIndexes(0); out.reserve(3*numTris); for(i=0; iFaceId; // if the face is not yet inserted. if(!inFaces[faceId].Inserted) { sint c= inFaces[faceId].countCacheMiss(vertexCache); // insert first any face which don't add any vertex in the cache. if(c==0) { inFaces[faceId].insertInPB(out, vertexCache); } // else the one which add the minimum of vertex possible: nextToInsert else { // Add cost of faces that use vertices pushed out (better results...) uint numVOut= c; uint k; for(k=cacheSize-numVOut; kFaceId; // if the face is not yet inserted AND not the one treated if(!inFaces[faceOutId].Inserted && faceOutId!=faceId) { // Add cache miss of this face c+= inFaces[faceOutId].countCacheMiss(vertexCache); } // next corner cornerOut= cornerOut->Next; } } // reset touch for(k=cacheSize-numVOut; kNext; } } // if nextToInsertFound, then nextToInsert has the face which add the minimum of vertex possible in the cache } } } } } // NL3D