2633 lines
86 KiB
C++
2633 lines
86 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/>.
|
|
|
|
#ifndef NL_DRIVER_DIRECT3D_H
|
|
#define NL_DRIVER_DIRECT3D_H
|
|
|
|
|
|
#include "nel/misc/types_nl.h"
|
|
|
|
// NeL includes
|
|
#include "nel/misc/matrix.h"
|
|
#include "nel/misc/smart_ptr.h"
|
|
#include "nel/misc/rgba.h"
|
|
#include "nel/misc/event_emitter.h"
|
|
#include "nel/misc/bit_set.h"
|
|
#include "nel/misc/heap_memory.h"
|
|
#include "nel/misc/event_emitter_multi.h"
|
|
#include "nel/misc/time_nl.h"
|
|
#include "nel/misc/hierarchical_timer.h"
|
|
#include "nel/misc/win_event_emitter.h"
|
|
#include "nel/3d/viewport.h"
|
|
#include "nel/3d/scissor.h"
|
|
#include "nel/3d/driver.h"
|
|
#include "nel/3d/material.h"
|
|
#include "nel/3d/shader.h"
|
|
#include "nel/3d/vertex_buffer.h"
|
|
#include "nel/3d/index_buffer.h"
|
|
#include "nel/3d/ptr_set.h"
|
|
#include "nel/3d/texture_cube.h"
|
|
#include "nel/3d/occlusion_query.h"
|
|
#include "nel/3d/vertex_program_parse.h"
|
|
#include "nel/3d/light.h"
|
|
//
|
|
#include <algorithm>
|
|
|
|
typedef HCURSOR nlCursor;
|
|
#define EmptyCursor NULL
|
|
|
|
|
|
// *** DEBUG MACRO
|
|
|
|
// Define this to activate the debug mode (default is undefined)
|
|
//#define NL_DEBUG_D3D
|
|
|
|
// Define this to enable the render state caching (default is defined)
|
|
#define NL_D3D_USE_RENDER_STATE_CACHE
|
|
|
|
// allows to enable / disable cache test at runtime (for debug)
|
|
//#define NL_D3D_RUNTIME_DEBUG_CACHE_TEST
|
|
#ifdef NL_D3D_RUNTIME_DEBUG_CACHE_TEST
|
|
#define NL_D3D_CACHE_TEST(label, cond) if (!_CacheTest[label] || (cond))
|
|
#else
|
|
#define NL_D3D_CACHE_TEST(label, cond) if (cond)
|
|
#endif
|
|
|
|
|
|
// Define this to disable hardware vertex program (default is undefined)
|
|
//#define NL_DISABLE_HARDWARE_VERTEX_PROGAM
|
|
|
|
// Define this to disable hardware pixel shaders program (default is undefined)
|
|
//#define NL_DISABLE_HARDWARE_PIXEL_SHADER
|
|
|
|
// Define this to disable hardware vertex array AGP (default is undefined)
|
|
//#define NL_DISABLE_HARDWARE_VERTEX_ARRAY_AGP
|
|
|
|
// Define this to force the texture stage count (default is undefined)
|
|
//#define NL_FORCE_TEXTURE_STAGE_COUNT 2
|
|
|
|
// Define this to force the use of pixel shader in the normal shaders (default is undefined)
|
|
//#define NL_FORCE_PIXEL_SHADER_USE_FOR_NORMAL_SHADERS
|
|
|
|
// Define this to enable profiling by the NV Perf HUD tool (default is undefined)
|
|
//#define NL_D3D_USE_NV_PERF_HUD
|
|
|
|
// Define this to enable profiling of driver functions (default is undefined).
|
|
//#define NL_PROFILE_DRIVER_D3D
|
|
|
|
|
|
|
|
#ifdef NL_PROFILE_DRIVER_D3D
|
|
#define H_AUTO_D3D(label) H_AUTO(label)
|
|
#else
|
|
#define H_AUTO_D3D(label)
|
|
#endif
|
|
|
|
class CFpuRestorer
|
|
{
|
|
public:
|
|
CFpuRestorer() { _FPControlWord = _controlfp(0, 0); }
|
|
~CFpuRestorer()
|
|
{
|
|
_controlfp(_FPControlWord, ~0);
|
|
}
|
|
private:
|
|
unsigned int _FPControlWord;
|
|
};
|
|
|
|
class CFpuChecker
|
|
{
|
|
public:
|
|
CFpuChecker(const char *label) { _FPControlWord = _controlfp(0, 0); _Label = label; }
|
|
~CFpuChecker()
|
|
{
|
|
unsigned int newFP = _controlfp(0, 0);
|
|
if ((newFP & (_MCW_DN | _MCW_IC | _MCW_RC | _MCW_PC)) != (_FPControlWord & (_MCW_DN | _MCW_IC | _MCW_RC | _MCW_PC)))
|
|
{
|
|
nlwarning(_Label);
|
|
nlassert(0);
|
|
}
|
|
}
|
|
private:
|
|
const char *_Label;
|
|
unsigned int _FPControlWord;
|
|
};
|
|
|
|
|
|
|
|
|
|
inline bool operator==(const D3DCOLORVALUE &lhs, const D3DCOLORVALUE &rhs)
|
|
{
|
|
return lhs.r == rhs.r && lhs.g == rhs.g && lhs.b == rhs.b && lhs.a == rhs.a;
|
|
}
|
|
inline bool operator!=(const D3DCOLORVALUE &lhs, const D3DCOLORVALUE &rhs)
|
|
{
|
|
return !(lhs == rhs);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
|
|
namespace NL3D
|
|
{
|
|
|
|
|
|
|
|
const uint MAX_NUM_QUADS = 32767; // max number of quads in a single draw call
|
|
|
|
using NLMISC::CMatrix;
|
|
using NLMISC::CVector;
|
|
|
|
class CDriverD3D;
|
|
class CTextureDrvInfosD3D;
|
|
class COcclusionQueryD3D;
|
|
class CVolatileVertexBuffer;
|
|
class CVolatileIndexBuffer;
|
|
|
|
typedef std::list<COcclusionQueryD3D *> TOcclusionQueryList;
|
|
|
|
// ***************************************************************************
|
|
class COcclusionQueryD3D : public IOcclusionQuery
|
|
{
|
|
public:
|
|
IDirect3DQuery9 *Query;
|
|
NLMISC::CRefPtr<CDriverD3D> Driver; // owner driver
|
|
TOcclusionQueryList::iterator Iterator; // iterator in owner driver list of queries
|
|
TOcclusionType OcclusionType; // current type of occlusion
|
|
uint VisibleCount; // number of samples that passed the test
|
|
bool QueryIssued;
|
|
bool WasLost; // tells that query was lost, so calls to end() will have not effects (there's no matching begin)
|
|
// From IOcclusionQuery
|
|
virtual void begin();
|
|
virtual void end();
|
|
virtual TOcclusionType getOcclusionType();
|
|
virtual uint getVisibleCount();
|
|
};
|
|
|
|
|
|
// ***************************************************************************
|
|
class CTextureDrvInfosD3D : public ITextureDrvInfos
|
|
{
|
|
public:
|
|
/*
|
|
ANY DATA ADDED HERE MUST BE SWAPPED IN swapTextureHandle() !!
|
|
*/
|
|
|
|
// The texture
|
|
LPDIRECT3DBASETEXTURE9 Texture;
|
|
IDirect3DTexture9 *Texture2d;
|
|
IDirect3DCubeTexture9 *TextureCube;
|
|
|
|
// The texture format and size
|
|
D3DFORMAT DestFormat;
|
|
uint Width;
|
|
uint Height;
|
|
|
|
// This is the owner driver.
|
|
CDriverD3D *_Driver;
|
|
|
|
// Is the internal format of the texture is a compressed one?
|
|
bool SrcCompressed;
|
|
bool IsCube;
|
|
|
|
// Is a render target ?
|
|
bool RenderTarget;
|
|
|
|
// Mipmap levels
|
|
uint8 Levels;
|
|
uint8 FirstMipMap;
|
|
|
|
// This is the computed size of what memory this texture take.
|
|
uint32 TextureMemory;
|
|
|
|
// The current wrap modes assigned to the texture.
|
|
D3DTEXTUREADDRESS WrapS;
|
|
D3DTEXTUREADDRESS WrapT;
|
|
D3DTEXTUREFILTERTYPE MagFilter;
|
|
D3DTEXTUREFILTERTYPE MinFilter;
|
|
D3DTEXTUREFILTERTYPE MipFilter;
|
|
|
|
CTextureDrvInfosD3D(IDriver *drv, ItTexDrvInfoPtrMap it, CDriverD3D *drvGl, bool renderTarget);
|
|
~CTextureDrvInfosD3D();
|
|
virtual uint getTextureMemoryUsed() const {return TextureMemory;}
|
|
};
|
|
|
|
|
|
// ***************************************************************************
|
|
class CVertexProgamDrvInfosD3D : public IVertexProgramDrvInfos
|
|
{
|
|
public:
|
|
|
|
// The shader
|
|
IDirect3DVertexShader9 *Shader;
|
|
|
|
CVertexProgamDrvInfosD3D(IDriver *drv, ItVtxPrgDrvInfoPtrList it);
|
|
~CVertexProgamDrvInfosD3D();
|
|
};
|
|
|
|
|
|
// ***************************************************************************
|
|
|
|
class CVertexDeclaration
|
|
{
|
|
public:
|
|
// The human readable values
|
|
D3DVERTEXELEMENT9 VertexElements[CVertexBuffer::NumValue+1];
|
|
|
|
// The driver pointer
|
|
IDirect3DVertexDeclaration9 *VertexDecl;
|
|
};
|
|
|
|
|
|
// ***************************************************************************
|
|
class CVBDrvInfosD3D : public IVBDrvInfos
|
|
{
|
|
public:
|
|
IDirect3DVertexDeclaration9 *VertexDecl;
|
|
IDirect3DVertexDeclaration9 *VertexDeclAliasDiffuseToSpecular;
|
|
IDirect3DVertexDeclaration9 *VertexDeclNoDiffuse;
|
|
uint ColorOffset; // Fix for Radeon 7xxx series -> see remarks in CDriverD3D::createVertexDeclaration
|
|
|
|
|
|
IDirect3DVertexBuffer9 *VertexBuffer;
|
|
uint Offset; // Vertex buffer offset
|
|
bool UseVertexColor:1;
|
|
bool Hardware:1;
|
|
bool Volatile:1; // Volatile vertex buffer
|
|
bool VolatileRAM:1;
|
|
uint8 Stride:8;
|
|
uint VolatileLockTime; // Volatile vertex buffer
|
|
DWORD Usage;
|
|
CVolatileVertexBuffer *VolatileVertexBuffer;
|
|
CDriverD3D *Driver;
|
|
#ifdef NL_DEBUG
|
|
bool Locked;
|
|
#endif
|
|
|
|
CVBDrvInfosD3D(CDriverD3D *drv, ItVBDrvInfoPtrList it, CVertexBuffer *vb);
|
|
virtual ~CVBDrvInfosD3D();
|
|
virtual uint8 *lock (uint first, uint last, bool readOnly);
|
|
virtual void unlock (uint first, uint last);
|
|
};
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
|
|
class CIBDrvInfosD3D : public IIBDrvInfos
|
|
{
|
|
public:
|
|
IDirect3DIndexBuffer9 *IndexBuffer;
|
|
uint Offset; // Index buffer offset
|
|
bool Volatile:1; // Volatile index buffer
|
|
bool VolatileRAM:1;
|
|
uint VolatileLockTime; // Volatile index buffer
|
|
CVolatileIndexBuffer *VolatileIndexBuffer;
|
|
CDriverD3D *Driver;
|
|
std::vector<uint32> RamVersion; // If device doesn't support 32 bit indexes, works in ram (unless it a 16 bit index buffer)
|
|
CIBDrvInfosD3D(CDriverD3D *drv, ItIBDrvInfoPtrList it, CIndexBuffer *ib);
|
|
virtual ~CIBDrvInfosD3D();
|
|
virtual void *lock (uint first, uint last, bool readOnly);
|
|
virtual void unlock (uint first, uint last);
|
|
};
|
|
|
|
// ***************************************************************************
|
|
|
|
class CShaderDrvInfosD3D : public IShaderDrvInfos
|
|
{
|
|
public:
|
|
enum
|
|
{
|
|
MaxShaderTexture=8,
|
|
};
|
|
|
|
|
|
ID3DXEffect *Effect;
|
|
bool Validated;
|
|
|
|
// Texture handles
|
|
D3DXHANDLE TextureHandle[MaxShaderTexture];
|
|
|
|
// Color handles
|
|
D3DXHANDLE ColorHandle[MaxShaderTexture];
|
|
|
|
// Factor handles
|
|
D3DXHANDLE FactorHandle[MaxShaderTexture];
|
|
|
|
// Scalar handles
|
|
D3DXHANDLE ScalarFloatHandle[MaxShaderTexture];
|
|
|
|
CShaderDrvInfosD3D(IDriver *drv, ItShaderDrvInfoPtrList it);
|
|
virtual ~CShaderDrvInfosD3D();
|
|
};
|
|
|
|
|
|
// ***************************************************************************
|
|
|
|
// Normal shader description
|
|
class CNormalShaderDesc
|
|
{
|
|
public:
|
|
CNormalShaderDesc ()
|
|
{
|
|
H_AUTO_D3D(CNormalShaderDesc_CNormalShaderDesc);
|
|
memset (this, 0, sizeof(CNormalShaderDesc));
|
|
}
|
|
~CNormalShaderDesc ()
|
|
{
|
|
if (PixelShader)
|
|
PixelShader->Release();
|
|
}
|
|
bool StageUsed[IDRV_MAT_MAXTEXTURES];
|
|
uint32 TexEnvMode[IDRV_MAT_MAXTEXTURES];
|
|
|
|
bool operator==(const CNormalShaderDesc &other) const
|
|
{
|
|
uint i;
|
|
for (i=0; i<IDRV_MAT_MAXTEXTURES; i++)
|
|
if ((StageUsed[i] != other.StageUsed[i]) || (StageUsed[i] && (TexEnvMode[i] != other.TexEnvMode[i])))
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
IDirect3DPixelShader9 *PixelShader;
|
|
};
|
|
|
|
|
|
|
|
// base class for recorded state in an effect
|
|
class CStateRecord
|
|
{
|
|
public:
|
|
// apply record in a driver
|
|
virtual void apply(class CDriverD3D &drv) = 0;
|
|
virtual ~CStateRecord() {}
|
|
// use STL allocator for fast alloc. this works because objects are small ( < 128 bytes)
|
|
void *operator new(size_t size) { return CStateRecord::Allocator.allocate(size); }
|
|
void operator delete(void *block) { CStateRecord::Allocator.deallocate((uint8 *) block, 1); }
|
|
|
|
static std::allocator<uint8> Allocator;
|
|
};
|
|
|
|
|
|
// record of a single .fx pass
|
|
class CFXPassRecord
|
|
{
|
|
public:
|
|
void apply(class CDriverD3D &drv);
|
|
~CFXPassRecord();
|
|
std::vector<CStateRecord *> States;
|
|
};
|
|
|
|
|
|
template <class T>
|
|
class CFXInputValue
|
|
{
|
|
public:
|
|
T Value;
|
|
bool Set;
|
|
CFXInputValue() : Set(false) {}
|
|
void reset();
|
|
bool operator==(const CFXInputValue &other)
|
|
{
|
|
if (!Set) return !(other.Set);
|
|
return (Value == other.Value) != 0;
|
|
}
|
|
};
|
|
|
|
class CFXInputParams
|
|
{
|
|
public:
|
|
enum { MaxNumParams = CShaderDrvInfosD3D::MaxShaderTexture };
|
|
CFXInputValue<LPDIRECT3DBASETEXTURE9> Textures[MaxNumParams];
|
|
CFXInputValue<DWORD> Colors[MaxNumParams];
|
|
CFXInputValue<D3DXVECTOR4> Vectors[MaxNumParams];
|
|
CFXInputValue<FLOAT> Floats[MaxNumParams];
|
|
bool Touched;
|
|
public:
|
|
CFXInputParams() { Touched = true; }
|
|
void setTexture(uint index, LPDIRECT3DBASETEXTURE9 value)
|
|
{
|
|
nlassert(index < MaxNumParams);
|
|
if (!Textures[index].Set || value != Textures[index].Value)
|
|
{
|
|
Textures[index].Value = value;
|
|
Textures[index].Set = true;
|
|
Touched = true;
|
|
}
|
|
}
|
|
void setColor(uint index, DWORD value)
|
|
{
|
|
nlassert(index < MaxNumParams);
|
|
if (!Colors[index].Set || value != Colors[index].Value)
|
|
{
|
|
Colors[index].Value = value;
|
|
Colors[index].Set = true;
|
|
Touched = true;
|
|
}
|
|
}
|
|
void setVector(uint index, const D3DXVECTOR4 &value)
|
|
{
|
|
nlassert(index < MaxNumParams);
|
|
if (!Vectors[index].Set || value != Vectors[index].Value)
|
|
{
|
|
Vectors[index].Value = value;
|
|
Vectors[index].Set = true;
|
|
Touched = true;
|
|
}
|
|
}
|
|
void setFloat(uint index, FLOAT value)
|
|
{
|
|
nlassert(index < MaxNumParams);
|
|
if (!Floats[index].Set || value != Floats[index].Value)
|
|
{
|
|
Floats[index].Value = value;
|
|
Floats[index].Set = true;
|
|
Touched = true;
|
|
}
|
|
}
|
|
//
|
|
/*
|
|
bool operator==(const CFXInputParams &other)
|
|
{
|
|
return std::equal(Textures, Textures + CShaderDrvInfosD3D::MaxShaderTexture, other.Textures) &&
|
|
std::equal(Vectors, Vectors + CShaderDrvInfosD3D::MaxShaderTexture, other.Vectors) &&
|
|
std::equal(Colors, Colors + CShaderDrvInfosD3D::MaxShaderTexture, other.Colors) &&
|
|
std::equal(Floats, Floats + CShaderDrvInfosD3D::MaxShaderTexture, other.Floats);
|
|
}
|
|
*/
|
|
void reset()
|
|
{
|
|
for(uint k = 0; k < MaxNumParams; ++k)
|
|
{
|
|
Textures[k].Set = false;
|
|
Colors[k].Set = false;
|
|
Vectors[k].Set = false;
|
|
Floats[k].Set = false;
|
|
}
|
|
Touched = true;
|
|
}
|
|
};
|
|
|
|
// .fx cache based on input parameters
|
|
class CFXCache
|
|
{
|
|
public:
|
|
// Input parameters
|
|
CFXInputParams Params;
|
|
// cache for .fx states
|
|
std::vector<CFXPassRecord> Passes;
|
|
uint Steadyness;
|
|
uint NumPasses;
|
|
public:
|
|
CFXCache() : Steadyness(0) {}
|
|
void begin(CShaderDrvInfosD3D *si, class CDriverD3D *driver);
|
|
void applyPass(class CDriverD3D &drv, CShaderDrvInfosD3D *si, uint passIndex);
|
|
void end(CShaderDrvInfosD3D *si);
|
|
void reset();
|
|
void setConstants(CShaderDrvInfosD3D *si);
|
|
};
|
|
|
|
|
|
|
|
// optimisation of an effect 2
|
|
class CFXPassRecorder : public ID3DXEffectStateManager
|
|
{
|
|
public:
|
|
CFXPassRecord *Target;
|
|
class CDriverD3D *Driver;
|
|
public:
|
|
CFXPassRecorder() : Target(NULL), Driver(NULL) {}
|
|
HRESULT STDMETHODCALLTYPE QueryInterface(REFIID riid, LPVOID *ppvObj);
|
|
ULONG STDMETHODCALLTYPE AddRef(VOID);
|
|
ULONG STDMETHODCALLTYPE Release(VOID);
|
|
HRESULT STDMETHODCALLTYPE LightEnable(DWORD Index, BOOL Enable);
|
|
HRESULT STDMETHODCALLTYPE SetFVF(DWORD FVF);
|
|
HRESULT STDMETHODCALLTYPE SetLight(DWORD Index, CONST D3DLIGHT9* pLight);
|
|
HRESULT STDMETHODCALLTYPE SetMaterial(CONST D3DMATERIAL9* pMaterial);
|
|
HRESULT STDMETHODCALLTYPE SetNPatchMode(FLOAT nSegments);
|
|
HRESULT STDMETHODCALLTYPE SetPixelShader(LPDIRECT3DPIXELSHADER9 pShader);
|
|
HRESULT STDMETHODCALLTYPE SetPixelShaderConstantB(UINT StartRegister, CONST BOOL* pConstantData, UINT RegisterCount);
|
|
HRESULT STDMETHODCALLTYPE SetPixelShaderConstantF(UINT StartRegister, CONST FLOAT* pConstantData, UINT RegisterCount);
|
|
HRESULT STDMETHODCALLTYPE SetPixelShaderConstantI(UINT StartRegister, CONST INT* pConstantData, UINT RegisterCount);
|
|
HRESULT STDMETHODCALLTYPE SetRenderState(D3DRENDERSTATETYPE State, DWORD Value);
|
|
HRESULT STDMETHODCALLTYPE SetSamplerState(DWORD Sampler, D3DSAMPLERSTATETYPE Type, DWORD Value);
|
|
HRESULT STDMETHODCALLTYPE SetTexture (DWORD Stage, LPDIRECT3DBASETEXTURE9 pTexture);
|
|
HRESULT STDMETHODCALLTYPE SetTextureStageState(DWORD Stage, D3DTEXTURESTAGESTATETYPE Type, DWORD Value);
|
|
HRESULT STDMETHODCALLTYPE SetTransform(D3DTRANSFORMSTATETYPE State, CONST D3DMATRIX* pMatrix);
|
|
HRESULT STDMETHODCALLTYPE SetVertexShader(LPDIRECT3DVERTEXSHADER9 pShader);
|
|
HRESULT STDMETHODCALLTYPE SetVertexShaderConstantB(UINT StartRegister, CONST BOOL* pConstantData, UINT RegisterCount);
|
|
HRESULT STDMETHODCALLTYPE SetVertexShaderConstantF(UINT StartRegister, CONST FLOAT* pConstantData, UINT RegisterCount);
|
|
HRESULT STDMETHODCALLTYPE SetVertexShaderConstantI(UINT StartRegister, CONST INT* pConstantData, UINT RegisterCount);
|
|
};
|
|
|
|
|
|
// ***************************************************************************
|
|
struct CMaterialDrvInfosD3D : public IMaterialDrvInfos
|
|
{
|
|
public:
|
|
D3DMATERIAL9 Material;
|
|
D3DCOLOR UnlightedColor;
|
|
BOOL SpecularEnabled;
|
|
D3DBLEND SrcBlend;
|
|
D3DBLEND DstBlend;
|
|
D3DCMPFUNC ZComp;
|
|
DWORD AlphaRef;
|
|
D3DTEXTUREOP ColorOp[IDRV_MAT_MAXTEXTURES];
|
|
DWORD ColorArg0[IDRV_MAT_MAXTEXTURES];
|
|
DWORD ColorArg1[IDRV_MAT_MAXTEXTURES];
|
|
DWORD ColorArg2[IDRV_MAT_MAXTEXTURES];
|
|
uint NumColorArg[IDRV_MAT_MAXTEXTURES];
|
|
uint NumAlphaArg[IDRV_MAT_MAXTEXTURES];
|
|
D3DTEXTUREOP AlphaOp[IDRV_MAT_MAXTEXTURES];
|
|
DWORD AlphaArg0[IDRV_MAT_MAXTEXTURES];
|
|
DWORD AlphaArg1[IDRV_MAT_MAXTEXTURES];
|
|
DWORD AlphaArg2[IDRV_MAT_MAXTEXTURES];
|
|
D3DCOLOR ConstantColor[IDRV_MAT_MAXTEXTURES];
|
|
DWORD TexGen[IDRV_MAT_MAXTEXTURES];
|
|
IDirect3DPixelShader9 *PixelShader;
|
|
IDirect3DPixelShader9 *PixelShaderUnlightedNoVertexColor;
|
|
bool ActivateSpecularWorldTexMT[IDRV_MAT_MAXTEXTURES];
|
|
bool ActivateInvViewModelTexMT[IDRV_MAT_MAXTEXTURES];
|
|
bool VertexColorLighted;
|
|
bool NeedsConstantForDiffuse; // Must use TFactor if not vertex color in the vertex buffer
|
|
bool MultipleConstantNoPixelShader; // Multiple constant are possibly needed to setup the material. This flag is set only if the device has no pixel shaders
|
|
// In this case diffuse color will be emulated by using an unlighted material with ambient
|
|
bool MultiplePerStageConstant; // Are there more than one per-stage constant in the material ?
|
|
uint8 ConstantIndex; // Index of the constant color to use (when only one constant color is needed and NeedsConstantForDiffuse == false);
|
|
uint8 ConstantIndex2; // stage at which the 2nd constant is used (for emulation without pixel shaders)
|
|
|
|
CRGBA Constant2; // value of the 2nd constant being used (for emulation without pixel shaders)
|
|
|
|
uint NumUsedTexStages; // Last number of textures that were set in the material
|
|
// Tex Env are only built for stages at which textures are set so if the number of used texture
|
|
// change they must be rebuilt
|
|
|
|
// Relevant parts of the pixel pipe for normal shader
|
|
bool RGBPipe[IDRV_MAT_MAXTEXTURES];
|
|
bool AlphaPipe[IDRV_MAT_MAXTEXTURES];
|
|
|
|
CFXCache *FXCache;
|
|
|
|
CMaterialDrvInfosD3D(IDriver *drv, ItMatDrvInfoPtrList it) : IMaterialDrvInfos(drv, it)
|
|
{
|
|
H_AUTO_D3D(CMaterialDrvInfosD3D_CMaterialDrvInfosD3D);
|
|
PixelShader = NULL;
|
|
PixelShaderUnlightedNoVertexColor = NULL;
|
|
std::fill(RGBPipe, RGBPipe + IDRV_MAT_MAXTEXTURES, true);
|
|
std::fill(AlphaPipe, AlphaPipe + IDRV_MAT_MAXTEXTURES, true);
|
|
FXCache = NULL;
|
|
NumUsedTexStages = 0;
|
|
}
|
|
~CMaterialDrvInfosD3D()
|
|
{
|
|
delete FXCache;
|
|
}
|
|
void buildTexEnv (uint stage, const CMaterial::CTexEnv &env, bool textured);
|
|
|
|
};
|
|
|
|
|
|
//
|
|
|
|
// ***************************************************************************
|
|
|
|
/* Volatile buffers.
|
|
*
|
|
* The volatile buffer system is a double buffer allocated during the render pass by objects that needs
|
|
* a temporary buffer to render vertex or indexes. Each lock allocates a buffer region
|
|
* and locks it with the NOOVERWRITE flag. Locks are not blocked. The buffer is reset at each frame begin.
|
|
* There is a double buffer to take benefit of the "over swapbuffer" parallelisme.
|
|
*/
|
|
|
|
// ***************************************************************************
|
|
|
|
class CVolatileVertexBuffer
|
|
{
|
|
public:
|
|
CVolatileVertexBuffer();
|
|
~CVolatileVertexBuffer();
|
|
|
|
CDriverD3D *Driver;
|
|
IDirect3DVertexBuffer9 *VertexBuffer;
|
|
uint Size;
|
|
CVertexBuffer::TLocation Location;
|
|
uint CurrentIndex;
|
|
uint MaxSize;
|
|
bool Locked;
|
|
|
|
/* size is in bytes */
|
|
void init (CVertexBuffer::TLocation location, uint size, uint maxSize, CDriverD3D *driver);
|
|
void release ();
|
|
|
|
// Runtime buffer access, no-blocking lock.
|
|
void *lock (uint size, uint stride, uint &offset);
|
|
void unlock ();
|
|
|
|
// Runtime reset (called at the begining of the frame rendering), blocking lock here.
|
|
void reset ();
|
|
};
|
|
|
|
// ***************************************************************************
|
|
|
|
class CVolatileIndexBuffer
|
|
{
|
|
public:
|
|
CVolatileIndexBuffer();
|
|
~CVolatileIndexBuffer();
|
|
|
|
CDriverD3D *Driver;
|
|
IDirect3DIndexBuffer9 *IndexBuffer;
|
|
uint Size;
|
|
CIndexBuffer::TLocation Location;
|
|
// current position in bytes!
|
|
uint CurrentIndex;
|
|
uint MaxSize;
|
|
bool Locked;
|
|
CIndexBuffer::TFormat Format;
|
|
|
|
/* size is in bytes */
|
|
void init (CIndexBuffer::TLocation location, uint sizeInBytes, uint maxSize, CDriverD3D *driver, CIndexBuffer::TFormat format);
|
|
void release ();
|
|
|
|
// Runtime buffer access, no-blocking lock. Re
|
|
void *lock (uint size, uint &offset);
|
|
void unlock ();
|
|
|
|
// Runtime reset (called at the begining of the frame rendering), blocking lock here.
|
|
void reset ();
|
|
};
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
|
|
class CDriverD3D : public IDriver, public ID3DXEffectStateManager
|
|
{
|
|
public:
|
|
|
|
enum
|
|
{
|
|
CacheTest_CullMode = 0,
|
|
CacheTest_RenderState = 1,
|
|
CacheTest_TextureState = 2,
|
|
CacheTest_TextureIndexMode = 3,
|
|
CacheTest_TextureIndexUV = 4,
|
|
CacheTest_Texture = 5,
|
|
CacheTest_VertexProgram = 6,
|
|
CacheTest_PixelShader = 7,
|
|
CacheTest_VertexProgramConstant = 8,
|
|
CacheTest_PixelShaderConstant = 9,
|
|
CacheTest_SamplerState = 10,
|
|
CacheTest_VertexBuffer = 11,
|
|
CacheTest_IndexBuffer = 12,
|
|
CacheTest_VertexDecl = 13,
|
|
CacheTest_Matrix = 14,
|
|
CacheTest_RenderTarget = 15,
|
|
CacheTest_MaterialState = 16,
|
|
CacheTest_DepthRange = 17,
|
|
CacheTest_Count
|
|
};
|
|
|
|
// Some constants
|
|
enum
|
|
{
|
|
MaxLight=8,
|
|
MaxRenderState=256,
|
|
MaxTextureState=36,
|
|
MaxTexture=8,
|
|
MaxSamplerState=16,
|
|
MaxSampler=8,
|
|
|
|
MatrixStateRemap = 24,
|
|
MaxMatrixState=32,
|
|
MaxVertexProgramConstantState=96,
|
|
MaxPixelShaderConstantState=96,
|
|
};
|
|
|
|
// Prefered pixel formats
|
|
enum
|
|
{
|
|
// Number of pixel format choice for each pixel format
|
|
FinalPixelFormatChoice = 5,
|
|
};
|
|
|
|
// Construction / destruction
|
|
CDriverD3D();
|
|
virtual ~CDriverD3D();
|
|
|
|
virtual bool isLost() const { return _Lost; }
|
|
// ***************************************************************************
|
|
// Implementation
|
|
// see nel\src\3d\driver.h
|
|
// ***************************************************************************
|
|
|
|
// Mode initialisation, requests
|
|
virtual bool init (uint windowIcon = 0, emptyProc exitFunc = 0);
|
|
virtual bool setDisplay(nlWindow wnd, const GfxMode& mode, bool show, bool resizeable) throw(EBadDisplay);
|
|
virtual bool release();
|
|
virtual bool setMode(const GfxMode& mode);
|
|
virtual bool getModes(std::vector<GfxMode> &modes);
|
|
virtual bool getCurrentScreenMode(GfxMode &mode);
|
|
virtual void beginDialogMode();
|
|
virtual void endDialogMode();
|
|
virtual bool activate();
|
|
virtual bool isActive ();
|
|
virtual bool initVertexBufferHard(uint agpMem, uint vramMem);
|
|
|
|
// Windows interface
|
|
virtual nlWindow getDisplay();
|
|
virtual emptyProc getWindowProc();
|
|
virtual NLMISC::IEventEmitter *getEventEmitter();
|
|
virtual void getWindowSize (uint32 &width, uint32 &height);
|
|
virtual void getWindowPos (sint32 &x, sint32 &y);
|
|
virtual uint8 getBitPerPixel ();
|
|
|
|
/// Set the title of the NeL window
|
|
virtual void setWindowTitle(const ucstring &title);
|
|
|
|
/// Set icon(s) of the NeL window
|
|
virtual void setWindowIcon(const std::vector<NLMISC::CBitmap> &bitmaps);
|
|
|
|
/// Set the position of the NeL window
|
|
virtual void setWindowPos(sint32 x, sint32 y);
|
|
|
|
/// Show or hide the NeL window
|
|
virtual void showWindow(bool show);
|
|
|
|
// Driver parameters
|
|
virtual void disableHardwareVertexProgram();
|
|
virtual void disableHardwareIndexArrayAGP();
|
|
virtual void disableHardwareVertexArrayAGP();
|
|
virtual void disableHardwareTextureShader();
|
|
virtual void forceDXTCCompression(bool dxtcComp);
|
|
virtual void forceTextureResize(uint divisor);
|
|
virtual void forceNativeFragmentPrograms(bool /* nativeOnly */) {} // ignored
|
|
|
|
// Driver information
|
|
virtual uint getNumAdapter() const;
|
|
virtual bool getAdapter(uint adapter, CAdapter &desc) const;
|
|
virtual bool setAdapter(uint adapter);
|
|
virtual uint32 getAvailableVertexAGPMemory ();
|
|
virtual uint32 getAvailableVertexVRAMMemory ();
|
|
virtual uint getNbTextureStages() const;
|
|
virtual void getNumPerStageConstant(uint &lightedMaterial, uint &unlightedMaterial) const;
|
|
virtual bool supportVertexBufferHard() const;
|
|
virtual bool supportVolatileVertexBuffer() const;
|
|
virtual bool supportIndexBufferHard() const;
|
|
// todo hulud d3d vertex buffer hard
|
|
virtual bool slowUnlockVertexBufferHard() const {return false;};
|
|
virtual uint getMaxVerticesByVertexBufferHard() const;
|
|
virtual uint32 getImplementationVersion () const;
|
|
virtual const char* getDriverInformation ();
|
|
virtual const char* getVideocardInformation ();
|
|
virtual CVertexBuffer::TVertexColorType getVertexColorFormat() const;
|
|
|
|
// Textures
|
|
virtual bool isTextureExist(const ITexture&tex);
|
|
virtual bool setupTexture (ITexture& tex);
|
|
virtual bool setupTextureEx (ITexture& tex, bool bUpload, bool &bAllUploaded, bool bMustRecreateSharedTexture= false);
|
|
virtual bool uploadTexture (ITexture& tex, NLMISC::CRect& rect, uint8 nNumMipMap);
|
|
// todo hulud d3d texture
|
|
virtual bool uploadTextureCube (ITexture& /* tex */, NLMISC::CRect& /* rect */, uint8 /* nNumMipMap */, uint8 /* nNumFace */) {return false;};
|
|
|
|
// Material
|
|
virtual bool setupMaterial(CMaterial& mat);
|
|
|
|
virtual bool supportCloudRenderSinglePass () const;
|
|
|
|
// Buffer
|
|
virtual bool clear2D(CRGBA rgba);
|
|
virtual bool clearZBuffer(float zval=1);
|
|
virtual bool clearStencilBuffer(float stencilval=0);
|
|
virtual void setColorMask (bool bRed, bool bGreen, bool bBlue, bool bAlpha);
|
|
virtual bool swapBuffers();
|
|
virtual void getBuffer (CBitmap &bitmap); // Only 32 bits back buffer supported
|
|
virtual void setDepthRange(float znear, float zfar);
|
|
virtual void getDepthRange(float &znear, float &zfar) const;
|
|
|
|
virtual void getZBuffer (std::vector<float> &zbuffer);
|
|
virtual void getBufferPart (CBitmap &bitmap, NLMISC::CRect &rect);
|
|
|
|
// return true if driver support Bloom effect.
|
|
virtual bool supportBloomEffect() const;
|
|
|
|
// return true if driver support non-power of two textures
|
|
virtual bool supportNonPowerOfTwoTextures() const;
|
|
|
|
// copy the first texture in a second one of different dimensions
|
|
virtual bool stretchRect (ITexture * srcText, NLMISC::CRect &srcRect, ITexture * destText, NLMISC::CRect &destRect); // Only 32 bits back buffer supported
|
|
virtual bool isTextureRectangle(ITexture * /* tex */) const {return false;}
|
|
IDirect3DSurface9* getSurfaceTexture(ITexture * text);
|
|
void getDirect3DRect(NLMISC::CRect &rect, RECT & d3dRect);
|
|
|
|
// todo hulud d3d buffers
|
|
virtual void getZBufferPart (std::vector<float> &zbuffer, NLMISC::CRect &rect);
|
|
virtual bool setRenderTarget (ITexture *tex, uint32 x, uint32 y, uint32 width, uint32 height, uint32 mipmapLevel, uint32 cubeFace);
|
|
virtual bool copyTargetToTexture (ITexture *tex, uint32 offsetx, uint32 offsety, uint32 x, uint32 y, uint32 width,
|
|
uint32 height, uint32 mipmapLevel);
|
|
virtual bool getRenderTargetSize (uint32 &width, uint32 &height);
|
|
virtual bool fillBuffer (CBitmap &bitmap);
|
|
|
|
// Performances
|
|
virtual void startSpecularBatch();
|
|
virtual void endSpecularBatch();
|
|
virtual void setSwapVBLInterval(uint interval);
|
|
virtual uint getSwapVBLInterval();
|
|
virtual void swapTextureHandle(ITexture &tex0, ITexture &tex1);
|
|
virtual uint getTextureHandle(const ITexture&tex);
|
|
|
|
// Matrix, viewport and frustum
|
|
virtual void setFrustum(float left, float right, float bottom, float top, float znear, float zfar, bool perspective = true);
|
|
virtual void setFrustumMatrix(CMatrix &frust);
|
|
virtual CMatrix getFrustumMatrix();
|
|
virtual float getClipSpaceZMin() const { return 0.f; }
|
|
virtual void setupViewMatrix(const CMatrix& mtx);
|
|
virtual void setupViewMatrixEx(const CMatrix& mtx, const CVector &cameraPos);
|
|
virtual void setupModelMatrix(const CMatrix& mtx);
|
|
virtual CMatrix getViewMatrix() const;
|
|
virtual void forceNormalize(bool normalize);
|
|
virtual bool isForceNormalize() const;
|
|
virtual void setupScissor (const class CScissor& scissor);
|
|
virtual void setupViewport (const class CViewport& viewport);
|
|
virtual void getViewport(CViewport &viewport);
|
|
|
|
// Vertex buffers
|
|
virtual bool activeVertexBuffer(CVertexBuffer& VB);
|
|
|
|
// Index buffers
|
|
virtual bool activeIndexBuffer(CIndexBuffer& IB);
|
|
|
|
// UV
|
|
virtual void mapTextureStageToUV(uint stage, uint uv);
|
|
|
|
// Indexed primitives
|
|
virtual bool renderLines(CMaterial& mat, uint32 firstIndex, uint32 nlines);
|
|
virtual bool renderTriangles(CMaterial& Mat, uint32 firstIndex, uint32 ntris);
|
|
virtual bool renderSimpleTriangles(uint32 firstTri, uint32 ntris);
|
|
// Indexed primitives with index offset
|
|
virtual bool renderLinesWithIndexOffset(CMaterial& mat, uint32 firstIndex, uint32 nlines, uint indexOffset);
|
|
virtual bool renderTrianglesWithIndexOffset(CMaterial& mat, uint32 firstIndex, uint32 ntris, uint indexOffset);
|
|
virtual bool renderSimpleTrianglesWithIndexOffset(uint32 firstIndex, uint32 ntris, uint indexOffset);
|
|
// Unindexed primitives with index offset
|
|
virtual bool renderRawPoints(CMaterial& Mat, uint32 startIndex, uint32 numPoints);
|
|
virtual bool renderRawLines(CMaterial& Mat, uint32 startIndex, uint32 numLines);
|
|
virtual bool renderRawTriangles(CMaterial& Mat, uint32 startIndex, uint32 numTris);
|
|
virtual bool renderRawQuads(CMaterial& Mat, uint32 startIndex, uint32 numQuads);
|
|
//
|
|
virtual void setPolygonMode (TPolygonMode mode);
|
|
virtual void finish();
|
|
virtual void flush();
|
|
|
|
// Profile
|
|
virtual void profileRenderedPrimitives(CPrimitiveProfile &pIn, CPrimitiveProfile &pOut);
|
|
virtual uint32 profileAllocatedTextureMemory();
|
|
virtual uint32 profileSetupedMaterials() const;
|
|
virtual uint32 profileSetupedModelMatrix() const;
|
|
virtual void enableUsedTextureMemorySum (bool enable);
|
|
virtual uint32 getUsedTextureMemory() const;
|
|
virtual void startProfileVBHardLock();
|
|
virtual void endProfileVBHardLock(std::vector<std::string> &result);
|
|
virtual void profileVBHardAllocation(std::vector<std::string> &result);
|
|
virtual void startProfileIBLock();
|
|
virtual void endProfileIBLock(std::vector<std::string> &result);
|
|
virtual void profileIBAllocation(std::vector<std::string> &result);
|
|
//virtual void profileIBAllocation(std::vector<std::string> &result);
|
|
|
|
// Misc
|
|
virtual TMessageBoxId systemMessageBox (const char* message, const char* title, TMessageBoxType type=okType, TMessageBoxIcon icon=noIcon);
|
|
virtual uint64 getSwapBufferCounter() const { return _SwapBufferCounter; }
|
|
|
|
// Inputs
|
|
virtual void showCursor (bool b);
|
|
virtual void setMousePos(float x, float y);
|
|
virtual void setCapture (bool b);
|
|
|
|
// see if system cursor is currently captured
|
|
virtual bool isSystemCursorCaptured();
|
|
|
|
// Add a new cursor (name is case unsensitive)
|
|
virtual void addCursor(const std::string &name, const NLMISC::CBitmap &bitmap);
|
|
|
|
// Display a cursor from its name (case unsensitive)
|
|
virtual void setCursor(const std::string &name, NLMISC::CRGBA col, uint8 rot, sint hotSpotX, sint hotSpotY, bool forceRebuild = false);
|
|
|
|
// Change default scale for all cursors
|
|
virtual void setCursorScale(float scale);
|
|
|
|
virtual NLMISC::IMouseDevice *enableLowLevelMouse(bool enable, bool exclusive);
|
|
virtual NLMISC::IKeyboardDevice *enableLowLevelKeyboard(bool enable);
|
|
virtual NLMISC::IInputDeviceManager *getLowLevelInputDeviceManager();
|
|
virtual uint getDoubleClickDelay(bool hardwareMouse);
|
|
|
|
// Lights
|
|
virtual uint getMaxLight () const;
|
|
virtual void setLight (uint8 num, const CLight& light);
|
|
virtual void enableLight (uint8 num, bool enable=true);
|
|
virtual void setLightMapDynamicLight (bool enable, const CLight& light);
|
|
// todo hulud d3d light
|
|
virtual void setPerPixelLightingLight(CRGBA /* diffuse */, CRGBA /* specular */, float /* shininess */) {}
|
|
virtual void setAmbientColor (CRGBA color);
|
|
|
|
// Fog
|
|
virtual bool fogEnabled();
|
|
virtual void enableFog(bool enable);
|
|
virtual void setupFog(float start, float end, CRGBA color);
|
|
virtual float getFogStart() const;
|
|
virtual float getFogEnd() const;
|
|
virtual CRGBA getFogColor() const;
|
|
|
|
// Texture addressing modes
|
|
// todo hulud d3d adressing mode
|
|
virtual bool supportTextureShaders() const {return false;};
|
|
virtual bool supportMADOperator() const;
|
|
// todo hulud d3d adressing mode
|
|
virtual bool isWaterShaderSupported() const;
|
|
// todo hulud d3d adressing mode
|
|
virtual bool isTextureAddrModeSupported(CMaterial::TTexAddressingMode /* mode */) const {return false;};
|
|
// todo hulud d3d adressing mode
|
|
virtual void setMatrix2DForTextureOffsetAddrMode(const uint /* stage */, const float /* mat */[4]) {}
|
|
|
|
// EMBM support
|
|
virtual bool supportEMBM() const;
|
|
virtual bool isEMBMSupportedAtStage(uint stage) const;
|
|
virtual void setEMBMMatrix(const uint stage, const float mat[4]);
|
|
virtual bool supportPerPixelLighting(bool /* specular */) const {return false;};
|
|
|
|
// index offset support
|
|
virtual bool supportIndexOffset() const { return true; /* always supported with D3D driver */ }
|
|
|
|
// Blend
|
|
virtual bool supportBlendConstantColor() const;
|
|
virtual void setBlendConstantColor(NLMISC::CRGBA col);
|
|
virtual NLMISC::CRGBA getBlendConstantColor() const;
|
|
|
|
// Monitor properties
|
|
virtual bool setMonitorColorProperties (const CMonitorColorProperties &properties);
|
|
|
|
// Polygon smoothing
|
|
virtual void enablePolygonSmoothing(bool smooth);
|
|
virtual bool isPolygonSmoothingEnabled() const;
|
|
|
|
// Material multipass
|
|
virtual sint beginMaterialMultiPass();
|
|
virtual void setupMaterialPass(uint pass);
|
|
virtual void endMaterialMultiPass();
|
|
|
|
// Vertex program
|
|
virtual bool isVertexProgramSupported () const;
|
|
virtual bool isVertexProgramEmulated () const;
|
|
virtual bool activeVertexProgram (CVertexProgram *program);
|
|
virtual void setConstant (uint index, float, float, float, float);
|
|
virtual void setConstant (uint index, double, double, double, double);
|
|
virtual void setConstant (uint index, const NLMISC::CVector& value);
|
|
virtual void setConstant (uint index, const NLMISC::CVectorD& value);
|
|
virtual void setConstant (uint index, uint num, const float *src);
|
|
virtual void setConstant (uint index, uint num, const double *src);
|
|
virtual void setConstantMatrix (uint index, IDriver::TMatrix matrix, IDriver::TTransform transform);
|
|
virtual void setConstantFog (uint index);
|
|
virtual void enableVertexProgramDoubleSidedColor(bool doubleSided);
|
|
virtual bool supportVertexProgramDoubleSidedColor() const;
|
|
|
|
// Occlusion query
|
|
virtual bool supportOcclusionQuery() const;
|
|
virtual IOcclusionQuery *createOcclusionQuery();
|
|
virtual void deleteOcclusionQuery(IOcclusionQuery *oq);
|
|
|
|
|
|
|
|
/** Shader implementation
|
|
*
|
|
* Shader can assume this states are setuped:
|
|
* TextureTransformFlags[n] = DISABLE;
|
|
* TexCoordIndex[n] = n;
|
|
* ColorOp[n] = DISABLE;
|
|
* AlphaOp[n] = DISABLE;
|
|
*/
|
|
virtual bool activeShader(CShader *shd);
|
|
|
|
// Bench
|
|
virtual void startBench (bool wantStandardDeviation = false, bool quick = false, bool reset = true);
|
|
virtual void endBench ();
|
|
virtual void displayBench (class NLMISC::CLog *log);
|
|
|
|
|
|
virtual void setCullMode(TCullMode cullMode);
|
|
virtual TCullMode getCullMode() const;
|
|
|
|
virtual void enableStencilTest(bool enable);
|
|
virtual bool isStencilTestEnabled() const;
|
|
virtual void stencilFunc(TStencilFunc stencilFunc, int ref, uint mask);
|
|
virtual void stencilOp(TStencilOp fail, TStencilOp zfail, TStencilOp zpass);
|
|
virtual void stencilMask(uint mask);
|
|
|
|
uint32 getMaxVertexIndex() const { return _MaxVertexIndex; }
|
|
|
|
bool supportPixelShaders() const { return _PixelShader; }
|
|
|
|
// *** Inline info
|
|
uint inlGetNumTextStages() const { return _NbNeLTextureStages; }
|
|
|
|
//private:
|
|
public:
|
|
|
|
// Hardware render variables, like matrices, render states
|
|
struct CRenderVariable
|
|
{
|
|
CRenderVariable()
|
|
{
|
|
NextModified = NULL;
|
|
Modified = false;
|
|
}
|
|
|
|
// Type of render state
|
|
enum
|
|
{
|
|
RenderState = 0,
|
|
TextureState,
|
|
TextureIndexState,
|
|
TexturePtrState,
|
|
VertexProgramPtrState,
|
|
PixelShaderPtrState,
|
|
VertexProgramConstantState,
|
|
PixelShaderConstantState,
|
|
SamplerState,
|
|
MatrixState,
|
|
VBState,
|
|
IBState,
|
|
VertexDecl,
|
|
LightState,
|
|
RenderTargetState,
|
|
} Type;
|
|
CRenderVariable *NextModified;
|
|
bool Modified;
|
|
virtual void apply(CDriverD3D *driver) = 0;
|
|
};
|
|
|
|
// Render state
|
|
struct CRenderState : public CRenderVariable
|
|
{
|
|
CRenderState()
|
|
{
|
|
Type = RenderState;
|
|
Value= 0;
|
|
ValueSet = false;
|
|
}
|
|
D3DRENDERSTATETYPE StateID;
|
|
DWORD Value;
|
|
bool ValueSet;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Render texture state
|
|
struct CTextureState : public CRenderVariable
|
|
{
|
|
CTextureState()
|
|
{
|
|
Type = TextureState;
|
|
DeviceValue = 0xcccccccc;
|
|
}
|
|
DWORD StageID;
|
|
D3DTEXTURESTAGESTATETYPE StateID;
|
|
DWORD Value;
|
|
DWORD DeviceValue;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
|
|
// Render texture index state
|
|
struct CTextureIndexState : public CRenderVariable
|
|
{
|
|
CTextureIndexState()
|
|
{
|
|
Type = TextureIndexState;
|
|
}
|
|
DWORD StageID;
|
|
DWORD TexGenMode;
|
|
DWORD UVChannel;
|
|
bool TexGen;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Render texture
|
|
struct CTexturePtrState : public CRenderVariable
|
|
{
|
|
CTexturePtrState()
|
|
{
|
|
Type = TexturePtrState;
|
|
Texture = NULL;
|
|
}
|
|
DWORD StageID;
|
|
LPDIRECT3DBASETEXTURE9 Texture;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Render texture
|
|
struct CVertexProgramPtrState : public CRenderVariable
|
|
{
|
|
CVertexProgramPtrState()
|
|
{
|
|
Type = VertexProgramPtrState;
|
|
VertexProgram = NULL;
|
|
}
|
|
LPDIRECT3DVERTEXSHADER9 VertexProgram;
|
|
// for debug
|
|
const CVertexProgram *VP;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Render texture
|
|
struct CPixelShaderPtrState : public CRenderVariable
|
|
{
|
|
CPixelShaderPtrState()
|
|
{
|
|
Type = PixelShaderPtrState;
|
|
PixelShader = NULL;
|
|
}
|
|
LPDIRECT3DPIXELSHADER9 PixelShader;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Vertex buffer constants state
|
|
struct CVertexProgramConstantState : public CRenderVariable
|
|
{
|
|
CVertexProgramConstantState()
|
|
{
|
|
Type = VertexProgramConstantState;
|
|
}
|
|
enum
|
|
{
|
|
Float= 0,
|
|
Int,
|
|
Undef,
|
|
} ValueType;
|
|
uint StateID;
|
|
DWORD Values[4];
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Pixel shader constants state
|
|
struct CPixelShaderConstantState : public CRenderVariable
|
|
{
|
|
CPixelShaderConstantState()
|
|
{
|
|
Type = PixelShaderConstantState;
|
|
}
|
|
enum
|
|
{
|
|
Float= 0,
|
|
Int,
|
|
Undef,
|
|
} ValueType;
|
|
uint StateID;
|
|
DWORD Values[4];
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Render sampler state
|
|
struct CSamplerState : public CRenderVariable
|
|
{
|
|
CSamplerState()
|
|
{
|
|
Type = SamplerState;
|
|
}
|
|
DWORD SamplerID;
|
|
D3DSAMPLERSTATETYPE StateID;
|
|
DWORD Value;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Render matrix
|
|
struct CMatrixState : public CRenderVariable
|
|
{
|
|
CMatrixState()
|
|
{
|
|
Type = MatrixState;
|
|
}
|
|
D3DTRANSFORMSTATETYPE TransformType;
|
|
D3DXMATRIX Matrix;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Render vertex buffer
|
|
struct CVBState : public CRenderVariable
|
|
{
|
|
CVBState()
|
|
{
|
|
Type = VBState;
|
|
VertexBuffer = NULL;
|
|
ColorOffset = 0;
|
|
}
|
|
IDirect3DVertexBuffer9 *VertexBuffer;
|
|
UINT Offset;
|
|
UINT Stride;
|
|
CVertexBuffer::TPreferredMemory PrefferedMemory;
|
|
DWORD Usage; // d3d vb usage
|
|
uint ColorOffset; // Fix for Radeon 7xxx series (see remark in CDriverD3D::createVertexDeclaration)
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Render index buffer
|
|
struct CIBState : public CRenderVariable
|
|
{
|
|
CIBState()
|
|
{
|
|
Type = IBState;
|
|
IndexBuffer = NULL;
|
|
}
|
|
IDirect3DIndexBuffer9 *IndexBuffer;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Render vertex decl
|
|
struct CVertexDeclState : public CRenderVariable
|
|
{
|
|
CVertexDeclState()
|
|
{
|
|
Type = VertexDecl;
|
|
Decl = NULL;
|
|
DeclAliasDiffuseToSpecular = NULL;
|
|
DeclNoDiffuse = NULL;
|
|
Stride= 0;
|
|
AliasDiffuseToSpecular = false;
|
|
EnableVertexColor = false;
|
|
}
|
|
IDirect3DVertexDeclaration9 *Decl;
|
|
IDirect3DVertexDeclaration9 *DeclAliasDiffuseToSpecular;
|
|
IDirect3DVertexDeclaration9 *DeclNoDiffuse;
|
|
uint Stride;
|
|
bool AliasDiffuseToSpecular;
|
|
bool EnableVertexColor;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Render vertex buffer
|
|
struct CLightState : public CRenderVariable
|
|
{
|
|
CLightState()
|
|
{
|
|
Type = LightState;
|
|
Enabled = false;
|
|
SettingsTouched = false;
|
|
EnabledTouched = true;
|
|
Light.Type = D3DLIGHT_POINT;
|
|
Light.Range = 1;
|
|
Light.Falloff = 1;
|
|
Light.Position.x = 0;
|
|
Light.Position.y = 0;
|
|
Light.Position.z = 0;
|
|
Light.Direction.x = 1;
|
|
Light.Direction.y = 0;
|
|
Light.Direction.z = 0;
|
|
Light.Attenuation0 = 1;
|
|
Light.Attenuation1 = 1;
|
|
Light.Attenuation2 = 1;
|
|
Light.Theta = 0;
|
|
Light.Phi = 0;
|
|
}
|
|
uint8 LightIndex;
|
|
bool SettingsTouched;
|
|
bool EnabledTouched;
|
|
bool Enabled;
|
|
D3DLIGHT9 Light;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// Render target
|
|
struct CRenderTargetState : public CRenderVariable
|
|
{
|
|
CRenderTargetState()
|
|
{
|
|
Type = RenderTargetState;
|
|
Target = NULL;
|
|
Texture = NULL;
|
|
Level = 0;
|
|
CubeFace = 0;
|
|
}
|
|
IDirect3DSurface9 *Target;
|
|
ITexture *Texture;
|
|
uint8 Level;
|
|
uint8 CubeFace;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
// material state
|
|
struct CMaterialState : public CRenderVariable
|
|
{
|
|
CMaterialState()
|
|
{
|
|
//Current.Power = -1.f;
|
|
Current.Diffuse.r = Current.Diffuse.g = Current.Diffuse.b = Current.Diffuse.a = 0.f;
|
|
Current.Ambient.r = Current.Ambient.g = Current.Ambient.b = Current.Ambient.a = 0.f;
|
|
Current.Specular.r = Current.Specular.g = Current.Specular.b = Current.Specular.a = 0.f;
|
|
Current.Emissive.r = Current.Emissive.g = Current.Emissive.b = Current.Emissive.a = 0.f;
|
|
Current.Power = 0.f;
|
|
}
|
|
D3DMATERIAL9 Current;
|
|
virtual void apply(CDriverD3D *driver);
|
|
};
|
|
|
|
|
|
// Friends
|
|
friend void D3DWndProc(CDriverD3D *driver, HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam);
|
|
friend class CTextureDrvInfosD3D;
|
|
friend class CVBDrvInfosD3D;
|
|
friend class CIBDrvInfosD3D;
|
|
friend class CVolatileVertexBuffer;
|
|
friend class CVolatileIndexBuffer;
|
|
|
|
// Init render states
|
|
void initRenderVariables();
|
|
|
|
// Reset render states
|
|
void resetRenderVariables();
|
|
|
|
// Replace all arguments of color / alpha operators in the pixel pipe with the given value
|
|
void replaceAllArgument(DWORD from, DWORD to, DWORD blendOpFrom);
|
|
|
|
// Replace all arguments of color / alpha operators in the pixel pipe at the given stage with the given value
|
|
void replaceAllArgumentAtStage(uint stage, DWORD from, DWORD to, DWORD blendOpFrom);
|
|
void replaceAllRGBArgumentAtStage(uint stage, DWORD from, DWORD to, DWORD blendOpFrom);
|
|
void replaceAllAlphaArgumentAtStage(uint stage, DWORD from, DWORD to, DWORD blendOpFrom);
|
|
|
|
// Replace a the given color / alpha op
|
|
void replaceArgumentAtStage(D3DTEXTURESTAGESTATETYPE state, DWORD stage, DWORD from, DWORD to);
|
|
|
|
// Setup lighting & material so that it produces the given constant color (useful to simulate a per stage constant using diffuse)
|
|
void setupConstantDiffuseColorFromLightedMaterial(D3DCOLOR color);
|
|
|
|
// Update all modified render states, reset current material
|
|
void updateRenderVariables();
|
|
|
|
// Update all modified render states, assume current material is still alive
|
|
void updateRenderVariablesInternal();
|
|
|
|
// Reset the driver, release the lost resources
|
|
bool reset (const GfxMode& mode);
|
|
|
|
// Handle window size change
|
|
bool handlePossibleSizeChange();
|
|
|
|
public:
|
|
// Touch a render variable
|
|
inline void touchRenderVariable (CRenderVariable *renderVariable)
|
|
{
|
|
// Modified ?
|
|
if (!renderVariable->Modified)
|
|
{
|
|
// Link to modified states
|
|
renderVariable->NextModified = _ModifiedRenderState;
|
|
_ModifiedRenderState = renderVariable;
|
|
renderVariable->Modified = true;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Access render states
|
|
inline void setRenderState (D3DRENDERSTATETYPE renderState, DWORD value)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setRenderState);
|
|
#ifdef NL_DEBUG
|
|
nlassert (_DeviceInterface);
|
|
nlassert (renderState<MaxRenderState);
|
|
#endif
|
|
// Ref on the state
|
|
CRenderState &_renderState = _RenderStateCache[renderState];
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_RenderState, _renderState.Value != value || !_renderState.ValueSet)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_renderState.Value = value;
|
|
_renderState.ValueSet = true;
|
|
touchRenderVariable (&_renderState);
|
|
}
|
|
}
|
|
|
|
|
|
public:
|
|
|
|
// compute wrap mode flag in D3D format for that texture
|
|
// texture must have been setuped already
|
|
void setupTextureWrapMode(ITexture& tex);
|
|
|
|
// Access texture states
|
|
inline void setTextureState (DWORD stage, D3DTEXTURESTAGESTATETYPE textureState, DWORD value)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setTextureState);
|
|
#ifdef NL_DEBUG
|
|
nlassert (_DeviceInterface);
|
|
nlassert (stage<MaxTexture);
|
|
nlassert (textureState<MaxTextureState);
|
|
#endif
|
|
// Ref on the state
|
|
CTextureState &_textureState = _TextureStateCache[stage][textureState];
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_TextureState, _textureState.Value != value)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_textureState.Value = value;
|
|
touchRenderVariable (&_textureState);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Access texture index states
|
|
inline void setTextureIndexMode (DWORD stage, bool texGenEnabled, DWORD value)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setTextureIndexMode);
|
|
nlassert (_DeviceInterface);
|
|
nlassert (stage<MaxTexture);
|
|
|
|
CTextureIndexState &_textureState = _TextureIndexStateCache[stage];
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_TextureIndexMode, _textureState.TexGen = (texGenEnabled || _textureState.TexGenMode != value))
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
// Ref on the state
|
|
_textureState.TexGen = texGenEnabled;
|
|
_textureState.TexGenMode = value;
|
|
touchRenderVariable (&_textureState);
|
|
}
|
|
}
|
|
|
|
|
|
// Access texture index states
|
|
inline void setTextureIndexUV (DWORD stage, DWORD value)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setTextureIndexUV);
|
|
nlassert (_DeviceInterface);
|
|
nlassert (stage<MaxTexture);
|
|
|
|
// Ref on the state
|
|
CTextureIndexState &_textureState = _TextureIndexStateCache[stage];
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_TextureIndexUV, _textureState.TexGen || _textureState.UVChannel != value)
|
|
#endif
|
|
{
|
|
_textureState.TexGen = false;
|
|
_textureState.UVChannel = value;
|
|
touchRenderVariable (&_textureState);
|
|
}
|
|
}
|
|
|
|
// Access texture states
|
|
inline void setTexture (DWORD stage, LPDIRECT3DBASETEXTURE9 texture)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setTexture);
|
|
nlassert (_DeviceInterface);
|
|
nlassert (stage<MaxTexture);
|
|
|
|
// Ref on the state
|
|
CTexturePtrState &_textureState = _TexturePtrStateCache[stage];
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_RenderState, _textureState.Texture != texture)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_textureState.Texture = texture;
|
|
touchRenderVariable (&_textureState);
|
|
}
|
|
}
|
|
|
|
// Set a NeL texture states
|
|
inline void setTexture (DWORD stage, ITexture *texture)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setTexture2);
|
|
nlassert (_DeviceInterface);
|
|
nlassert (stage<MaxTexture);
|
|
|
|
// Set the texture parameters
|
|
CTextureDrvInfosD3D *d3dtext = getTextureD3D(*texture);
|
|
setTexture (stage, d3dtext->Texture);
|
|
setSamplerState (stage, D3DSAMP_ADDRESSU, d3dtext->WrapS);
|
|
setSamplerState (stage, D3DSAMP_ADDRESSV, d3dtext->WrapT);
|
|
setSamplerState (stage, D3DSAMP_MAGFILTER, d3dtext->MagFilter);
|
|
setSamplerState (stage, D3DSAMP_MINFILTER, d3dtext->MinFilter);
|
|
setSamplerState (stage, D3DSAMP_MIPFILTER, d3dtext->MipFilter);
|
|
|
|
// Profile, log the use of this texture
|
|
if (_SumTextureMemoryUsed)
|
|
{
|
|
// Insert the pointer of this texture
|
|
_TextureUsed.insert (d3dtext);
|
|
}
|
|
}
|
|
|
|
// Access vertex program
|
|
inline void setVertexProgram (LPDIRECT3DVERTEXSHADER9 vertexProgram, const CVertexProgram *vp)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setVertexProgram);
|
|
nlassert (_DeviceInterface);
|
|
|
|
// Ref on the state
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_VertexProgram, _VertexProgramCache.VertexProgram != vertexProgram)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_VertexProgramCache.VertexProgram = vertexProgram;
|
|
_VertexProgramCache.VP = vp;
|
|
touchRenderVariable (&_VertexProgramCache);
|
|
}
|
|
}
|
|
|
|
// Access pixel shader
|
|
inline void setPixelShader (LPDIRECT3DPIXELSHADER9 pixelShader)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setPixelShader);
|
|
nlassert (_DeviceInterface);
|
|
|
|
// Ref on the state
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_PixelShader, _PixelShaderCache.PixelShader != pixelShader)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_PixelShaderCache.PixelShader = pixelShader;
|
|
touchRenderVariable (&_PixelShaderCache);
|
|
}
|
|
}
|
|
|
|
// Access vertex program constant
|
|
inline void setVertexProgramConstant (uint index, const float *values)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setVertexProgramConstant);
|
|
nlassert (_DeviceInterface);
|
|
nlassert (index<MaxVertexProgramConstantState);
|
|
|
|
// Ref on the state
|
|
CVertexProgramConstantState &state = _VertexProgramConstantCache[index];
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
bool doUpdate = ((*(float*)(state.Values+0)) != values[0]) ||
|
|
((*(float*)(state.Values+1)) != values[1]) ||
|
|
((*(float*)(state.Values+2)) != values[2]) ||
|
|
((*(float*)(state.Values+3)) != values[3]) ||
|
|
(state.ValueType != CVertexProgramConstantState::Float);
|
|
NL_D3D_CACHE_TEST(CacheTest_VertexProgramConstant, doUpdate)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
*(float*)(state.Values+0) = values[0];
|
|
*(float*)(state.Values+1) = values[1];
|
|
*(float*)(state.Values+2) = values[2];
|
|
*(float*)(state.Values+3) = values[3];
|
|
state.ValueType = CVertexProgramConstantState::Float;
|
|
touchRenderVariable (&state);
|
|
}
|
|
}
|
|
|
|
// Access vertex program constant
|
|
inline void setVertexProgramConstant (uint index, const int *values)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setVertexProgramConstant);
|
|
nlassert (_DeviceInterface);
|
|
nlassert (index<MaxVertexProgramConstantState);
|
|
|
|
// Ref on the state
|
|
CVertexProgramConstantState &state = _VertexProgramConstantCache[index];
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
bool doUpdate = ((*(int*)(state.Values+0)) != values[0]) ||
|
|
((*(int*)(state.Values+1)) != values[1]) ||
|
|
((*(int*)(state.Values+2)) != values[2]) ||
|
|
((*(int*)(state.Values+3)) != values[3]) ||
|
|
(state.ValueType != CVertexProgramConstantState::Int);
|
|
NL_D3D_CACHE_TEST(CacheTest_VertexProgramConstant, doUpdate)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
*(int*)(state.Values+0) = values[0];
|
|
*(int*)(state.Values+1) = values[1];
|
|
*(int*)(state.Values+2) = values[2];
|
|
*(int*)(state.Values+3) = values[3];
|
|
state.ValueType = CVertexProgramConstantState::Int;
|
|
touchRenderVariable (&state);
|
|
}
|
|
}
|
|
|
|
// Access pixel shader constant
|
|
inline void setPixelShaderConstant (uint index, const float *values)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setPixelShaderConstant);
|
|
nlassert (_DeviceInterface);
|
|
nlassert (index<MaxPixelShaderConstantState);
|
|
|
|
// Ref on the state
|
|
CPixelShaderConstantState &state = _PixelShaderConstantCache[index];
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
bool doUpdate = ((*(float*)(state.Values+0)) != values[0]) ||
|
|
((*(float*)(state.Values+1)) != values[1]) ||
|
|
((*(float*)(state.Values+2)) != values[2]) ||
|
|
((*(float*)(state.Values+3)) != values[3]) ||
|
|
(state.ValueType != CPixelShaderConstantState::Float);
|
|
NL_D3D_CACHE_TEST(CacheTest_PixelShaderConstant, doUpdate)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
*(float*)(state.Values+0) = values[0];
|
|
*(float*)(state.Values+1) = values[1];
|
|
*(float*)(state.Values+2) = values[2];
|
|
*(float*)(state.Values+3) = values[3];
|
|
state.ValueType = CPixelShaderConstantState::Float;
|
|
touchRenderVariable (&state);
|
|
}
|
|
}
|
|
|
|
// Access vertex program constant
|
|
inline void setPixelShaderConstant (uint index, const int *values)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setPixelShaderConstant);
|
|
nlassert (_DeviceInterface);
|
|
nlassert (index<MaxPixelShaderConstantState);
|
|
|
|
// Ref on the state
|
|
CPixelShaderConstantState &state = _PixelShaderConstantCache[index];
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
bool doUpdate = ((*(int*)(state.Values+0)) != values[0]) ||
|
|
((*(int*)(state.Values+1)) != values[1]) ||
|
|
((*(int*)(state.Values+2)) != values[2]) ||
|
|
((*(int*)(state.Values+3)) != values[3]) ||
|
|
(state.ValueType != CPixelShaderConstantState::Int);
|
|
NL_D3D_CACHE_TEST(CacheTest_PixelShaderConstant, doUpdate)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
*(int*)(state.Values+0) = values[0];
|
|
*(int*)(state.Values+1) = values[1];
|
|
*(int*)(state.Values+2) = values[2];
|
|
*(int*)(state.Values+3) = values[3];
|
|
state.ValueType = CPixelShaderConstantState::Int;
|
|
touchRenderVariable (&state);
|
|
}
|
|
}
|
|
|
|
// Access sampler states
|
|
inline void setSamplerState (DWORD sampler, D3DSAMPLERSTATETYPE samplerState, DWORD value)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setSamplerState);
|
|
nlassert (_DeviceInterface);
|
|
nlassert (sampler<MaxSampler);
|
|
nlassert (samplerState<MaxSamplerState);
|
|
|
|
// Ref on the state
|
|
CSamplerState &_samplerState = _SamplerStateCache[sampler][samplerState];
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_SamplerState, _samplerState.Value != value)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_samplerState.Value = value;
|
|
touchRenderVariable (&_samplerState);
|
|
}
|
|
}
|
|
|
|
// Set the vertex buffer
|
|
inline void setVertexBuffer (IDirect3DVertexBuffer9 *vertexBuffer, UINT offset, UINT stride, bool useVertexColor, uint size, CVertexBuffer::TPreferredMemory pm, DWORD usage, uint colorOffset)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setVertexBuffer);
|
|
nlassert (_DeviceInterface);
|
|
|
|
// Ref on the state
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
//NL_D3D_CACHE_TEST(CacheTest_VertexBuffer, (_VertexBufferCache.VertexBuffer != vertexBuffer) || (_VertexBufferCache.Offset != offset) || (stride != _VertexBufferCache.Stride) || (colorOffset != _VertexBufferCache.ColorOffset))
|
|
NL_D3D_CACHE_TEST(CacheTest_VertexBuffer, (_VertexBufferCache.VertexBuffer != vertexBuffer) || (_VertexBufferCache.Offset != offset) || (stride != _VertexBufferCache.Stride) || (colorOffset != _VertexBufferCache.ColorOffset))
|
|
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_VertexBufferCache.VertexBuffer = vertexBuffer;
|
|
_VertexBufferCache.Offset = 0;
|
|
_VertexBufferCache.Stride = stride;
|
|
_VertexBufferCache.PrefferedMemory = pm;
|
|
_VertexBufferCache.Usage = usage;
|
|
_VertexBufferCache.ColorOffset = colorOffset;
|
|
touchRenderVariable (&_VertexBufferCache);
|
|
|
|
/* Work around for a NVIDIA bug in driver 53.03 - 56.72
|
|
* Sometime, after a lock D3DLOCK_NOOVERWRITE, the D3DTSS_TEXCOORDINDEX state seams to change.
|
|
* So, force it at every vertex buffer set.
|
|
**/
|
|
if (_IsGeforce)
|
|
{
|
|
touchRenderVariable (&_TextureStateCache[0][D3DTSS_TEXCOORDINDEX]);
|
|
touchRenderVariable (&_TextureStateCache[1][D3DTSS_TEXCOORDINDEX]);
|
|
touchRenderVariable (&_TextureStateCache[2][D3DTSS_TEXCOORDINDEX]);
|
|
touchRenderVariable (&_TextureStateCache[3][D3DTSS_TEXCOORDINDEX]);
|
|
}
|
|
}
|
|
_UseVertexColor = useVertexColor;
|
|
_VertexBufferSize = size;
|
|
_VertexBufferOffset = offset;
|
|
}
|
|
|
|
// Force to alias diffuse color to specular color in the current vertex buffer
|
|
inline void setAliasDiffuseToSpecular(bool enable)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setAliasDiffuseToSpecular);
|
|
// Ref on the state
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_VertexBuffer, enable != _VertexDeclCache.AliasDiffuseToSpecular)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_VertexDeclCache.AliasDiffuseToSpecular = enable;
|
|
touchRenderVariable (&_VertexDeclCache);
|
|
}
|
|
}
|
|
|
|
// Enable / disable vertex color from the vertex declaration
|
|
inline void setEnableVertexColor(bool enable)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setEnableVertexColor);
|
|
// Ref on the state
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_VertexBuffer, enable != _VertexDeclCache.EnableVertexColor)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_VertexDeclCache.EnableVertexColor = enable;
|
|
touchRenderVariable (&_VertexDeclCache);
|
|
touchRenderVariable (&_VertexBufferCache);
|
|
}
|
|
}
|
|
|
|
// Set the index buffer
|
|
inline void setIndexBuffer (IDirect3DIndexBuffer9 *indexBuffer, uint offset)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setIndexBuffer);
|
|
nlassert (_DeviceInterface);
|
|
|
|
// Ref on the state
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_IndexBuffer, _IndexBufferCache.IndexBuffer != indexBuffer)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_IndexBufferCache.IndexBuffer = indexBuffer;
|
|
touchRenderVariable (&_IndexBufferCache);
|
|
}
|
|
_IndexBufferOffset = offset;
|
|
}
|
|
|
|
// Set the vertex declaration
|
|
inline void setVertexDecl (IDirect3DVertexDeclaration9 *vertexDecl, IDirect3DVertexDeclaration9 *vertexDeclAliasDiffuseToSpecular, IDirect3DVertexDeclaration9 *vertexDeclNoDiffuse, uint stride)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setVertexDecl);
|
|
nlassert (_DeviceInterface);
|
|
|
|
// Ref on the state
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_VertexDecl, stride != _VertexDeclCache.Stride || _VertexDeclCache.Decl != vertexDecl || _VertexDeclCache.DeclAliasDiffuseToSpecular != vertexDeclAliasDiffuseToSpecular || vertexDeclNoDiffuse != _VertexDeclCache.DeclNoDiffuse)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_VertexDeclCache.Decl = vertexDecl;
|
|
_VertexDeclCache.DeclAliasDiffuseToSpecular = vertexDeclAliasDiffuseToSpecular;
|
|
_VertexDeclCache.Stride = stride;
|
|
_VertexDeclCache.DeclNoDiffuse = vertexDeclNoDiffuse;
|
|
touchRenderVariable (&_VertexDeclCache);
|
|
}
|
|
}
|
|
|
|
// Access matrices
|
|
inline uint remapMatrixIndex (D3DTRANSFORMSTATETYPE type)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_remapMatrixIndex);
|
|
if (type>=256)
|
|
return (D3DTRANSFORMSTATETYPE)(MatrixStateRemap + type - 256);
|
|
else
|
|
return (uint)type;
|
|
}
|
|
inline void setMatrix (D3DTRANSFORMSTATETYPE type, const D3DXMATRIX &matrix)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setMatrix);
|
|
nlassert (_DeviceInterface);
|
|
|
|
// Remap high matrices indexes
|
|
type = (D3DTRANSFORMSTATETYPE)remapMatrixIndex (type);
|
|
nlassert (type<MaxMatrixState);
|
|
|
|
CMatrixState &theMatrix = _MatrixCache[type];
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
bool doUpdate = (matrix._11 != theMatrix.Matrix._11) ||
|
|
(matrix._12 != theMatrix.Matrix._12) ||
|
|
(matrix._13 != theMatrix.Matrix._13) ||
|
|
(matrix._14 != theMatrix.Matrix._14) ||
|
|
(matrix._21 != theMatrix.Matrix._21) ||
|
|
(matrix._22 != theMatrix.Matrix._22) ||
|
|
(matrix._23 != theMatrix.Matrix._23) ||
|
|
(matrix._24 != theMatrix.Matrix._24) ||
|
|
(matrix._31 != theMatrix.Matrix._31) ||
|
|
(matrix._32 != theMatrix.Matrix._32) ||
|
|
(matrix._33 != theMatrix.Matrix._33) ||
|
|
(matrix._34 != theMatrix.Matrix._34) ||
|
|
(matrix._41 != theMatrix.Matrix._41) ||
|
|
(matrix._42 != theMatrix.Matrix._42) ||
|
|
(matrix._43 != theMatrix.Matrix._43) ||
|
|
(matrix._44 != theMatrix.Matrix._44);
|
|
NL_D3D_CACHE_TEST(CacheTest_Matrix, doUpdate)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
theMatrix.Matrix = matrix;
|
|
touchRenderVariable (&theMatrix);
|
|
}
|
|
}
|
|
|
|
// Access texture states
|
|
inline void setRenderTarget (IDirect3DSurface9 *target, ITexture *texture, uint8 level, uint8 cubeFace)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_setRenderTarget);
|
|
nlassert (_DeviceInterface);
|
|
|
|
// Ref on the state
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
NL_D3D_CACHE_TEST(CacheTest_RenderTarget, _RenderTarget.Target != target)
|
|
#endif // NL_D3D_USE_RENDER_STATE_CACHE
|
|
{
|
|
_RenderTarget.Target = target;
|
|
_RenderTarget.Texture = texture;
|
|
_RenderTarget.Level = level;
|
|
_RenderTarget.CubeFace = cubeFace;
|
|
|
|
touchRenderVariable (&_RenderTarget);
|
|
|
|
_ScissorTouched = true;
|
|
}
|
|
}
|
|
|
|
|
|
public:
|
|
void setMaterialState(const D3DMATERIAL9 &material)
|
|
{
|
|
H_AUTO_D3D(setMaterialState);
|
|
#ifdef NL_D3D_USE_RENDER_STATE_CACHE
|
|
bool update = material.Power != _MaterialState.Current.Power ||
|
|
material.Ambient != _MaterialState.Current.Ambient ||
|
|
material.Emissive != _MaterialState.Current.Emissive ||
|
|
material.Diffuse != _MaterialState.Current.Diffuse ||
|
|
material.Specular != _MaterialState.Current.Specular;
|
|
NL_D3D_CACHE_TEST(CacheTest_MaterialState, update)
|
|
#endif
|
|
{
|
|
_MaterialState.Current = material;
|
|
touchRenderVariable(&_MaterialState);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
// Get the d3dtext mirror of an existing setuped texture.
|
|
static inline CTextureDrvInfosD3D* getTextureD3D(ITexture& tex)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_getTextureD3D);
|
|
CTextureDrvInfosD3D* d3dtex;
|
|
d3dtex= (CTextureDrvInfosD3D*)(ITextureDrvInfos*)(tex.TextureDrvShare->DrvTexture);
|
|
return d3dtex;
|
|
}
|
|
|
|
// Get the d3dtext mirror of an existing setuped vertex program.
|
|
static inline CVertexProgamDrvInfosD3D* getVertexProgramD3D(CVertexProgram& vertexProgram)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_getVertexProgramD3D);
|
|
CVertexProgamDrvInfosD3D* d3dVertexProgram;
|
|
d3dVertexProgram = (CVertexProgamDrvInfosD3D*)(IVertexProgramDrvInfos*)(vertexProgram._DrvInfo);
|
|
return d3dVertexProgram;
|
|
}
|
|
|
|
// *** Init helper
|
|
|
|
bool isDepthFormatOk(UINT adapter, D3DDEVTYPE rasterizer, D3DFORMAT DepthFormat, D3DFORMAT AdapterFormat, D3DFORMAT BackBufferFormat);
|
|
bool isTextureFormatOk(UINT adapter, D3DDEVTYPE rasterizer, D3DFORMAT TextureFormat, D3DFORMAT AdapterFormat);
|
|
bool fillPresentParameter (D3DPRESENT_PARAMETERS ¶meters, D3DFORMAT &adapterFormat, const GfxMode& mode, const D3DDISPLAYMODE &adapterMode);
|
|
|
|
// *** Texture helper
|
|
|
|
D3DFORMAT getD3DDestTextureFormat (ITexture& tex);
|
|
// Generates the texture, degades it, creates / resizes the d3d surface. Don't fill the surface.
|
|
bool generateD3DTexture (ITexture& tex, bool textureDegradation, D3DFORMAT &destFormat, D3DFORMAT &srcFormat, bool &cube);
|
|
// Return the memory used by the surface described iwith the parameters
|
|
uint32 computeTextureMemoryUsage (uint width, uint height, uint levels, D3DFORMAT destFormat, bool cube);
|
|
// Upload a texture part
|
|
bool uploadTextureInternal (ITexture& tex, NLMISC::CRect& rect, uint8 destMipmap, uint8 srcMipmap, D3DFORMAT destFormat, D3DFORMAT srcFormat);
|
|
|
|
// *** Matrix helper
|
|
|
|
// Update _D3DModelView and _D3DModelViewProjection matrices. Call this if the model, the projection or the view matrix are modified
|
|
void updateMatrices ();
|
|
// Update the projection matrix. Call this if the driver resolution, the viewport or the frustum changes.
|
|
void updateProjectionMatrix ();
|
|
|
|
// *** Vertex buffer helper
|
|
|
|
// Create a vertex declaration
|
|
bool createVertexDeclaration (uint16 vertexFormat, const uint8 *typeArray,
|
|
IDirect3DVertexDeclaration9 **vertexDecl,
|
|
uint &colorOffset,
|
|
bool aliasDiffuseToSpecular,
|
|
bool bypassDiffuse,
|
|
uint *stride = NULL
|
|
);
|
|
|
|
// *** Index buffer helper
|
|
|
|
// *** Multipass helpers
|
|
|
|
void initInternalShaders();
|
|
void releaseInternalShaders();
|
|
bool setShaderTexture (uint textureHandle, ITexture *texture, CFXCache *cache);
|
|
|
|
bool validateShader(CShader *shader);
|
|
|
|
void activePass (uint pass)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_activePass);
|
|
if (_CurrentShader)
|
|
{
|
|
CShaderDrvInfosD3D *drvInfo = static_cast<CShaderDrvInfosD3D*>((IShaderDrvInfos*)_CurrentShader->_DrvInfo);
|
|
if (_CurrentMaterialInfo->FXCache)
|
|
{
|
|
nlassert(_CurrentMaterialInfo);
|
|
_CurrentMaterialInfo->FXCache->applyPass(*this, drvInfo, pass);
|
|
}
|
|
else
|
|
{
|
|
#if (DIRECT3D_VERSION >= 0x0900) && (D3D_SDK_VERSION >= 32)
|
|
drvInfo->Effect->BeginPass (pass);
|
|
drvInfo->Effect->EndPass ();
|
|
#else
|
|
drvInfo->Effect->Pass (pass);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
// Update render states
|
|
updateRenderVariablesInternal();
|
|
}
|
|
|
|
void beginMultiPass ()
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_beginMultiPass);
|
|
if (_CurrentShader)
|
|
{
|
|
// Does the shader validated ?
|
|
validateShader(_CurrentShader);
|
|
|
|
// Init default state value
|
|
uint i;
|
|
for (i=0; i<MaxTexture; i++)
|
|
{
|
|
setTextureState (i, D3DTSS_TEXTURETRANSFORMFLAGS, D3DTTFF_DISABLE);
|
|
setTextureIndexMode (i, false, D3DTSS_TCI_PASSTHRU);
|
|
setTextureIndexUV (i, i);
|
|
setTextureState (i, D3DTSS_COLOROP, D3DTOP_DISABLE);
|
|
//setTextureState (i, D3DTSS_ALPHAOP, D3DTOP_DISABLE);
|
|
}
|
|
|
|
CShaderDrvInfosD3D *drvInfo = static_cast<CShaderDrvInfosD3D*>((IShaderDrvInfos*)_CurrentShader->_DrvInfo);
|
|
if (_CurrentMaterialInfo->FXCache)
|
|
{
|
|
nlassert(_CurrentMaterialInfo);
|
|
_CurrentMaterialInfo->FXCache->begin(drvInfo, this);
|
|
}
|
|
else
|
|
{
|
|
drvInfo->Effect->Begin (&_CurrentShaderPassCount, D3DXFX_DONOTSAVESTATE|D3DXFX_DONOTSAVESHADERSTATE);
|
|
}
|
|
}
|
|
else
|
|
// No shader setuped
|
|
_CurrentShaderPassCount = 1;
|
|
}
|
|
|
|
void endMultiPass ()
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_endMultiPass);
|
|
if (_CurrentShader)
|
|
{
|
|
CShaderDrvInfosD3D *drvInfo = static_cast<CShaderDrvInfosD3D*>((IShaderDrvInfos*)_CurrentShader->_DrvInfo);
|
|
if (_CurrentMaterialInfo->FXCache)
|
|
{
|
|
_CurrentMaterialInfo->FXCache->end(drvInfo);
|
|
}
|
|
else
|
|
{
|
|
drvInfo->Effect->End ();
|
|
}
|
|
}
|
|
}
|
|
|
|
// render helpers
|
|
bool renderPrimitives(D3DPRIMITIVETYPE primitiveType, uint numVertexPerPrim, CMaterial& mat, uint firstVertex, uint32 nPrims);
|
|
bool renderIndexedPrimitives(D3DPRIMITIVETYPE primitiveType, uint numVertexPerPrim, CMaterial& mat, uint32 firstIndex, uint32 nPrims, uint indexOffset = 0);
|
|
bool renderSimpleIndexedPrimitives(D3DPRIMITIVETYPE primitiveType, uint numVertexPerPrim, uint32 firstIndex, uint32 nPrims, uint indexOffset = 0);
|
|
void convertToIndices16(uint firstIndex, uint numIndices);
|
|
|
|
// Returns true if this material needs a constant color for the diffuse component
|
|
static bool needsConstantForDiffuse (CMaterial &mat);
|
|
|
|
/* Returns true if this normal needs constant. If true, numConstant is the number of needed constants, firstConstant is the first
|
|
constants needed. */
|
|
static bool needsConstants (uint &numConstant, uint &firstConstant, uint &secondConstant, CMaterial &mat);
|
|
|
|
// For normal shader, compute part of the pipeline that are worth setting
|
|
// For example : if alpha output if not used (no alpha test or blend), then the alpha part won't be set at all
|
|
static void computeRelevantTexEnv(CMaterial &mat, bool rgbPipe[IDRV_MAT_MAXTEXTURES], bool alphaPipe[IDRV_MAT_MAXTEXTURES]);
|
|
|
|
// Build a pixel shader for normal shader
|
|
IDirect3DPixelShader9 *buildPixelShader (const CNormalShaderDesc &normalShaderDesc, bool unlightedNoVertexColor);
|
|
|
|
|
|
// Notify all the shaders drivers info that a device has been lost
|
|
void notifyAllShaderDrvOfLostDevice();
|
|
// Notify all the shaders drivers info that a device has been reset
|
|
void notifyAllShaderDrvOfResetDevice();
|
|
|
|
// *** Debug helpers
|
|
|
|
void setDebugMaterial();
|
|
|
|
// ** From ID3DXEffectStateManager
|
|
public:
|
|
HRESULT STDMETHODCALLTYPE QueryInterface(REFIID riid, LPVOID *ppvObj);
|
|
ULONG STDMETHODCALLTYPE AddRef(VOID);
|
|
ULONG STDMETHODCALLTYPE Release(VOID);
|
|
HRESULT STDMETHODCALLTYPE LightEnable(DWORD Index, BOOL Enable);
|
|
HRESULT STDMETHODCALLTYPE SetFVF(DWORD FVF);
|
|
HRESULT STDMETHODCALLTYPE SetLight(DWORD Index, CONST D3DLIGHT9* pLight);
|
|
HRESULT STDMETHODCALLTYPE SetMaterial(CONST D3DMATERIAL9* pMaterial);
|
|
HRESULT STDMETHODCALLTYPE SetNPatchMode(FLOAT nSegments);
|
|
HRESULT STDMETHODCALLTYPE SetPixelShader(LPDIRECT3DPIXELSHADER9 pShader);
|
|
HRESULT STDMETHODCALLTYPE SetPixelShaderConstantB(UINT StartRegister, CONST BOOL* pConstantData, UINT RegisterCount);
|
|
HRESULT STDMETHODCALLTYPE SetPixelShaderConstantF(UINT StartRegister, CONST FLOAT* pConstantData, UINT RegisterCount);
|
|
HRESULT STDMETHODCALLTYPE SetPixelShaderConstantI(UINT StartRegister, CONST INT* pConstantData, UINT RegisterCount);
|
|
HRESULT STDMETHODCALLTYPE SetRenderState(D3DRENDERSTATETYPE State, DWORD Value);
|
|
HRESULT STDMETHODCALLTYPE SetSamplerState(DWORD Sampler, D3DSAMPLERSTATETYPE Type, DWORD Value);
|
|
HRESULT STDMETHODCALLTYPE SetTexture (DWORD Stage, LPDIRECT3DBASETEXTURE9 pTexture);
|
|
HRESULT STDMETHODCALLTYPE SetTextureStageState(DWORD Stage, D3DTEXTURESTAGESTATETYPE Type, DWORD Value);
|
|
HRESULT STDMETHODCALLTYPE SetTransform(D3DTRANSFORMSTATETYPE State, CONST D3DMATRIX* pMatrix);
|
|
HRESULT STDMETHODCALLTYPE SetVertexShader(LPDIRECT3DVERTEXSHADER9 pShader);
|
|
HRESULT STDMETHODCALLTYPE SetVertexShaderConstantB(UINT StartRegister, CONST BOOL* pConstantData, UINT RegisterCount);
|
|
HRESULT STDMETHODCALLTYPE SetVertexShaderConstantF(UINT StartRegister, CONST FLOAT* pConstantData, UINT RegisterCount);
|
|
HRESULT STDMETHODCALLTYPE SetVertexShaderConstantI(UINT StartRegister, CONST INT* pConstantData, UINT RegisterCount);
|
|
private:
|
|
|
|
void findNearestFullscreenVideoMode();
|
|
|
|
// Windows
|
|
std::string _WindowClass;
|
|
HWND _HWnd;
|
|
sint32 _WindowX;
|
|
sint32 _WindowY;
|
|
bool _DestroyWindow;
|
|
bool _Maximized;
|
|
bool _HandlePossibleSizeChangeNextSize;
|
|
GfxMode _CurrentMode;
|
|
uint _Interval;
|
|
bool _FullScreen;
|
|
|
|
// cursors
|
|
enum TColorDepth { ColorDepth16 = 0, ColorDepth32, ColorDepthCount };
|
|
|
|
TColorDepth _ColorDepth;
|
|
std::string _CurrName;
|
|
NLMISC::CRGBA _CurrCol;
|
|
uint8 _CurrRot;
|
|
uint _CurrHotSpotX;
|
|
uint _CurrHotSpotY;
|
|
float _CursorScale;
|
|
|
|
nlCursor _DefaultCursor;
|
|
|
|
bool _AlphaBlendedCursorSupported;
|
|
bool _AlphaBlendedCursorSupportRetrieved;
|
|
|
|
class CCursor
|
|
{
|
|
public:
|
|
NLMISC::CBitmap Src;
|
|
TColorDepth ColorDepth;
|
|
uint OrigHeight;
|
|
float HotspotScale;
|
|
uint HotspotOffsetX;
|
|
uint HotspotOffsetY;
|
|
sint HotSpotX;
|
|
sint HotSpotY;
|
|
nlCursor Cursor;
|
|
NLMISC::CRGBA Col;
|
|
uint8 Rot;
|
|
#if defined(NL_OS_UNIX) && !defined(NL_OS_MAC)
|
|
Display *Dpy;
|
|
#endif
|
|
public:
|
|
CCursor();
|
|
~CCursor();
|
|
CCursor& operator= (const CCursor& from);
|
|
|
|
void reset();
|
|
};
|
|
|
|
struct CStrCaseUnsensitiveCmp
|
|
{
|
|
bool operator()(const std::string &lhs, const std::string &rhs) const
|
|
{
|
|
return NLMISC::nlstricmp(lhs, rhs) < 0;
|
|
}
|
|
};
|
|
|
|
typedef std::map<std::string, CCursor, CStrCaseUnsensitiveCmp> TCursorMap;
|
|
|
|
TCursorMap _Cursors;
|
|
|
|
// Directx
|
|
uint32 _Adapter;
|
|
D3DDEVTYPE _Rasterizer;
|
|
LPDIRECT3D9 _D3D;
|
|
public:
|
|
IDirect3DDevice9 *_DeviceInterface;
|
|
private:
|
|
|
|
// Events
|
|
NLMISC::CEventEmitterMulti _EventEmitter; // this can contains a win emitter and eventually a direct input emitter
|
|
|
|
// Some matrices (Local -> Model -> World -> Screen)
|
|
public:
|
|
CVector _PZBCameraPos;
|
|
private:
|
|
CMatrix _UserModelMtx;
|
|
CMatrix _UserViewMtx;
|
|
public:
|
|
CViewport _Viewport;
|
|
private:
|
|
D3DVIEWPORT9 _D3DViewport;
|
|
public:
|
|
CScissor _Scissor;
|
|
private:
|
|
float _OODeltaZ; // Backup znear and zfar
|
|
D3DXMATRIX _D3DSpecularWorldTex; // World (as in NeL) to model matrix.
|
|
D3DXMATRIX _D3DModelView; // Local to world (as in D3D) matrix.
|
|
D3DXMATRIX _D3DModelViewProjection; // Local to screen (as in D3D) matrix.
|
|
D3DXMATRIX _D3DInvModelView; // World (as in DX) to local matrix.
|
|
bool _ForceNormalize;
|
|
bool _FrustumPerspective;
|
|
bool _InvertCullMode;
|
|
bool _DoubleSided;
|
|
float _FrustumLeft;
|
|
float _FrustumRight;
|
|
float _FrustumTop;
|
|
float _FrustumBottom;
|
|
float _FrustumZNear;
|
|
float _FrustumZFar;
|
|
float _FogStart;
|
|
float _FogEnd;
|
|
|
|
// Vertex memory available
|
|
uint32 _AGPMemoryAllocated;
|
|
uint32 _VRAMMemoryAllocated;
|
|
|
|
// Textures caps
|
|
D3DFORMAT _PreferedTextureFormat[ITexture::UploadFormatCount];
|
|
uint _ForceTextureResizePower;
|
|
bool _ForceDXTCCompression:1;
|
|
bool _TextureCubeSupported;
|
|
bool _VertexProgram;
|
|
bool _PixelShader;
|
|
bool _DisableHardwareVertexProgram;
|
|
bool _DisableHardwareVertexArrayAGP;
|
|
bool _DisableHardwareIndexArrayAGP;
|
|
bool _DisableHardwarePixelShader;
|
|
bool _MADOperatorSupported;
|
|
bool _EMBMSupported;
|
|
bool _CubbedMipMapSupported;
|
|
bool _IsGeforce;
|
|
bool _NonPowerOfTwoTexturesSupported;
|
|
uint _NbNeLTextureStages; // Number of texture stage for NeL (max IDRV_MAT_MAXTEXTURES)
|
|
uint _MaxVerticesByVertexBufferHard;
|
|
uint _MaxLight;
|
|
uint32 _PixelShaderVersion;
|
|
uint32 _MaxPrimitiveCount;
|
|
uint32 _MaxVertexIndex;
|
|
uint _MaxNumPerStageConstantLighted;
|
|
uint _MaxNumPerStageConstantUnlighted;
|
|
|
|
// Profiling
|
|
CPrimitiveProfile _PrimitiveProfileIn;
|
|
CPrimitiveProfile _PrimitiveProfileOut;
|
|
uint32 _AllocatedTextureMemory;
|
|
uint32 _NbSetupMaterialCall;
|
|
uint32 _NbSetupModelMatrixCall;
|
|
bool _SumTextureMemoryUsed;
|
|
std::set<CTextureDrvInfosD3D*> _TextureUsed;
|
|
|
|
// VBHard Lock Profiling
|
|
struct CVBHardProfile
|
|
{
|
|
NLMISC::CRefPtr<CVertexBuffer> VBHard;
|
|
NLMISC::TTicks AccumTime;
|
|
// true if the VBHard was not always the same for the same chronogical place.
|
|
bool Change;
|
|
CVBHardProfile()
|
|
{
|
|
AccumTime= 0;
|
|
Change= false;
|
|
}
|
|
};
|
|
|
|
// Index buffer lock profiling
|
|
struct CIBProfile
|
|
{
|
|
NLMISC::CRefPtr<CIndexBuffer> IB;
|
|
NLMISC::TTicks AccumTime;
|
|
// true if the VBHard was not always the same for the same chronogical place.
|
|
bool Change;
|
|
CIBProfile()
|
|
{
|
|
AccumTime= 0;
|
|
Change= false;
|
|
}
|
|
};
|
|
|
|
// The vb hard Profiles in chronogical order.
|
|
bool _VBHardProfiling;
|
|
std::vector<CVBHardProfile> _VBHardProfiles;
|
|
uint _CurVBHardLockCount;
|
|
uint _NumVBHardProfileFrame;
|
|
void appendVBHardLockProfile(NLMISC::TTicks time, CVertexBuffer *vb);
|
|
|
|
// The index buffer profile in chronogical order.
|
|
bool _IBProfiling;
|
|
std::vector<CIBProfile> _IBProfiles;
|
|
uint _CurIBLockCount;
|
|
uint _NumIBProfileFrame;
|
|
public:
|
|
NLMISC::TTicks _VolatileIBLockTime;
|
|
NLMISC::TTicks _VolatileVBLockTime;
|
|
private:
|
|
void appendIBLockProfile(NLMISC::TTicks time, CIndexBuffer *ib);
|
|
|
|
// VBHard profile
|
|
std::set<CVBDrvInfosD3D*> _VertexBufferHardSet;
|
|
|
|
// The render variables
|
|
public:
|
|
CRenderState _RenderStateCache[MaxRenderState];
|
|
CTextureState _TextureStateCache[MaxTexture][MaxTextureState];
|
|
CTextureIndexState _TextureIndexStateCache[MaxTexture];
|
|
CTexturePtrState _TexturePtrStateCache[MaxTexture];
|
|
CSamplerState _SamplerStateCache[MaxSampler][MaxSamplerState];
|
|
CMatrixState _MatrixCache[MaxMatrixState];
|
|
CVertexProgramPtrState _VertexProgramCache;
|
|
CPixelShaderPtrState _PixelShaderCache;
|
|
CVertexProgramConstantState _VertexProgramConstantCache[MaxVertexProgramConstantState];
|
|
CPixelShaderConstantState _PixelShaderConstantCache[MaxPixelShaderConstantState];
|
|
CVertexDeclState _VertexDeclCache;
|
|
CLightState _LightCache[MaxLight];
|
|
CRenderTargetState _RenderTarget;
|
|
CMaterialState _MaterialState;
|
|
private:
|
|
|
|
// last activation of vertex buffer / index buffer
|
|
NLMISC::CRefPtr<CIBDrvInfosD3D> _LastIndexBufferInfo;
|
|
|
|
// Vertex buffer cache
|
|
CVBState _VertexBufferCache;
|
|
uint _VertexBufferSize;
|
|
uint _VertexBufferOffset;
|
|
public:
|
|
bool _UseVertexColor;
|
|
private:
|
|
|
|
// Index buffer cache
|
|
CIBState _IndexBufferCache;
|
|
uint _IndexBufferOffset; // Current index buffer offset
|
|
CIndexBuffer::TFormat _CurrIndexBufferFormat; // updated at call to activeIndexBuffer
|
|
|
|
// The last vertex buffer needs vertex color
|
|
bool _FogEnabled;
|
|
|
|
// *** Internal resources
|
|
|
|
// Current render pass
|
|
uint _CurrentRenderPass;
|
|
|
|
|
|
// Volatile double buffers
|
|
CVolatileVertexBuffer *_VolatileVertexBufferRAM[2];
|
|
CVolatileVertexBuffer *_VolatileVertexBufferAGP[2];
|
|
CVolatileIndexBuffer *_VolatileIndexBuffer16RAM[2];
|
|
CVolatileIndexBuffer *_VolatileIndexBuffer16AGP[2];
|
|
CVolatileIndexBuffer *_VolatileIndexBuffer32RAM[2];
|
|
CVolatileIndexBuffer *_VolatileIndexBuffer32AGP[2];
|
|
|
|
// Special 16 bit index buffer for quads
|
|
IDirect3DIndexBuffer9 *_QuadIB;
|
|
|
|
// Vertex declaration list
|
|
std::list<CVertexDeclaration> _VertexDeclarationList;
|
|
|
|
// Pixel shader list
|
|
std::list<CNormalShaderDesc> _NormalPixelShaders[2];
|
|
|
|
// Quad indexes
|
|
CIndexBuffer _QuadIndexes;
|
|
CIndexBuffer _QuadIndexesAGP;
|
|
|
|
// The last setuped shader
|
|
CShader *_CurrentShader;
|
|
UINT _CurrentShaderPassCount;
|
|
public:
|
|
struct CTextureRef
|
|
{
|
|
CRefPtr<ITexture> NeLTexture;
|
|
LPDIRECT3DBASETEXTURE9 D3DTexture;
|
|
};
|
|
const std::vector<CTextureRef> &getCurrentShaderTextures() const { return _CurrentShaderTextures; }
|
|
private:
|
|
std::vector<CTextureRef> _CurrentShaderTextures;
|
|
|
|
// The last material setuped
|
|
CMaterial *_CurrentMaterial;
|
|
public:
|
|
CMaterialDrvInfosD3D *_CurrentMaterialInfo;
|
|
private:
|
|
|
|
// Optim: To not test change in Materials states if just texture has changed. Very useful for landscape.
|
|
uint32 _MaterialAllTextureTouchedFlag;
|
|
|
|
// The modified render variables list
|
|
CRenderVariable *_ModifiedRenderState;
|
|
|
|
// Internal shaders
|
|
CShader _ShaderLightmap0;
|
|
CShader _ShaderLightmap1;
|
|
CShader _ShaderLightmap2;
|
|
CShader _ShaderLightmap3;
|
|
CShader _ShaderLightmap4;
|
|
CShader _ShaderLightmap0Blend;
|
|
CShader _ShaderLightmap1Blend;
|
|
CShader _ShaderLightmap2Blend;
|
|
CShader _ShaderLightmap3Blend;
|
|
CShader _ShaderLightmap4Blend;
|
|
CShader _ShaderLightmap0X2;
|
|
CShader _ShaderLightmap1X2;
|
|
CShader _ShaderLightmap2X2;
|
|
CShader _ShaderLightmap3X2;
|
|
CShader _ShaderLightmap4X2;
|
|
CShader _ShaderLightmap0BlendX2;
|
|
CShader _ShaderLightmap1BlendX2;
|
|
CShader _ShaderLightmap2BlendX2;
|
|
CShader _ShaderLightmap3BlendX2;
|
|
CShader _ShaderLightmap4BlendX2;
|
|
CShader _ShaderCloud;
|
|
CShader _ShaderWaterNoDiffuse;
|
|
CShader _ShaderWaterDiffuse;
|
|
|
|
|
|
// Backup frame buffer
|
|
IDirect3DSurface9 *_BackBuffer;
|
|
|
|
// Static bitmap buffer to convert rgba to bgra
|
|
static std::vector<uint8> _TempBuffer;
|
|
|
|
// Version of the driver. Not the interface version!! Increment when implementation of the driver change.
|
|
static const uint32 ReleaseVersion;
|
|
|
|
uint64 _SwapBufferCounter;
|
|
|
|
// occlusion query
|
|
bool _OcclusionQuerySupported;
|
|
TOcclusionQueryList _OcclusionQueryList;
|
|
|
|
// depth range
|
|
float _DepthRangeNear;
|
|
float _DepthRangeFar;
|
|
//
|
|
bool _ScissorTouched;
|
|
uint8 _CurrentUVRouting[MaxTexture];
|
|
bool _MustRestoreLight;
|
|
D3DXMATRIX _D3DMatrixIdentity;
|
|
DWORD _FogColor;
|
|
|
|
// stencil buffer
|
|
bool _CurStencilTest;
|
|
DWORD _CurStencilFunc;
|
|
DWORD _CurStencilRef;
|
|
DWORD _CurStencilMask;
|
|
DWORD _CurStencilOpFail;
|
|
DWORD _CurStencilOpZFail;
|
|
DWORD _CurStencilOpZPass;
|
|
DWORD _CurStencilWriteMask;
|
|
|
|
public:
|
|
|
|
// private, for access by COcclusionQueryD3D
|
|
COcclusionQueryD3D *_CurrentOcclusionQuery;
|
|
|
|
// *** Lightmap Dynamic Light
|
|
// For Lightmap Dynamic Lighting
|
|
CLight _LightMapDynamicLight;
|
|
bool _LightMapDynamicLightEnabled;
|
|
bool _LightMapDynamicLightDirty;
|
|
CMaterial::TShader _CurrentMaterialSupportedShader;
|
|
// this is the backup of standard lighting (cause GL states may be modified by Lightmap Dynamic Lighting)
|
|
CLight _UserLight0;
|
|
bool _UserLightEnable[MaxLight];
|
|
// methods to enable / disable DX light, without affecting _LightMapDynamicLight*, or _UserLight0*
|
|
void setLightInternal(uint8 num, const CLight& light);
|
|
void enableLightInternal(uint8 num, bool enable);
|
|
// on/off Lights for LightMap mode: only the first light is enabled in lightmap mode
|
|
void setupLightMapDynamicLighting(bool enable);
|
|
|
|
TCullMode _CullMode;
|
|
|
|
bool _Lost;
|
|
bool _SceneBegun;
|
|
|
|
WORD _DesktopGammaRamp[256 * 3];
|
|
bool _DesktopGammaRampValid;
|
|
|
|
// for debug only
|
|
static bool _CacheTest[CacheTest_Count];
|
|
|
|
static std::vector<uint16> _QuadIndices; // tmp : quads indices -> to allow support of quads on devices that don't have 32 bit indices
|
|
|
|
// reset an index buffer and force it to be reallocated
|
|
void deleteIndexBuffer(CIBDrvInfosD3D *ib);
|
|
// Build 16 bit index buffer for quad
|
|
bool buildQuadIndexBuffer();
|
|
|
|
// Test if cursor is in the client area. always true when software cursor is used and window visible
|
|
// (displayed in software when DirectInput is used)
|
|
bool isSystemCursorInClientArea();
|
|
|
|
// Check if RGBA cursors are supported
|
|
bool isAlphaBlendedCursorSupported();
|
|
|
|
// Update cursor appearance
|
|
void updateCursor(bool forceRebuild = false);
|
|
|
|
// Create default cursors
|
|
void createCursors();
|
|
|
|
// Release all cursors
|
|
void releaseCursors();
|
|
|
|
// Convert a NLMISC::CBitmap to nlCursor
|
|
bool convertBitmapToCursor(const NLMISC::CBitmap &bitmap, nlCursor &cursor, uint iconWidth, uint iconHeight, uint iconDepth, const NLMISC::CRGBA &col, sint hotSpotX, sint hotSpotY);
|
|
|
|
// build a cursor from src, src should have the same size that the hardware cursor
|
|
// or a assertion is thrown
|
|
nlCursor buildCursor(const NLMISC::CBitmap &src, NLMISC::CRGBA col, uint8 rot, sint hotSpotX, sint hotSpotY);
|
|
|
|
// reset the cursor shape to the system arrow
|
|
void setSystemArrow();
|
|
|
|
bool convertBitmapToIcon(const NLMISC::CBitmap &bitmap, HICON &icon, uint iconWidth, uint iconHeight, uint iconDepth, const NLMISC::CRGBA &col = NLMISC::CRGBA::White, sint hotSpotX = 0, sint hotSpotY = 0, bool cursor = false);
|
|
|
|
virtual bool copyTextToClipboard(const ucstring &text);
|
|
virtual bool pasteTextFromClipboard(ucstring &text);
|
|
|
|
public:
|
|
#ifdef NL_DEBUG
|
|
std::set<CVBDrvInfosD3D *> _LockedBuffers;
|
|
#endif
|
|
|
|
emptyProc ExitFunc;
|
|
|
|
bool beginScene()
|
|
{
|
|
nlassert(!_SceneBegun);
|
|
if (_DeviceInterface->BeginScene() != D3D_OK) return false;
|
|
_SceneBegun = true;
|
|
return true;
|
|
}
|
|
|
|
bool endScene()
|
|
{
|
|
nlassert(_SceneBegun);
|
|
if (_DeviceInterface->EndScene() != D3D_OK) return false;
|
|
_SceneBegun = false;
|
|
return true;
|
|
}
|
|
|
|
bool hasSceneBegun() const { return _SceneBegun; }
|
|
|
|
// Clip the wanted rectangle with window. return true if rect is not NULL.
|
|
bool clipRect(NLMISC::CRect &rect);
|
|
|
|
};
|
|
|
|
#define NL_D3DCOLOR_RGBA(rgba) (D3DCOLOR_ARGB(rgba.A,rgba.R,rgba.G,rgba.B))
|
|
#define D3DCOLOR_NL_RGBA(rgba) (NLMISC::CRGBA((uint8)((rgba>>16)&0xff), (uint8)((rgba>>8)&0xff), (uint8)(rgba&0xff), (uint8)((rgba>>24)&0xff)))
|
|
#define NL_D3DCOLORVALUE_RGBA(dest,rgba) \
|
|
dest.a=(float)rgba.A*(1.f/255.f);dest.r=(float)rgba.R*(1.f/255.f);dest.g=(float)rgba.G*(1.f/255.f);dest.b=(float)rgba.B*(1.f/255.f);
|
|
#define NL_D3D_MATRIX(d3d_mt,nl_mt) \
|
|
{ const float *nl_mt_ptr = nl_mt.get(); \
|
|
d3d_mt._11 = nl_mt_ptr[0]; \
|
|
d3d_mt._21 = nl_mt_ptr[4]; \
|
|
d3d_mt._31 = nl_mt_ptr[8]; \
|
|
d3d_mt._41 = nl_mt_ptr[12]; \
|
|
d3d_mt._12 = nl_mt_ptr[1]; \
|
|
d3d_mt._22 = nl_mt_ptr[5]; \
|
|
d3d_mt._32 = nl_mt_ptr[9]; \
|
|
d3d_mt._42 = nl_mt_ptr[13]; \
|
|
d3d_mt._13 = nl_mt_ptr[2]; \
|
|
d3d_mt._23 = nl_mt_ptr[6]; \
|
|
d3d_mt._33 = nl_mt_ptr[10]; \
|
|
d3d_mt._43 = nl_mt_ptr[14]; \
|
|
d3d_mt._14 = nl_mt_ptr[3]; \
|
|
d3d_mt._24 = nl_mt_ptr[7]; \
|
|
d3d_mt._34 = nl_mt_ptr[11]; \
|
|
d3d_mt._44 = nl_mt_ptr[15]; }
|
|
// build matrix for texture 2D transform (special case in D3D)
|
|
#define NL_D3D_TEX2D_MATRIX(d3d_mt,nl_mt) \
|
|
{ const float *nl_mt_ptr = nl_mt.get(); \
|
|
d3d_mt._11 = nl_mt_ptr[0]; \
|
|
d3d_mt._12 = nl_mt_ptr[1]; \
|
|
d3d_mt._13 = nl_mt_ptr[3]; \
|
|
d3d_mt._14 = 0.f; \
|
|
d3d_mt._21 = nl_mt_ptr[4]; \
|
|
d3d_mt._22 = nl_mt_ptr[5]; \
|
|
d3d_mt._23 = nl_mt_ptr[7]; \
|
|
d3d_mt._24 = 0.f; \
|
|
d3d_mt._31 = nl_mt_ptr[12]; \
|
|
d3d_mt._32 = nl_mt_ptr[13]; \
|
|
d3d_mt._33 = nl_mt_ptr[15]; \
|
|
d3d_mt._34 = 0.f; \
|
|
d3d_mt._41 = 0.f; \
|
|
d3d_mt._42 = 0.f; \
|
|
d3d_mt._43 = 0.f; \
|
|
d3d_mt._44 = 1.f; }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#define NL_D3DVECTOR_VECTOR(dest,vect) dest.x=(vect).x;dest.y=(vect).y;dest.z=(vect).z;
|
|
#define FTODW(f) (*((DWORD*)&(f)))
|
|
#define D3DCOLOR_FLOATS(floats,rgba) {floats[0]=(float)((rgba>>16)&0xff) * (1.f/255.f);floats[1]=(float)((rgba>>8)&0xff) * (1.f/255.f);\
|
|
floats[2]=(float)(rgba&0xff) * (1.f/255.f);floats[3]=(float)((rgba>>24)&0xff) * (1.f/255.f);}
|
|
#define NL_FLOATS(floats,rgba) {floats[0] = (float)rgba.R * (1.f/255.f);floats[1] = (float)rgba.G * (1.f/255.f);\
|
|
floats[2] = (float)rgba.B * (1.f/255.f);floats[3] = (float)rgba.A * (1.f/255.f);}
|
|
|
|
// ***************************************************************************
|
|
// nbPixels is pixel count, not a byte count !
|
|
inline void copyRGBA2BGRA (uint32 *dest, const uint32 *src, uint nbPixels)
|
|
{
|
|
H_AUTO_D3D(CDriverD3D_copyRGBA2BGRA);
|
|
while (nbPixels != 0)
|
|
{
|
|
register uint32 color = *src;
|
|
*dest = (color & 0xff00ff00) | ((color&0xff)<<16) | ((color&0xff0000)>>16);
|
|
dest++;
|
|
src++;
|
|
nbPixels--;
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
// set a D3DCOLORVALUE
|
|
inline void setColor(D3DCOLORVALUE &dest, float r, float g, float b, float a)
|
|
{
|
|
dest.r = r;
|
|
dest.g = g;
|
|
dest.b = b;
|
|
dest.a = a;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
// set a D3DCOLORVALUE from & D3DCOLOR
|
|
inline void setColor(D3DCOLORVALUE &dest, const D3DCOLOR &src)
|
|
{
|
|
dest.r = (1.f / 255.f) * ((src >> 16) & 0xff);
|
|
dest.g = (1.f / 255.f) * ((src >> 8) & 0xff);
|
|
dest.b = (1.f / 255.f) * (src & 0xff);
|
|
dest.a = (1.f / 255.f) * (src >> 24);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
// set a D3DCOLORVALUE from a CRGBA
|
|
inline void setColor(D3DCOLORVALUE &dest, const NLMISC::CRGBA &src)
|
|
{
|
|
dest.r = (1.f / 255.f) * src.R;
|
|
dest.g = (1.f / 255.f) * src.G;
|
|
dest.b = (1.f / 255.f) * src.B;
|
|
dest.a = (1.f / 255.f) * src.A;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void fillQuadIndexes (uint16 *indexes, uint first, uint last);
|
|
|
|
} // NL3D
|
|
|
|
#ifndef NL_STATIC
|
|
extern HINSTANCE HInstDLL;
|
|
#endif
|
|
|
|
#endif // NL_DRIVER_DIRECT3D_H
|