// 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/vertex_buffer.h"
#include "nel/misc/vector.h"
#include "nel/misc/fast_mem.h"
#include "nel/3d/driver.h"
using namespace NLMISC;
#ifdef DEBUG_NEW
#define new DEBUG_NEW
#endif
namespace NL3D
{
// --------------------------------------------------
const uint CVertexBuffer::SizeType[NumType]=
{
1*sizeof(double),
1*sizeof(float),
1*sizeof(short),
2*sizeof(double),
2*sizeof(float),
2*sizeof(short),
3*sizeof(double),
3*sizeof(float),
3*sizeof(short),
4*sizeof(double),
4*sizeof(float),
4*sizeof(short),
4*sizeof(char),
};
const uint CVertexBuffer::NumComponentsType[NumType] =
{
1,
1,
1,
2,
2,
2,
3,
3,
3,
4,
4,
4,
4
};
// --------------------------------------------------
const CVertexBuffer::TType CVertexBuffer::DefaultValueType[NumValue]=
{
Float3, // Position
Float3, // Normal
Float2, // TexCoord0
Float2, // TexCoord1
Float2, // TexCoord2
Float2, // TexCoord3
Float2, // TexCoord4
Float2, // TexCoord5
Float2, // TexCoord6
Float2, // TexCoord7
UChar4, // Primary color
UChar4, // Secondary color
Float4, // 4 Weights
UChar4, // PaletteSkin
Float1, // Fog
Float1, // Empty
};
// --------------------------------------------------
void CVertexBuffer::construct()
{
/* ***********************************************
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
* It can be loaded/called through CAsyncFileManager for instance
* ***********************************************/
_Flags = 0;
_Capacity = 0;
_NbVerts = 0;
_InternalFlags = 0;
_VertexSize = 0;
_VertexColorFormat = TRGBA;
_LockCounter = 0;
_LockedBuffer = NULL;
_PreferredMemory = RAMPreferred;
_Location = NotResident;
_ResidentSize = 0;
_KeepLocalMemory = false;
// Default routing
uint i;
for (i=0; iVertexBufferPtr = NULL; // Tell the driver info to not restore memory when it will die
// Must kill the drv mirror of this VB.
DrvInfos.kill();
}
// --------------------------------------------------
CVertexBuffer &CVertexBuffer::operator=(const CVertexBuffer &vb)
{
nlassertex (!isLocked(), ("The vertex buffer is locked."));
nlassertex (!vb.isLocked(), ("Source buffer is locked."));
// Single value
_VertexSize = vb._VertexSize;
_Flags = vb._Flags;
_InternalFlags = vb._InternalFlags;
_NbVerts = vb._NbVerts;
_Capacity = vb._Capacity;
_NonResidentVertices = vb._NonResidentVertices;
_VertexColorFormat = vb._VertexColorFormat;
_PreferredMemory = vb._PreferredMemory;
_KeepLocalMemory = vb._KeepLocalMemory;
uint i;
_LockCounter = 0;
_LockedBuffer = NULL;
// Arraies
for (uint value=0; value(this)->lock(srcDatas);
nlassert(dest.getLocation() == NotResident);
CVertexBufferReadWrite destDatas;
dest.lock(destDatas); // will be in vram
NLMISC::CFastMem::memcpy (destDatas.getVertexCoordPointer(), srcDatas.getVertexCoordPointer(), getVertexSize() * getNumVertices());
}
// --------------------------------------------------
bool CVertexBuffer::setVertexFormat(uint32 flags)
{
nlassertex (!isLocked(), ("The vertex buffer is locked."));
uint i;
// Clear extended values
clearValueEx ();
// Position ?
if (flags & PositionFlag)
{
// Add a standard position value
addValueEx (Position, Float3);
}
// Normal ?
if (flags & NormalFlag)
{
// Add a standard normal value
addValueEx (Normal, Float3);
}
// For each uv values
for(i=0 ; i= 2); break;
case Normal: nlassert(numComp == 3); break;
case PrimaryColor: nlassert(numComp == 4); break;
case SecondaryColor: nlassert(numComp == 4); break;
case Weight: nlassert(numComp == 4); break;
case PaletteSkin: nlassert(numComp == 4); break;
case Fog: nlassert(numComp == 4); break;
default: break;
}
}
// --------------------------------------------------
bool CVertexBuffer::hasValueEx(TValue valueId) const
{
return (_Flags & (1 << valueId)) != 0;
}
// --------------------------------------------------
void CVertexBuffer::initEx ()
{
nlassert (!isLocked());
// Calc vertex size and set value's offset
_VertexSize=0;
for (uint value=0; value=1) && ((_Flags & PaletteSkinFlag) == CVertexBuffer::PaletteSkinFlag) )
{
CPaletteSkin &ps= *(CPaletteSkin*)(pointer + stridedId + _Offset[PaletteSkin]);
f.serial(ps);
}
}
// Set touch flags
_InternalFlags = 0;
if(f.isReading())
{
// Force non resident
restoreNonResidentMemory();
}
}
// --------------------------------------------------
void CVertexBuffer::serial(NLMISC::IStream &f)
{
/* ***********************************************
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
* It can be loaded/called through CAsyncFileManager for instance
* ***********************************************/
/*
Version 2:
- cut to use serialHeader() serialSubset().
Version 1:
- PaletteSkin version.
Version 0:
- base verison.
*/
nlassert (!isLocked());
sint ver= f.serialVersion(2);
if (ver<2)
{
// old serial method
serialOldV1Minus(f, ver);
}
else
{
// read write the header of the VBuffer.
serialHeader(f);
// read write the entire subset.
serialSubset(f, 0, _NbVerts);
}
}
// --------------------------------------------------
void CVertexBuffer::serialHeader(NLMISC::IStream &f)
{
/* ***********************************************
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
* It can be loaded/called through CAsyncFileManager for instance
* ***********************************************/
/*
Version 3:
- Preferred memory.
Version 2:
- Vertex color format management.
Version 1:
- Extended vertex format management.
Version 0:
- base verison of the header serialisation.
*/
sint ver= f.serialVersion(3); // Hulud
// Serial VBuffers format/size.
//=============================
// Flags
uint16 flags=_Flags;
if (ver<1)
{
// Must be reading
nlassert (f.isReading());
// Serial old flags
uint32 oldFlags;
f.serial(oldFlags);
// Remap flags
uint weightCount;
flags=remapV2Flags (oldFlags, weightCount);
// Set default value type
for (uint i=0; i=2)
f.serial (_VertexColorFormat);
if (ver>=3)
{
f.serialEnum(_PreferredMemory);
f.serial(_Name);
}
else
{
// Init preferred memory
if(f.isReading())
{
_PreferredMemory = RAMPreferred;
_Name.clear();
}
}
}
// --------------------------------------------------
uint CVertexBuffer:: getNumTexCoordUsed() const
{
for (sint k = (MaxStage - 1); k >= 0; --k)
{
if (_Flags & (TexCoord0Flag << k)) return (uint) (k + 1);
}
return 0;
}
// --------------------------------------------------
uint8 CVertexBuffer::getNumWeight () const
{
// Num weight
switch (_Type[Weight])
{
case Float1:
return 1;
case Float2:
return 2;
case Float3:
return 3;
case Float4:
return 4;
}
// No weight
return 0;
}
// --------------------------------------------------
void CVertexBuffer::serialSubset(NLMISC::IStream &f, uint vertexStart, uint vertexEnd)
{
/* ***********************************************
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
* It can be loaded/called through CAsyncFileManager for instance
* ***********************************************/
/*
Version 2:
- UVRouting
Version 1:
- weight is 4 float in standard format.
Version 0:
- base verison of a vbuffer subset serialisation.
*/
sint ver = f.serialVersion(2);
// Serial VBuffers components.
//============================
nlassert(vertexStart<_NbVerts || _NbVerts==0);
nlassert(vertexEnd<=_NbVerts);
for(uint id=vertexStart; id=2)
{
f.serialBuffer (_UVRouting, sizeof(uint8)*MaxStage);
}
else
{
// Reset the table
uint i;
for (i=0; i>16);
#else // NL_LITTLE_ENDIAN
*ptr0 = (value&0x00ff00ff)|((value&0xff00)<<16)|((value&0xff000000)>>16);
#endif // NL_LITTLE_ENDIAN
ptr0 = (uint32*)(((uint8*)ptr0)+_VertexSize);
}
if (ptr1)
{
const register uint32 value = *ptr1;
#ifdef NL_LITTLE_ENDIAN
*ptr1 = (value&0xff00ff00)|((value&0xff)<<16)|((value&0xff0000)>>16);
#else // NL_LITTLE_ENDIAN
*ptr1 = (value&0x00ff00ff)|((value&0xff00)<<16)|((value&0xff000000)>>16);
#endif // NL_LITTLE_ENDIAN
ptr1 = (uint32*)(((uint8*)ptr1)+_VertexSize);
}
}
}
_VertexColorFormat = (uint8)format;
// Force non resident
restoreNonResidentMemory();
}
return true;
}
// --------------------------------------------------
void CVertexBuffer::setPreferredMemory (TPreferredMemory preferredMemory, bool keepLocalMemory)
{
if ((_PreferredMemory != preferredMemory) || (_KeepLocalMemory != keepLocalMemory))
{
_PreferredMemory = preferredMemory;
_KeepLocalMemory = keepLocalMemory;
// Force non resident
restoreNonResidentMemory();
}
}
// --------------------------------------------------
void CVertexBuffer::setLocation (TLocation newLocation)
{
// Upload ?
if (newLocation != NotResident)
{
// The driver must have setuped the driver info
nlassert (DrvInfos);
// Current size of the buffer
const uint size = ((_PreferredMemory==RAMVolatile)||(_PreferredMemory==AGPVolatile))?_NbVerts*_VertexSize:_Capacity*_VertexSize;
// The buffer must not be resident
if (_Location != NotResident)
setLocation (NotResident);
// Copy the buffer containt
uint8 *dest = DrvInfos->lock (0, size, false);
nlassert (size<=_NonResidentVertices.size()); // Internal buffer must have the good size
memcpy (dest, &(_NonResidentVertices[0]), size);
DrvInfos->unlock(0, 0);
// Reset the non resident container if not a static preferred memory and not put in RAM
if ((_PreferredMemory != StaticPreferred) && (_Location != RAMResident) && !_KeepLocalMemory)
contReset(_NonResidentVertices);
// Clear touched flags
resetTouchFlags ();
_Location = newLocation;
_ResidentSize = size;
}
else
{
// Current size of the buffer
const uint size = _Capacity*_VertexSize;
// Resize the non resident buffer
_NonResidentVertices.resize (size);
// If resident in RAM, backup the data in non resident memory
if ((_Location == RAMResident) && (_PreferredMemory != RAMVolatile) && (_PreferredMemory != AGPVolatile) && !_KeepLocalMemory)
{
// The driver must have setuped the driver info
nlassert (DrvInfos);
// Copy the old buffer data
const uint8 *src = DrvInfos->lock (0, _ResidentSize, true);
if (!_NonResidentVertices.empty())
memcpy (&(_NonResidentVertices[0]), src, std::min (size, (uint)_ResidentSize));
DrvInfos->unlock(0, 0);
}
_Location = NotResident;
_ResidentSize = 0;
// Touch the buffer
_InternalFlags |= TouchedAll;
}
}
// --------------------------------------------------
void CVertexBuffer::restoreNonResidentMemory()
{
setLocation (NotResident);
if (DrvInfos)
DrvInfos->VertexBufferPtr = NULL; // Tell the driver info to not restore memory when it will die
// Must kill the drv mirror of this VB.
DrvInfos.kill();
}
// --------------------------------------------------
void CVertexBuffer::fillBuffer ()
{
if (DrvInfos && _KeepLocalMemory)
{
// Copy the local memory in local memory
const uint size = _NbVerts*_VertexSize;
nlassert (size<=_NonResidentVertices.size());
uint8 *dest = DrvInfos->lock (0, size, false);
NLMISC::CFastMem::memcpy (dest, &(_NonResidentVertices[0]), size);
DrvInfos->unlock(0, size);
}
}
// --------------------------------------------------
// CPaletteSkin serial (no version chek).
void CPaletteSkin::serial(NLMISC::IStream &f)
{
f.serial(MatrixId[0], MatrixId[1], MatrixId[2], MatrixId[3]);
}
// --------------------------------------------------
IVBDrvInfos::~IVBDrvInfos()
{
_Driver->removeVBDrvInfoPtr(_DriverIterator);
}
// --------------------------------------------------
// CVertexBufferReadWrite
// --------------------------------------------------
NLMISC::CVector* CVertexBufferReadWrite::getVertexCoordPointer(uint idx)
{
nlassert (_Parent->checkLockedBuffer());
uint8* ptr;
ptr=_Parent->_LockedBuffer;
ptr+=(idx*_Parent->_VertexSize);
return((NLMISC::CVector*)ptr);
}
// --------------------------------------------------
NLMISC::CVector* CVertexBufferReadWrite::getNormalCoordPointer(uint idx)
{
nlassert (_Parent->checkLockedBuffer());
uint8* ptr;
if ( !(_Parent->_Flags & CVertexBuffer::NormalFlag) )
{
return(NULL);
}
ptr=_Parent->_LockedBuffer;
ptr+=_Parent->_Offset[CVertexBuffer::Normal];
ptr+=idx*_Parent->_VertexSize;
return((NLMISC::CVector*)ptr);
}
// --------------------------------------------------
void* CVertexBufferReadWrite::getColorPointer(uint idx)
{
nlassert (_Parent->checkLockedBuffer());
uint8* ptr;
if ( !(_Parent->_Flags & CVertexBuffer::PrimaryColorFlag) )
{
return(NULL);
}
ptr=_Parent->_LockedBuffer;
ptr+=_Parent->_Offset[CVertexBuffer::PrimaryColor];
ptr+=idx*_Parent->_VertexSize;
return((void*)ptr);
}
// --------------------------------------------------
void* CVertexBufferReadWrite::getSpecularPointer(uint idx)
{
nlassert (_Parent->checkLockedBuffer());
uint8* ptr;
if ( !(_Parent->_Flags & CVertexBuffer::SecondaryColorFlag) )
{
return(NULL);
}
ptr=_Parent->_LockedBuffer;
ptr+=_Parent->_Offset[CVertexBuffer::SecondaryColor];
ptr+=idx*_Parent->_VertexSize;
return((void*)ptr);
}
// --------------------------------------------------
NLMISC::CUV* CVertexBufferReadWrite::getTexCoordPointer(uint idx, uint8 stage)
{
nlassert (_Parent->checkLockedBuffer());
uint8* ptr;
if ( !(_Parent->_Flags & (CVertexBuffer::TexCoord0Flag<_LockedBuffer;
ptr+=_Parent->_Offset[CVertexBuffer::TexCoord0+stage];
ptr+=idx*_Parent->_VertexSize;
return((NLMISC::CUV*)ptr);
}
// --------------------------------------------------
float* CVertexBufferReadWrite::getWeightPointer(uint idx, uint8 wgt)
{
nlassert (_Parent->checkLockedBuffer());
uint8* ptr;
nlassert(wgt_Flags & CVertexBuffer::WeightFlag))
return NULL;
ptr=(uint8*)(&_Parent->_LockedBuffer[idx*_Parent->_VertexSize]);
ptr+=_Parent->_Offset[CVertexBuffer::Weight]+wgt*sizeof(float);
return (float*)ptr;
}
// --------------------------------------------------
CPaletteSkin* CVertexBufferReadWrite::getPaletteSkinPointer(uint idx)
{
nlassert (_Parent->checkLockedBuffer());
uint8* ptr;
if ( (_Parent->_Flags & CVertexBuffer::PaletteSkinFlag) != CVertexBuffer::PaletteSkinFlag )
{
return(NULL);
}
ptr=_Parent->_LockedBuffer;
ptr+=_Parent->_Offset[CVertexBuffer::PaletteSkin];
ptr+=idx*_Parent->_VertexSize;
return((CPaletteSkin*)ptr);
}
// --------------------------------------------------
void CVertexBufferReadWrite::touchVertices (uint first, uint last)
{
nlassert (_Parent->checkLockedBuffer());
_First = first;
_Last = last;
}
// --------------------------------------------------
// CVertexBufferRead
// --------------------------------------------------
const NLMISC::CVector* CVertexBufferRead::getVertexCoordPointer(uint idx) const
{
nlassert (_Parent->checkLockedBuffer());
const uint8* ptr;
ptr=_Parent->_LockedBuffer;
ptr+=(idx*_Parent->_VertexSize);
return((const NLMISC::CVector*)ptr);
}
// --------------------------------------------------
const NLMISC::CVector* CVertexBufferRead::getNormalCoordPointer(uint idx) const
{
nlassert (_Parent->checkLockedBuffer());
const uint8* ptr;
if ( !(_Parent->_Flags & CVertexBuffer::NormalFlag) )
{
return(NULL);
}
ptr=_Parent->_LockedBuffer;
ptr+=_Parent->_Offset[CVertexBuffer::Normal];
ptr+=idx*_Parent->_VertexSize;
return((const NLMISC::CVector*)ptr);
}
// --------------------------------------------------
const void* CVertexBufferRead::getColorPointer(uint idx) const
{
nlassert (_Parent->checkLockedBuffer());
const uint8* ptr;
if ( !(_Parent->_Flags & CVertexBuffer::PrimaryColorFlag) )
{
return(NULL);
}
ptr=_Parent->_LockedBuffer;
ptr+=_Parent->_Offset[CVertexBuffer::PrimaryColor];
ptr+=idx*_Parent->_VertexSize;
return((const void*)ptr);
}
// --------------------------------------------------
const void* CVertexBufferRead::getSpecularPointer(uint idx) const
{
nlassert (_Parent->checkLockedBuffer());
const uint8* ptr;
if ( !(_Parent->_Flags & CVertexBuffer::SecondaryColorFlag) )
{
return(NULL);
}
ptr=_Parent->_LockedBuffer;
ptr+=_Parent->_Offset[CVertexBuffer::SecondaryColor];
ptr+=idx*_Parent->_VertexSize;
return((const void*)ptr);
}
// --------------------------------------------------
const NLMISC::CUV* CVertexBufferRead::getTexCoordPointer(uint idx, uint8 stage) const
{
nlassert (_Parent->checkLockedBuffer());
const uint8* ptr;
if ( !(_Parent->_Flags & (CVertexBuffer::TexCoord0Flag<_LockedBuffer;
ptr+=_Parent->_Offset[CVertexBuffer::TexCoord0+stage];
ptr+=idx*_Parent->_VertexSize;
return((const NLMISC::CUV*)ptr);
}
// --------------------------------------------------
const float* CVertexBufferRead::getWeightPointer(uint idx, uint8 wgt) const
{
nlassert (_Parent->checkLockedBuffer());
const uint8* ptr;
nlassert(wgt_Flags & CVertexBuffer::WeightFlag))
return NULL;
ptr=(uint8*)(&_Parent->_LockedBuffer[idx*_Parent->_VertexSize]);
ptr+=_Parent->_Offset[CVertexBuffer::Weight]+wgt*sizeof(float);
return (float*)ptr;
}
// --------------------------------------------------
const CPaletteSkin* CVertexBufferRead::getPaletteSkinPointer(uint idx) const
{
nlassert (_Parent->checkLockedBuffer());
const uint8* ptr;
if ( (_Parent->_Flags & CVertexBuffer::PaletteSkinFlag) != CVertexBuffer::PaletteSkinFlag )
{
return(NULL);
}
ptr=_Parent->_LockedBuffer;
ptr+=_Parent->_Offset[CVertexBuffer::PaletteSkin];
ptr+=idx*_Parent->_VertexSize;
return((const CPaletteSkin*)ptr);
}
// --------------------------------------------------
}