khanat-opennel-code/code/nel/src/3d/meshvp_wind_tree.cpp
2013-09-10 00:31:26 +02:00

428 lines
13 KiB
C++

// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
// 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 <http://www.gnu.org/licenses/>.
#include "std3d.h"
#include "nel/3d/meshvp_wind_tree.h"
#include "nel/3d/mesh_base_instance.h"
#include "nel/3d/scene.h"
#include "nel/3d/driver.h"
#include <cmath>
#include "nel/misc/common.h"
#include "nel/3d/render_trav.h"
using namespace NLMISC;
using namespace std;
namespace NL3D
{
// ***************************************************************************
// Light VP fragment constants start at 24
static const uint VPLightConstantStart= 24;
// ***************************************************************************
NLMISC::CSmartPtr<CVertexProgram> CMeshVPWindTree::_VertexProgram[CMeshVPWindTree::NumVp];
static const char* WindTreeVPCodeWave=
"!!VP1.0 \n\
# extract from color.R the 3 factors into R0.xyz \n\
MAD R0, v[3].x, c[9].x, c[9].yzww; # col.R*3 \n\
MIN R0, R0, c[8].yyyy; # clamp each to 0,1 \n\
MAX R0, R0, c[8].xxxx; \n\
\n\
# Add influence of Bone Level1 \n\
MAD R5, c[15], R0.x, v[0]; \n\
\n\
# Sample LevelPhase into R7.yz: 0 to 3. \n\
MUL R7, v[3].xyzw, c[10].x; \n\
\n\
# Add influence of Bone Level2 \n\
ARL A0.x, R7.y; \n\
MAD R5, c[A0.x+16], R0.y, R5; \n\
\n\
# Add influence of Bone Level3 \n\
ARL A0.x, R7.z; \n\
MAD R5, c[A0.x+20], R0.z, R5; \n\
\n\
# Get normal in R6 for lighting. \n\
MOV R6, v[2]; \n\
";
static const char* WindTreeVPCodeEnd=
" # compute in Projection space \n\
DP4 o[HPOS].x, c[0], R5; \n\
DP4 o[HPOS].y, c[1], R5; \n\
DP4 o[HPOS].z, c[2], R5; \n\
DP4 o[HPOS].w, c[3], R5; \n\
MOV o[TEX0], v[8]; \n\
# hulud : remove this line for the moment because it doesn't work under d3d, if it is needed, we will have to create 2 CVertexProgram objects.\n\
#MOV o[TEX1], v[9]; \n\
DP4 o[FOGC].x, c[6], R5; \n\
END \n\
";
// ***************************************************************************
float CMeshVPWindTree::speedCos(float angle)
{
// \todo yoyo TODO_OPTIM
return cosf(angle * 2*(float)Pi);
}
// ***************************************************************************
CMeshVPWindTree::CMeshVPWindTree()
{
for(uint i=0; i<HrcDepth; i++)
{
Frequency[i]= 1;
FrequencyWindFactor[i]= 0;
PowerXY[i]= 0;
PowerZ[i]= 0;
Bias[i]= 0;
// Init currentTime.
_CurrentTime[i]= 0;
}
SpecularLighting= false;
_LastSceneTime= 0;
_MaxVertexMove= 0;
}
// ***************************************************************************
CMeshVPWindTree::~CMeshVPWindTree()
{
}
// ***************************************************************************
void CMeshVPWindTree::serial(NLMISC::IStream &f) throw(NLMISC::EStream)
{
(void)f.serialVersion(0);
nlassert(HrcDepth==3);
for(uint i=0; i<HrcDepth; i++)
{
f.serial(Frequency[i]);
f.serial(FrequencyWindFactor[i]);
f.serial(PowerXY[i]);
f.serial(PowerZ[i]);
f.serial(Bias[i]);
}
f.serial(SpecularLighting);
}
// ***************************************************************************
void CMeshVPWindTree::initInstance(CMeshBaseInstance *mbi)
{
// init the vertexProgram code.
static bool vpCreated= false;
if(!vpCreated)
{
vpCreated= true;
// All vpcode and begin() written for HrcDepth==3
nlassert(HrcDepth==3);
// combine fragments.
string vpCode;
// For all possible VP.
for(uint i=0;i<NumVp;i++)
{
// setup of the VPLight fragment
uint numPls= i/4;
bool normalize= (i&1)!=0;
bool specular= (i&2)!=0;
// combine fragments
vpCode= string(WindTreeVPCodeWave)
+ CRenderTrav::getLightVPFragment(numPls, VPLightConstantStart, specular, normalize)
+ WindTreeVPCodeEnd;
_VertexProgram[i] = new CVertexProgram(vpCode.c_str());
// TODO_VP_GLSL
}
}
// init a random phase.
mbi->_VPWindTreePhase= frand(1);
}
// ***************************************************************************
inline void CMeshVPWindTree::setupPerMesh(IDriver *driver, CScene *scene)
{
// process current times and current power. Only one time per render() and per CMeshVPWindTree.
if(scene->getCurrentTime() != _LastSceneTime)
{
// Get info from scene
float windPower= scene->getGlobalWindPower();
float dt= (float)(scene->getCurrentTime() - _LastSceneTime);
_LastSceneTime= scene->getCurrentTime();
// Update each boneLevel time according to frequency.
uint i;
for(i=0; i<HrcDepth; i++)
{
_CurrentTime[i]+= dt*(Frequency[i] + FrequencyWindFactor[i]*windPower);
// get it between 0 and 1. Important for float precision problems.
_CurrentTime[i]= (float)fmod(_CurrentTime[i], 1);
}
// Update each boneLevel maximum amplitude vector.
for(i=0; i<HrcDepth; i++)
{
_MaxDeltaPos[i]= scene->getGlobalWindDirection() * PowerXY[i] * windPower;
_MaxDeltaPos[i].z= PowerZ[i] * windPower;
}
/* Update the Max amplitude distance
in world space, since maxdeltaPos are applied in world space, see setupPerInstanceConstants()
*/
_MaxVertexMove= 0;
for(i=0; i<HrcDepth; i++)
{
_MaxVertexMove+= _MaxDeltaPos[i].norm();
}
}
// Setup common constants for each instances.
// c[8] take useful constants.
static float ct8[4]= {0, 1, 0.5f, 2};
driver->setConstant(8, 1, ct8);
// c[9] take other useful constants.
static float ct9[4]= {3.f, 0.f, -1.f, -2.f};
driver->setConstant(9, 1, ct9);
// c[10] take Number of phase (4) for level2 and 3. -0.01 to avoid int value == 4.
static float ct10[4]= {4-0.01f, 0, 0, 0};
driver->setConstant(10, 1, ct10);
}
// ***************************************************************************
inline void CMeshVPWindTree::setupPerInstanceConstants(IDriver *driver, CScene *scene, CMeshBaseInstance *mbi, const NLMISC::CMatrix &invertedModelMat)
{
// get instance info
float instancePhase= mbi->_VPWindTreePhase;
// maxDeltaPos in ObjectSpace. So same world Wind direction is applied to all objects
static CMatrix invWorldMatrix;
// Keep only rotation part. (just need it and faster invert)
invWorldMatrix.setRot(mbi->getWorldMatrix());
invWorldMatrix.invert();
static CVector maxDeltaPosOS[HrcDepth];
for(uint i=0; i<HrcDepth; i++)
{
maxDeltaPosOS[i]= invWorldMatrix.mulVector(_MaxDeltaPos[i]);
}
// Setup lighting and lighting constants
setupLighting(scene, mbi, invertedModelMat);
// c[0..3] take the ModelViewProjection Matrix. After setupModelMatrix();
driver->setConstantMatrix(0, IDriver::ModelViewProjection, IDriver::Identity);
// c[4..7] take the ModelView Matrix. After setupModelMatrix();00
driver->setConstantFog(6);
// c[15] take Wind of level 0.
float f;
f= _CurrentTime[0] + instancePhase;
f= speedCos(f) + Bias[0];
driver->setConstant(15, maxDeltaPosOS[0]*f );
// c[16-19] take Wind of level 1.
// Unrolled.
float instTime1= _CurrentTime[1] + instancePhase;
// phase 0.
f= speedCos( instTime1+0 ) + Bias[1];
driver->setConstant(16+0, maxDeltaPosOS[1]*f);
// phase 1.
f= speedCos( instTime1+0.25f ) + Bias[1];
driver->setConstant(16+1, maxDeltaPosOS[1]*f);
// phase 2.
f= speedCos( instTime1+0.50f ) + Bias[1];
driver->setConstant(16+2, maxDeltaPosOS[1]*f);
// phase 3.
f= speedCos( instTime1+0.75f ) + Bias[1];
driver->setConstant(16+3, maxDeltaPosOS[1]*f);
// c[20, 23] take Wind of level 2.
// Unrolled.
float instTime2= _CurrentTime[2] + instancePhase;
// phase 0.
f= speedCos( instTime2+0 ) + Bias[2];
driver->setConstant(20+0, maxDeltaPosOS[2]*f);
// phase 1.
f= speedCos( instTime2+0.25f ) + Bias[2];
driver->setConstant(20+1, maxDeltaPosOS[2]*f);
// phase 2.
f= speedCos( instTime2+0.50f ) + Bias[2];
driver->setConstant(20+2, maxDeltaPosOS[2]*f);
// phase 3.
f= speedCos( instTime2+0.75f ) + Bias[2];
driver->setConstant(20+3, maxDeltaPosOS[2]*f);
}
// ***************************************************************************
bool CMeshVPWindTree::begin(IDriver *driver, CScene *scene, CMeshBaseInstance *mbi, const NLMISC::CMatrix &invertedModelMat, const NLMISC::CVector & /*viewerPos*/)
{
if (!(driver->supportVertexProgram() && !driver->isVertexProgramEmulated())) return false;
// precompute mesh
setupPerMesh(driver, scene);
// Setup instance constants
setupPerInstanceConstants(driver, scene, mbi, invertedModelMat);
// Activate the good VertexProgram
//===============
// Get how many pointLights are setuped now.
nlassert(scene != NULL);
CRenderTrav *renderTrav= &scene->getRenderTrav();
sint numPls= renderTrav->getNumVPLights()-1;
clamp(numPls, 0, CRenderTrav::MaxVPLight-1);
// Enable normalize only if requested by user. Because lighting don't manage correct "scale lighting"
uint idVP= (SpecularLighting?2:0) + (driver->isForceNormalize()?1:0) ;
// correct VP id for correct unmber of pls.
idVP= numPls*4 + idVP;
// activate VP.
driver->activeVertexProgram(_VertexProgram[idVP]);
return true;
}
// ***************************************************************************
void CMeshVPWindTree::end(IDriver *driver)
{
// Disable the VertexProgram
driver->activeVertexProgram(NULL);
}
// ***************************************************************************
// tool fct
static inline void SetupForMaterial(const CMaterial &mat, CScene *scene)
{
CRenderTrav *renderTrav= &scene->getRenderTrav();
renderTrav->changeVPLightSetupMaterial(mat, false /* don't exclude strongest */);
}
// ***************************************************************************
void CMeshVPWindTree::setupForMaterial(const CMaterial &mat,
IDriver *drv,
CScene *scene,
CVertexBuffer *)
{
SetupForMaterial(mat, scene);
}
// ***************************************************************************
void CMeshVPWindTree::setupLighting(CScene *scene, CMeshBaseInstance *mbi, const NLMISC::CMatrix &invertedModelMat)
{
nlassert(scene != NULL);
CRenderTrav *renderTrav= &scene->getRenderTrav();
// setup cte for lighting
renderTrav->beginVPLightSetup(VPLightConstantStart, SpecularLighting, invertedModelMat);
}
// ***************************************************************************
// ***************************************************************************
// MBR interface
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
bool CMeshVPWindTree::supportMeshBlockRendering() const
{
return true;
}
// ***************************************************************************
bool CMeshVPWindTree::isMBRVpOk(IDriver *driver) const
{
return driver->supportVertexProgram() && !driver->isVertexProgramEmulated();
}
// ***************************************************************************
void CMeshVPWindTree::beginMBRMesh(IDriver *driver, CScene *scene)
{
// precompute mesh
setupPerMesh(driver, scene);
/* Since need a VertexProgram Activation before activeVBHard, activate a default one
bet the common one will be "NoPointLight, NoSpecular, No ForceNormalize" => 0.
*/
_LastMBRIdVP= 0;
// activate VP.
driver->activeVertexProgram(_VertexProgram[_LastMBRIdVP]);
}
// ***************************************************************************
void CMeshVPWindTree::beginMBRInstance(IDriver *driver, CScene *scene, CMeshBaseInstance *mbi, const NLMISC::CMatrix &invertedModelMat)
{
// setup first constants for this instance
setupPerInstanceConstants(driver, scene, mbi, invertedModelMat);
// Get how many pointLights are setuped now.
nlassert(scene != NULL);
CRenderTrav *renderTrav= &scene->getRenderTrav();
sint numPls= renderTrav->getNumVPLights()-1;
clamp(numPls, 0, CRenderTrav::MaxVPLight-1);
// Enable normalize only if requested by user. Because lighting don't manage correct "scale lighting"
uint idVP= (SpecularLighting?2:0) + (driver->isForceNormalize()?1:0) ;
// correct VP id for correct number of pls.
idVP= numPls*4 + idVP;
// re-activate VP if idVP different from last setup
if( idVP!=_LastMBRIdVP )
{
_LastMBRIdVP= idVP;
driver->activeVertexProgram(_VertexProgram[_LastMBRIdVP]);
}
}
// ***************************************************************************
void CMeshVPWindTree::endMBRMesh(IDriver *driver)
{
// Disable the VertexProgram
driver->activeVertexProgram(NULL);
}
// ***************************************************************************
float CMeshVPWindTree::getMaxVertexMove()
{
return _MaxVertexMove;
}
} // NL3D