2014-06-29 02:42:36 +02:00
# include "terraingrid.hpp"
2015-06-03 01:18:36 +02:00
# include <memory>
2016-02-10 15:34:06 +01:00
# include <osg/UserDataContainer>
2016-02-19 01:45:28 +01:00
# include <osg/Material>
2016-02-10 15:34:06 +01:00
2016-02-09 20:57:30 +01:00
# include <OpenThreads/ScopedLock>
2015-06-03 01:18:36 +02:00
# include <components/resource/resourcesystem.hpp>
2016-02-05 23:03:53 +01:00
# include <components/resource/imagemanager.hpp>
2016-02-05 22:58:02 +01:00
# include <components/resource/scenemanager.hpp>
2015-06-03 01:18:36 +02:00
# include <components/sceneutil/lightmanager.hpp>
2015-11-20 21:57:04 +01:00
# include <components/sceneutil/positionattitudetransform.hpp>
2016-02-09 15:30:53 +01:00
# include <components/sceneutil/unrefqueue.hpp>
2015-06-03 01:18:36 +02:00
2015-11-06 15:23:37 +01:00
# include <components/esm/loadland.hpp>
2015-06-03 01:18:36 +02:00
# include <osg/Geometry>
# include <osg/Geode>
2015-06-09 02:29:56 +02:00
# include <osg/KdTree>
2015-06-03 01:18:36 +02:00
# include <osgFX/Effect>
# include <osgUtil/IncrementalCompileOperation>
2014-06-29 02:42:36 +02:00
2015-06-03 01:18:36 +02:00
# include "material.hpp"
# include "storage.hpp"
namespace
{
class StaticBoundingBoxCallback : public osg : : Drawable : : ComputeBoundingBoxCallback
{
public :
StaticBoundingBoxCallback ( const osg : : BoundingBox & bounds )
: mBoundingBox ( bounds )
{
}
virtual osg : : BoundingBox computeBound ( const osg : : Drawable & ) const
{
return mBoundingBox ;
}
private :
osg : : BoundingBox mBoundingBox ;
} ;
}
2014-06-29 02:42:36 +02:00
2014-08-07 20:43:33 +02:00
namespace Terrain
2014-06-29 02:42:36 +02:00
{
2016-02-09 15:30:53 +01:00
TerrainGrid : : TerrainGrid ( osg : : Group * parent , Resource : : ResourceSystem * resourceSystem , osgUtil : : IncrementalCompileOperation * ico , Storage * storage , int nodeMask , SceneUtil : : UnrefQueue * unrefQueue )
2015-06-03 01:18:36 +02:00
: Terrain : : World ( parent , resourceSystem , ico , storage , nodeMask )
2015-11-06 20:14:57 +01:00
, mNumSplits ( 4 )
2016-02-09 20:57:30 +01:00
, mCache ( ( storage - > getCellVertices ( ) - 1 ) / static_cast < float > ( mNumSplits ) + 1 )
2016-02-09 15:30:53 +01:00
, mUnrefQueue ( unrefQueue )
2014-06-29 02:42:36 +02:00
{
2016-02-19 01:45:28 +01:00
osg : : ref_ptr < osg : : Material > material ( new osg : : Material ) ;
material - > setColorMode ( osg : : Material : : AMBIENT_AND_DIFFUSE ) ;
mTerrainRoot - > getOrCreateStateSet ( ) - > setAttributeAndModes ( material , osg : : StateAttribute : : ON ) ;
2014-06-29 02:42:36 +02:00
}
TerrainGrid : : ~ TerrainGrid ( )
{
while ( ! mGrid . empty ( ) )
{
unloadCell ( mGrid . begin ( ) - > first . first , mGrid . begin ( ) - > first . second ) ;
}
}
2016-02-09 20:57:30 +01:00
osg : : ref_ptr < osg : : Node > TerrainGrid : : cacheCell ( int x , int y )
{
{
OpenThreads : : ScopedLock < OpenThreads : : Mutex > lock ( mGridCacheMutex ) ;
Grid : : iterator found = mGridCache . find ( std : : make_pair ( x , y ) ) ;
if ( found ! = mGridCache . end ( ) )
return found - > second ;
}
osg : : ref_ptr < osg : : Node > node = buildTerrain ( NULL , 1.f , osg : : Vec2f ( x + 0.5 , y + 0.5 ) ) ;
OpenThreads : : ScopedLock < OpenThreads : : Mutex > lock ( mGridCacheMutex ) ;
mGridCache . insert ( std : : make_pair ( std : : make_pair ( x , y ) , node ) ) ;
return node ;
}
2015-11-06 20:14:57 +01:00
osg : : ref_ptr < osg : : Node > TerrainGrid : : buildTerrain ( osg : : Group * parent , float chunkSize , const osg : : Vec2f & chunkCenter )
2014-06-29 02:42:36 +02:00
{
2015-11-06 20:14:57 +01:00
if ( chunkSize * mNumSplits > 1.f )
{
// keep splitting
osg : : ref_ptr < osg : : Group > group ( new osg : : Group ) ;
if ( parent )
parent - > addChild ( group ) ;
2015-11-06 20:21:39 +01:00
2015-11-06 20:14:57 +01:00
float newChunkSize = chunkSize / 2.f ;
buildTerrain ( group , newChunkSize , chunkCenter + osg : : Vec2f ( newChunkSize / 2.f , newChunkSize / 2.f ) ) ;
buildTerrain ( group , newChunkSize , chunkCenter + osg : : Vec2f ( newChunkSize / 2.f , - newChunkSize / 2.f ) ) ;
buildTerrain ( group , newChunkSize , chunkCenter + osg : : Vec2f ( - newChunkSize / 2.f , newChunkSize / 2.f ) ) ;
buildTerrain ( group , newChunkSize , chunkCenter + osg : : Vec2f ( - newChunkSize / 2.f , - newChunkSize / 2.f ) ) ;
return group ;
}
else
{
float minH , maxH ;
if ( ! mStorage - > getMinMaxHeights ( chunkSize , chunkCenter , minH , maxH ) )
return NULL ; // no terrain defined
2014-06-29 02:42:36 +02:00
2015-11-06 20:14:57 +01:00
osg : : Vec2f worldCenter = chunkCenter * mStorage - > getCellWorldSize ( ) ;
2015-11-20 21:57:04 +01:00
osg : : ref_ptr < SceneUtil : : PositionAttitudeTransform > transform ( new SceneUtil : : PositionAttitudeTransform ) ;
2015-11-06 20:14:57 +01:00
transform - > setPosition ( osg : : Vec3f ( worldCenter . x ( ) , worldCenter . y ( ) , 0.f ) ) ;
2014-06-29 02:42:36 +02:00
2015-11-06 20:14:57 +01:00
if ( parent )
parent - > addChild ( transform ) ;
2014-06-29 02:42:36 +02:00
2015-11-06 20:14:57 +01:00
osg : : ref_ptr < osg : : Vec3Array > positions ( new osg : : Vec3Array ) ;
osg : : ref_ptr < osg : : Vec3Array > normals ( new osg : : Vec3Array ) ;
osg : : ref_ptr < osg : : Vec4Array > colors ( new osg : : Vec4Array ) ;
2014-06-29 02:42:36 +02:00
2015-11-06 20:14:57 +01:00
osg : : ref_ptr < osg : : VertexBufferObject > vbo ( new osg : : VertexBufferObject ) ;
positions - > setVertexBufferObject ( vbo ) ;
normals - > setVertexBufferObject ( vbo ) ;
colors - > setVertexBufferObject ( vbo ) ;
2015-06-03 02:24:09 +02:00
2015-11-06 20:14:57 +01:00
mStorage - > fillVertexBuffers ( 0 , chunkSize , chunkCenter , positions , normals , colors ) ;
2014-06-29 02:42:36 +02:00
2015-11-06 20:14:57 +01:00
osg : : ref_ptr < osg : : Geometry > geometry ( new osg : : Geometry ) ;
geometry - > setVertexArray ( positions ) ;
geometry - > setNormalArray ( normals , osg : : Array : : BIND_PER_VERTEX ) ;
geometry - > setColorArray ( colors , osg : : Array : : BIND_PER_VERTEX ) ;
geometry - > setUseDisplayList ( false ) ;
geometry - > setUseVertexBufferObjects ( true ) ;
2014-06-29 02:42:36 +02:00
2015-11-06 20:14:57 +01:00
geometry - > addPrimitiveSet ( mCache . getIndexBuffer ( 0 ) ) ;
2014-06-29 02:42:36 +02:00
2015-11-06 20:14:57 +01:00
// we already know the bounding box, so no need to let OSG compute it.
osg : : Vec3f min ( - 0.5f * mStorage - > getCellWorldSize ( ) * chunkSize ,
- 0.5f * mStorage - > getCellWorldSize ( ) * chunkSize ,
minH ) ;
osg : : Vec3f max ( 0.5f * mStorage - > getCellWorldSize ( ) * chunkSize ,
0.5f * mStorage - > getCellWorldSize ( ) * chunkSize ,
maxH ) ;
osg : : BoundingBox bounds ( min , max ) ;
geometry - > setComputeBoundingBoxCallback ( new StaticBoundingBoxCallback ( bounds ) ) ;
2014-06-29 02:42:36 +02:00
2015-11-06 20:14:57 +01:00
std : : vector < LayerInfo > layerList ;
std : : vector < osg : : ref_ptr < osg : : Image > > blendmaps ;
mStorage - > getBlendmaps ( chunkSize , chunkCenter , false , blendmaps , layerList ) ;
2014-06-29 02:42:36 +02:00
2015-11-06 20:14:57 +01:00
// For compiling textures, I don't think the osgFX::Effect does it correctly
osg : : ref_ptr < osg : : Node > textureCompileDummy ( new osg : : Node ) ;
2016-02-05 21:02:02 +01:00
unsigned int dummyTextureCounter = 0 ;
2016-02-05 22:58:02 +01:00
2016-02-19 01:30:15 +01:00
bool useShaders = mResourceSystem - > getSceneManager ( ) - > getForceShaders ( ) ;
if ( ! mResourceSystem - > getSceneManager ( ) - > getClampLighting ( ) )
useShaders = true ; // always use shaders when lighting is unclamped, this is to avoid lighting seams between a terrain chunk with normal maps and one without normal maps
std : : vector < TextureLayer > layers ;
2015-11-06 20:14:57 +01:00
{
2016-02-09 20:57:30 +01:00
OpenThreads : : ScopedLock < OpenThreads : : Mutex > lock ( mTextureCacheMutex ) ;
for ( std : : vector < LayerInfo > : : const_iterator it = layerList . begin ( ) ; it ! = layerList . end ( ) ; + + it )
2016-02-05 22:58:02 +01:00
{
2016-02-19 01:30:15 +01:00
TextureLayer textureLayer ;
2016-02-09 20:57:30 +01:00
osg : : ref_ptr < osg : : Texture2D > texture = mTextureCache [ it - > mDiffuseMap ] ;
if ( ! texture )
{
texture = new osg : : Texture2D ( mResourceSystem - > getImageManager ( ) - > getImage ( it - > mDiffuseMap ) ) ;
texture - > setWrap ( osg : : Texture : : WRAP_S , osg : : Texture : : REPEAT ) ;
texture - > setWrap ( osg : : Texture : : WRAP_T , osg : : Texture : : REPEAT ) ;
mResourceSystem - > getSceneManager ( ) - > applyFilterSettings ( texture ) ;
mTextureCache [ it - > mDiffuseMap ] = texture ;
}
2016-02-19 01:30:15 +01:00
textureLayer . mDiffuseMap = texture ;
textureCompileDummy - > getOrCreateStateSet ( ) - > setTextureAttributeAndModes ( dummyTextureCounter + + , texture ) ;
if ( ! it - > mNormalMap . empty ( ) )
{
texture = mTextureCache [ it - > mNormalMap ] ;
if ( ! texture )
{
texture = new osg : : Texture2D ( mResourceSystem - > getImageManager ( ) - > getImage ( it - > mNormalMap ) ) ;
texture - > setWrap ( osg : : Texture : : WRAP_S , osg : : Texture : : REPEAT ) ;
texture - > setWrap ( osg : : Texture : : WRAP_T , osg : : Texture : : REPEAT ) ;
mResourceSystem - > getSceneManager ( ) - > applyFilterSettings ( texture ) ;
mTextureCache [ it - > mNormalMap ] = texture ;
}
textureCompileDummy - > getOrCreateStateSet ( ) - > setTextureAttributeAndModes ( dummyTextureCounter + + , texture ) ;
textureLayer . mNormalMap = texture ;
useShaders = true ;
}
layers . push_back ( textureLayer ) ;
2016-02-05 22:58:02 +01:00
}
2015-11-06 20:14:57 +01:00
}
2014-06-29 02:42:36 +02:00
2015-11-06 20:14:57 +01:00
std : : vector < osg : : ref_ptr < osg : : Texture2D > > blendmapTextures ;
for ( std : : vector < osg : : ref_ptr < osg : : Image > > : : const_iterator it = blendmaps . begin ( ) ; it ! = blendmaps . end ( ) ; + + it )
{
osg : : ref_ptr < osg : : Texture2D > texture ( new osg : : Texture2D ) ;
texture - > setImage ( * it ) ;
texture - > setWrap ( osg : : Texture : : WRAP_S , osg : : Texture : : CLAMP_TO_EDGE ) ;
texture - > setWrap ( osg : : Texture : : WRAP_T , osg : : Texture : : CLAMP_TO_EDGE ) ;
texture - > setResizeNonPowerOfTwoHint ( false ) ;
blendmapTextures . push_back ( texture ) ;
2016-02-05 21:02:02 +01:00
textureCompileDummy - > getOrCreateStateSet ( ) - > setTextureAttributeAndModes ( dummyTextureCounter + + , blendmapTextures . back ( ) ) ;
2015-11-06 20:14:57 +01:00
}
// use texture coordinates for both texture units, the layer texture and blend texture
for ( unsigned int i = 0 ; i < 2 ; + + i )
geometry - > setTexCoordArray ( i , mCache . getUVBuffer ( ) ) ;
float blendmapScale = ESM : : Land : : LAND_TEXTURE_SIZE * chunkSize ;
2016-02-19 01:30:15 +01:00
osg : : ref_ptr < osgFX : : Effect > effect ( new Terrain : : Effect ( useShaders , mResourceSystem - > getSceneManager ( ) - > getForcePerPixelLighting ( ) , mResourceSystem - > getSceneManager ( ) - > getClampLighting ( ) ,
mResourceSystem - > getSceneManager ( ) - > getShaderManager ( ) , layers , blendmapTextures , blendmapScale , blendmapScale ) ) ;
2015-11-06 20:14:57 +01:00
effect - > addCullCallback ( new SceneUtil : : LightListCallback ) ;
transform - > addChild ( effect ) ;
2015-11-16 15:11:20 +01:00
osg : : Node * toAttach = geometry . get ( ) ;
effect - > addChild ( toAttach ) ;
2015-11-06 20:14:57 +01:00
2015-11-06 20:22:07 +01:00
if ( mIncrementalCompileOperation )
{
2015-11-16 15:11:20 +01:00
mIncrementalCompileOperation - > add ( toAttach ) ;
2015-11-06 20:22:07 +01:00
mIncrementalCompileOperation - > add ( textureCompileDummy ) ;
}
2015-11-06 20:14:57 +01:00
return transform ;
2015-06-03 01:18:36 +02:00
}
2015-11-06 20:14:57 +01:00
}
void TerrainGrid : : loadCell ( int x , int y )
{
if ( mGrid . find ( std : : make_pair ( x , y ) ) ! = mGrid . end ( ) )
return ; // already loaded
2014-06-29 02:42:36 +02:00
2016-02-09 20:57:30 +01:00
// try to get it from the cache
osg : : ref_ptr < osg : : Node > terrainNode ;
{
OpenThreads : : ScopedLock < OpenThreads : : Mutex > lock ( mGridCacheMutex ) ;
Grid : : const_iterator found = mGridCache . find ( std : : make_pair ( x , y ) ) ;
if ( found ! = mGridCache . end ( ) )
{
terrainNode = found - > second ;
if ( ! terrainNode )
return ; // no terrain defined
}
}
2014-06-29 02:42:36 +02:00
2016-02-09 20:57:30 +01:00
// didn't find in cache, build it
2015-11-06 20:14:57 +01:00
if ( ! terrainNode )
2016-02-09 20:57:30 +01:00
{
osg : : Vec2f center ( x + 0.5f , y + 0.5f ) ;
terrainNode = buildTerrain ( NULL , 1.f , center ) ;
if ( ! terrainNode )
return ; // no terrain defined
}
2015-11-06 20:14:57 +01:00
2016-02-09 20:23:53 +01:00
mTerrainRoot - > addChild ( terrainNode ) ;
2014-06-29 02:42:36 +02:00
2016-02-09 20:23:53 +01:00
mGrid [ std : : make_pair ( x , y ) ] = terrainNode ;
2014-06-29 02:42:36 +02:00
}
2015-06-03 01:18:36 +02:00
void TerrainGrid : : unloadCell ( int x , int y )
2014-06-29 02:42:36 +02:00
{
2015-06-03 01:18:36 +02:00
Grid : : iterator it = mGrid . find ( std : : make_pair ( x , y ) ) ;
2014-06-29 02:42:36 +02:00
if ( it = = mGrid . end ( ) )
2015-06-03 01:18:36 +02:00
return ;
2014-06-29 02:42:36 +02:00
2016-02-09 20:57:30 +01:00
osg : : ref_ptr < osg : : Node > terrainNode = it - > second ;
2016-02-09 20:23:53 +01:00
mTerrainRoot - > removeChild ( terrainNode ) ;
2016-02-09 15:30:53 +01:00
if ( mUnrefQueue . get ( ) )
2016-02-09 20:23:53 +01:00
mUnrefQueue - > push ( terrainNode ) ;
2014-06-29 02:42:36 +02:00
2015-06-03 01:18:36 +02:00
mGrid . erase ( it ) ;
2014-06-29 02:42:36 +02:00
}
2016-02-09 20:57:30 +01:00
void TerrainGrid : : updateCache ( )
2016-02-05 22:58:02 +01:00
{
{
2016-02-09 21:17:10 +01:00
OpenThreads : : ScopedLock < OpenThreads : : Mutex > lock ( mGridCacheMutex ) ;
for ( Grid : : iterator it = mGridCache . begin ( ) ; it ! = mGridCache . end ( ) ; )
2016-02-09 20:57:30 +01:00
{
if ( it - > second - > referenceCount ( ) < = 1 )
2016-02-09 21:17:10 +01:00
mGridCache . erase ( it + + ) ;
2016-02-09 20:57:30 +01:00
else
+ + it ;
}
}
{
2016-02-09 21:17:10 +01:00
OpenThreads : : ScopedLock < OpenThreads : : Mutex > lock ( mTextureCacheMutex ) ;
for ( TextureCache : : iterator it = mTextureCache . begin ( ) ; it ! = mTextureCache . end ( ) ; )
2016-02-09 15:30:53 +01:00
{
2016-02-09 20:57:30 +01:00
if ( it - > second - > referenceCount ( ) < = 1 )
2016-02-09 21:17:10 +01:00
mTextureCache . erase ( it + + ) ;
2016-02-09 20:57:30 +01:00
else
+ + it ;
2016-02-09 15:30:53 +01:00
}
2016-02-05 22:58:02 +01:00
}
}
2016-02-14 23:14:52 +01:00
void TerrainGrid : : updateTextureFiltering ( )
{
OpenThreads : : ScopedLock < OpenThreads : : Mutex > lock ( mTextureCacheMutex ) ;
for ( TextureCache : : iterator it = mTextureCache . begin ( ) ; it ! = mTextureCache . end ( ) ; + + it )
mResourceSystem - > getSceneManager ( ) - > applyFilterSettings ( it - > second ) ;
}
2014-06-29 02:42:36 +02:00
}