// 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/vegetablevb_allocator.h" #include "nel/3d/vegetable_def.h" using namespace std; using namespace NLMISC; namespace NL3D { /* Once a reallocation of a VB occurs, how many vertices we add to the re-allocation, to avoid as possible reallocations. */ #define NL3D_VEGETABLE_VERTEX_ALLOCATE_SECURITY 1024 /* The start size of the array. */ #define NL3D_VEGETABLE_VERTEX_ALLOCATE_START 4048 #define NL3D_VEGETABLE_VERTEX_FREE_MEMORY_RESERVE 1024 // *************************************************************************** CVegetableVBAllocator::CVegetableVBAllocator() { _Type= VBTypeUnlit; _MaxVertexInBufferHard= 0; // Init free list _VertexFreeMemory.reserve(NL3D_VEGETABLE_VERTEX_FREE_MEMORY_RESERVE); _NumVerticesAllocated= 0; // Init vbhard _VBHardOk= false; _AGPBufferPtr= NULL; _RAMBufferPtr= NULL; } // *************************************************************************** void CVegetableVBAllocator::init(TVBType type, uint maxVertexInBufferHard) { _Type= type; _MaxVertexInBufferHard= maxVertexInBufferHard; // According to _Type, build VB format, and create VertexProgram setupVBFormat(); } // *************************************************************************** CVegetableVBAllocator::~CVegetableVBAllocator() { clear(); } // *************************************************************************** void CVegetableVBAllocator::updateDriver(IDriver *driver) { // test change of driver. nlassert(driver && !_VBHard.isLocked()); // If change of driver if( _Driver==NULL || driver!=_Driver || (!_VBHard.isResident() && (_VBHard.capacity()!=0))) { // delete old VBHard. deleteVertexBufferHard(); _Driver= driver; _VBHardOk= (_MaxVertexInBufferHard>0) && (_Driver->supportVertexBufferHard()); /* Because so much lock/unlock are performed during a frame (refine/clip etc...). we must disable VBHard for ATI Gl extension. NB: CLandscape don't do this and fast copy the entire VB each frame. This is not possible for vegetables because the VB describe all Vegetable around the camera, not only what is in frustrum. Hence a fast copy each frame would copy far too much unseen vertices (4x). */ if(_Driver->slowUnlockVertexBufferHard()) _VBHardOk= false; // Driver must support VP. nlassert(_Driver->supportVertexProgram(CVertexProgram::nelvp) // || _Driver->supportVertexProgram(CVertexProgram::glsl330v) // TODO_VP_GLSL ); // must reallocate the VertexBuffer. if( _NumVerticesAllocated>0 ) allocateVertexBufferAndFillVBHard(_NumVerticesAllocated); } else { // if VBHard possible, and if vbHardDeleted but space needed, reallocate. if( _VBHardOk && _VBHard.getNumVertices()==0 && _NumVerticesAllocated>0 ) allocateVertexBufferAndFillVBHard(_NumVerticesAllocated); } } // *************************************************************************** void CVegetableVBAllocator::clear() { // clear list. _VertexFreeMemory.clear(); _NumVerticesAllocated= 0; // must unlock for vbhard and vbsoft unlockBuffer(); // delete the VB. deleteVertexBufferHard(); // really delete the VB soft too _VBSoft.deleteAllVertices(); // clear other states. _Driver= NULL; _VBHardOk= false; } // *************************************************************************** void CVegetableVBAllocator::lockBuffer() { // force unlock unlockBuffer(); // need to lock only if the VBHard is created if(_VBHardOk) { // lock the VBHard for writing _VBHard.lock(_VBAHard); _AGPBufferPtr=(uint8*)_VBAHard.getVertexCoordPointer(); // lock the Input VertexBuffer for reading _VBSoft.lock(_VBASoft); _RAMBufferPtr=(const uint8*)_VBASoft.getVertexCoordPointer(); } } // *************************************************************************** void CVegetableVBAllocator::unlockBuffer() { // unlock the VBHard _VBAHard.unlock(); _AGPBufferPtr= NULL; // unlock the VBSoft _VBASoft.unlock(); _RAMBufferPtr= NULL; } // *************************************************************************** uint CVegetableVBAllocator::getNumUserVerticesAllocated() const { // get the number of vertices which are allocated by allocateVertex(). return _NumVerticesAllocated - (uint)_VertexFreeMemory.size(); } // *************************************************************************** bool CVegetableVBAllocator::exceedMaxVertexInBufferHard(uint numAddVerts) const { return (getNumUserVerticesAllocated() + numAddVerts) > _MaxVertexInBufferHard; } // *************************************************************************** uint CVegetableVBAllocator::allocateVertex() { // if no more free, allocate. if( _VertexFreeMemory.empty() ) { // enlarge capacity. uint newResize; if(_NumVerticesAllocated==0) newResize= NL3D_VEGETABLE_VERTEX_ALLOCATE_START; else newResize= NL3D_VEGETABLE_VERTEX_ALLOCATE_SECURITY; // try to not overlap _MaxVertexInBufferHard limit, to avoid VBufferHard to be disabled. if(_NumVerticesAllocated<_MaxVertexInBufferHard && _NumVerticesAllocated+newResize > _MaxVertexInBufferHard) { newResize= _MaxVertexInBufferHard - _NumVerticesAllocated; } _NumVerticesAllocated+= newResize; // re-allocate VB. allocateVertexBufferAndFillVBHard(_NumVerticesAllocated); // resize infos on vertices. _VertexInfos.resize(_NumVerticesAllocated); // Fill list of free elements. for(uint i=0;iactiveVertexBuffer(_VBHard); else _Driver->activeVertexBuffer(_VBSoft); } // *************************************************************************** // *************************************************************************** // Vertex Buffer hard. // *************************************************************************** // *************************************************************************** // *************************************************************************** void CVegetableVBAllocator::deleteVertexBufferHard() { // must unlock VBhard before. unlockBuffer(); // test (refptr) if the object still exist in memory. if(_VBHard.getNumVertices()!=0) { // A vbufferhard should still exist only if driver still exist. nlassert(_Driver!=NULL); // delete it from driver. _VBHard.deleteAllVertices (); } } // *************************************************************************** void CVegetableVBAllocator::allocateVertexBufferAndFillVBHard(uint32 numVertices) { // no allocation must be done if the Driver is not setuped, or if the driver has been deleted by refPtr. nlassert(_Driver); // must unlock VBhard and VBSoft before. bool wasLocked= bufferLocked(); unlockBuffer(); // resize the Soft VB. _VBSoft.setNumVertices(numVertices); // try to allocate a vbufferhard if possible. if( _VBHardOk ) { /* Prefer allocate the VBHard with the Max Vertex count only ONCE, to avoid problems with AGP allocation The problem is with 50000 AGP vertices, it costs 3 Mo. If we do iterative allocation (500 Ko, 600 Ko, 700 Ko,.....) we may have problem with free holes (Vegetable could no more enter in the 16 or 8 Mo AGP limit!) */ if(_VBHard.getNumVertices() != _MaxVertexInBufferHard) { // delete possible old _VBHard. if(_VBHard.getNumVertices()!=0) { // VertexBufferHard lifetime < Driver lifetime. nlassert(_Driver!=NULL); _VBHard.deleteAllVertices(); } // try to create new one, in AGP Ram // If too many vertices wanted, abort VBHard. if(numVertices <= _MaxVertexInBufferHard) { _VBHard = _VBSoft; _VBHard.setPreferredMemory(CVertexBuffer::AGPPreferred, false); _VBHard.setNumVertices (_MaxVertexInBufferHard); // Force this VB to be hard nlverify (_Driver->activeVertexBuffer (_VBHard)); nlassert (_VBHard.isResident()); // if fails, abort VBHard. if (_VBHard.getLocation() == CVertexBuffer::RAMResident) _VBHard.deleteAllVertices(); // Set Name For lock Profiling. if(_VBHard.getNumVertices()!=0) _VBHard.setName("VegetableVB"); } else _VBHard.deleteAllVertices(); // If KO, never try again. if(_VBHard.getNumVertices()==0) _VBHardOk= false; } } // if still OK, must refill the VBHard. Slow, but rare if(_VBHardOk) { // else, fill this AGP VBuffer Hard. // lock before the AGP buffer lockBuffer(); // copy all the vertices to AGP. memcpy(_AGPBufferPtr, _RAMBufferPtr, _VBSoft.getVertexSize() * numVertices); // If was not locked before, unlock this VB if(!wasLocked) unlockBuffer(); } //nlinfo("VEGET: Alloc %d verts. %s", numVertices, _VBHardOk?"VBHard":"VBSoft"); } // *************************************************************************** void CVegetableVBAllocator::setupVBFormat() { // Build the Vertex Format. _VBSoft.clearValueEx(); // if lighted, need world space normal and AmbientColor for each vertex. if( _Type == VBTypeLighted ) { _VBSoft.addValueEx(NL3D_VEGETABLE_VPPOS_POS, CVertexBuffer::Float3); // v[0] _VBSoft.addValueEx(NL3D_VEGETABLE_VPPOS_NORMAL, CVertexBuffer::Float3); // v[2] _VBSoft.addValueEx(NL3D_VEGETABLE_VPPOS_BENDINFO, CVertexBuffer::Float3); // v[9] } // If unlit else { // slightly different VertexProgram, v[0].w== BendWeight, and v[9].x== v[0].norm() _VBSoft.addValueEx(NL3D_VEGETABLE_VPPOS_POS, CVertexBuffer::Float4); // v[0] // Unlit VP has BlendDistance in v[9].w _VBSoft.addValueEx(NL3D_VEGETABLE_VPPOS_BENDINFO, CVertexBuffer::Float4); // v[9] } _VBSoft.addValueEx(NL3D_VEGETABLE_VPPOS_COLOR0, CVertexBuffer::UChar4); // v[3] _VBSoft.addValueEx(NL3D_VEGETABLE_VPPOS_COLOR1, CVertexBuffer::UChar4); // v[4] _VBSoft.addValueEx(NL3D_VEGETABLE_VPPOS_TEX0, CVertexBuffer::Float2); // v[8] _VBSoft.addValueEx(NL3D_VEGETABLE_VPPOS_CENTER, CVertexBuffer::Float3); // v[10] _VBSoft.initEx(); } } // NL3D