1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-05 18:16:23 +03:00

0029283: Visualization - allow defining more than 8 light sources

OpenGl_ShaderManager now overrides THE_MAX_LIGHTS within built-in shading programs
so that maximum number of lights is now limited only by OpenGL hardware
(e.g. length of GLSL program, number of defined uniforms, result performance, etc.).
THE_MAX_CLIP_PLANES is now also defined by OpenGl_ShaderManager,
so that unused lights and clipping planes do not reserve extra uniforms in GLSL programs.

V3d_View::SetLightOn() does not throw exception anymore, when the number of lights exceeds 8.
Instead, OpenGl_ShaderManager::PushLightSourceState() emits warning
in case of usage of FFP providing consistent behavior with Clipping Planes number limit.
This commit is contained in:
kgv 2017-10-31 20:02:13 +03:00 committed by bugmaster
parent 182bd7bc42
commit daf73ab7c9
13 changed files with 266 additions and 80 deletions

View File

@ -77,6 +77,8 @@ const TCollection_AsciiString& Graphic3d_ShaderProgram::ShadersFolder()
// purpose : Creates new empty program object // purpose : Creates new empty program object
// ======================================================================= // =======================================================================
Graphic3d_ShaderProgram::Graphic3d_ShaderProgram() Graphic3d_ShaderProgram::Graphic3d_ShaderProgram()
: myNbLightsMax (THE_MAX_LIGHTS_DEFAULT),
myNbClipPlanesMax (THE_MAX_CLIP_PLANES_DEFAULT)
{ {
myID = TCollection_AsciiString ("Graphic3d_ShaderProgram_") myID = TCollection_AsciiString ("Graphic3d_ShaderProgram_")
+ TCollection_AsciiString (Standard_Atomic_Increment (&THE_PROGRAM_OBJECT_COUNTER)); + TCollection_AsciiString (Standard_Atomic_Increment (&THE_PROGRAM_OBJECT_COUNTER));

View File

@ -34,6 +34,14 @@ typedef NCollection_Sequence<Handle(Graphic3d_ShaderAttribute)> Graphic3d_Shader
//! This class is responsible for managing shader programs. //! This class is responsible for managing shader programs.
class Graphic3d_ShaderProgram : public Standard_Transient class Graphic3d_ShaderProgram : public Standard_Transient
{ {
DEFINE_STANDARD_RTTIEXT(Graphic3d_ShaderProgram, Standard_Transient)
public:
//! Default value of THE_MAX_LIGHTS macros within GLSL program (see Declarations.glsl).
static const Standard_Integer THE_MAX_LIGHTS_DEFAULT = 8;
//! Default value of THE_MAX_CLIP_PLANES macros within GLSL program (see Declarations.glsl).
static const Standard_Integer THE_MAX_CLIP_PLANES_DEFAULT = 8;
public: public:
@ -61,6 +69,32 @@ public:
//! @endcode //! @endcode
void SetHeader (const TCollection_AsciiString& theHeader) { myHeader = theHeader; } void SetHeader (const TCollection_AsciiString& theHeader) { myHeader = theHeader; }
//! Append line to GLSL header.
void AppendToHeader (const TCollection_AsciiString& theHeaderLine)
{
if (!myHeader.IsEmpty())
{
myHeader += "\n";
}
myHeader += theHeaderLine;
}
//! Return the length of array of light sources (THE_MAX_LIGHTS),
//! to be used for initialization occLightSources.
//! Default value is THE_MAX_LIGHTS_DEFAULT.
Standard_Integer NbLightsMax() const { return myNbLightsMax; }
//! Specify the length of array of light sources (THE_MAX_LIGHTS).
void SetNbLightsMax (Standard_Integer theNbLights) { myNbLightsMax = theNbLights; }
//! Return the length of array of clipping planes (THE_MAX_CLIP_PLANES),
//! to be used for initialization occClipPlaneEquations.
//! Default value is THE_MAX_CLIP_PLANES_DEFAULT.
Standard_Integer NbClipPlanesMax() const { return myNbClipPlanesMax; }
//! Specify the length of array of clipping planes (THE_MAX_CLIP_PLANES).
void SetNbClipPlanesMax (Standard_Integer theNbPlanes) { myNbClipPlanesMax = theNbPlanes; }
//! Attaches shader object to the program object. //! Attaches shader object to the program object.
Standard_EXPORT Standard_Boolean AttachShader (const Handle(Graphic3d_ShaderObject)& theShader); Standard_EXPORT Standard_Boolean AttachShader (const Handle(Graphic3d_ShaderObject)& theShader);
@ -121,10 +155,6 @@ public:
//! @return the root folder with default GLSL programs. //! @return the root folder with default GLSL programs.
Standard_EXPORT static const TCollection_AsciiString& ShadersFolder(); Standard_EXPORT static const TCollection_AsciiString& ShadersFolder();
public:
DEFINE_STANDARD_RTTIEXT(Graphic3d_ShaderProgram,Standard_Transient)
private: private:
TCollection_AsciiString myID; //!< the unique identifier of program object TCollection_AsciiString myID; //!< the unique identifier of program object
@ -132,6 +162,8 @@ private:
Graphic3d_ShaderVariableList myVariables; //!< the list of custom uniform variables Graphic3d_ShaderVariableList myVariables; //!< the list of custom uniform variables
Graphic3d_ShaderAttributeList myAttributes; //!< the list of custom vertex attributes Graphic3d_ShaderAttributeList myAttributes; //!< the list of custom vertex attributes
TCollection_AsciiString myHeader; //!< GLSL header with version code and used extensions TCollection_AsciiString myHeader; //!< GLSL header with version code and used extensions
Standard_Integer myNbLightsMax; //!< length of array of light sources (THE_MAX_LIGHTS)
Standard_Integer myNbClipPlanesMax; //!< length of array of clipping planes (THE_MAX_CLIP_PLANES)
}; };

View File

@ -406,7 +406,7 @@ Standard_Integer OpenGl_GraphicDriver::InquireLimit (const Graphic3d_TypeOfLimit
switch (theType) switch (theType)
{ {
case Graphic3d_TypeOfLimit_MaxNbLights: case Graphic3d_TypeOfLimit_MaxNbLights:
return OpenGLMaxLights; return Graphic3d_ShaderProgram::THE_MAX_LIGHTS_DEFAULT;
case Graphic3d_TypeOfLimit_MaxNbClipPlanes: case Graphic3d_TypeOfLimit_MaxNbClipPlanes:
return !aCtx.IsNull() ? aCtx->MaxClipPlanes() : 0; return !aCtx.IsNull() ? aCtx->MaxClipPlanes() : 0;
case Graphic3d_TypeOfLimit_MaxNbViews: case Graphic3d_TypeOfLimit_MaxNbViews:

View File

@ -33,9 +33,6 @@ IMPLEMENT_STANDARD_RTTIEXT(OpenGl_ShaderManager,Standard_Transient)
namespace namespace
{ {
//! Clipping planes limit (see the same definition in Declarations.glsl).
static const Standard_Size THE_MAX_CLIP_PLANES = 8;
#define EOL "\n" #define EOL "\n"
//! Definition of TexCoord varying. //! Definition of TexCoord varying.
@ -596,70 +593,85 @@ void OpenGl_ShaderManager::PushLightSourceState (const Handle(OpenGl_ShaderProgr
return; return;
} }
if (myContext->core11 != NULL) GLenum aLightGlId = GL_LIGHT0;
OpenGl_Vec4 anAmbient (0.0f, 0.0f, 0.0f, 0.0f);
const OpenGl_Mat4 aModelView = myWorldViewState.WorldViewMatrix() * myModelWorldState.ModelWorldMatrix();
if (myLightSourceState.LightSources() != NULL)
{ {
GLenum aLightGlId = GL_LIGHT0; for (Graphic3d_ListOfCLight::Iterator aLightIt (*myLightSourceState.LightSources()); aLightIt.More(); aLightIt.Next())
OpenGl_Vec4 anAmbient (0.0f, 0.0f, 0.0f, 0.0f);
const OpenGl_Mat4 aModelView = myWorldViewState.WorldViewMatrix() * myModelWorldState.ModelWorldMatrix();
for (OpenGl_ListOfLight::Iterator aLightIt (*myLightSourceState.LightSources()); aLightIt.More(); aLightIt.Next())
{ {
const OpenGl_Light& aLight = aLightIt.Value(); const Graphic3d_CLight& aLight = aLightIt.Value();
if (aLight.Type == Graphic3d_TOLS_AMBIENT) if (aLight.Type == Graphic3d_TOLS_AMBIENT)
{ {
anAmbient += aLight.Color; anAmbient += aLight.Color;
continue; continue;
} }
else if (aLightGlId > GL_LIGHT7) // OpenGLMaxLights - only 8 lights in OpenGL... else if (aLightGlId > GL_LIGHT7) // only 8 lights in FFP...
{ {
myContext->PushMessage (GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_PORTABILITY, 0, GL_DEBUG_SEVERITY_MEDIUM,
"Warning: light sources limit (8) has been exceeded within Fixed-function pipeline.");
continue; continue;
} }
bindLight (aLightIt.Value(), aLightGlId, aModelView, myContext); bindLight (aLight, aLightGlId, aModelView, myContext);
++aLightGlId; ++aLightGlId;
} }
}
// apply accumulated ambient color // apply accumulated ambient color
anAmbient.a() = 1.0f; anAmbient.a() = 1.0f;
myContext->core11->glLightModelfv (GL_LIGHT_MODEL_AMBIENT, anAmbient.GetData()); myContext->core11->glLightModelfv (GL_LIGHT_MODEL_AMBIENT, anAmbient.GetData());
// GL_LIGHTING is managed by drawers to switch between shaded / no lighting output, // GL_LIGHTING is managed by drawers to switch between shaded / no lighting output,
// therefore managing the state here does not have any effect - do it just for consistency. // therefore managing the state here does not have any effect - do it just for consistency.
if (aLightGlId != GL_LIGHT0) if (aLightGlId != GL_LIGHT0)
{ {
::glEnable (GL_LIGHTING); ::glEnable (GL_LIGHTING);
} }
else else
{ {
::glDisable (GL_LIGHTING); ::glDisable (GL_LIGHTING);
} }
// switch off unused lights // switch off unused lights
for (; aLightGlId <= GL_LIGHT7; ++aLightGlId) for (; aLightGlId <= GL_LIGHT7; ++aLightGlId)
{ {
::glDisable (aLightGlId); ::glDisable (aLightGlId);
}
} }
#endif #endif
return; return;
} }
for (Standard_Integer aLightIt = 0; aLightIt < OpenGLMaxLights; ++aLightIt) const Standard_Integer aNbLightsMax = theProgram->NbLightsMax();
const GLint anAmbientLoc = theProgram->GetStateLocation (OpenGl_OCC_LIGHT_AMBIENT);
if (aNbLightsMax == 0
&& anAmbientLoc == OpenGl_ShaderProgram::INVALID_LOCATION)
{ {
myLightTypeArray[aLightIt].Type = -1; return;
} }
const Standard_Integer aLightsDefNb = Min (myLightSourceState.LightSources()->Size(), OpenGLMaxLights); if (myLightTypeArray.Size() < aNbLightsMax)
if (aLightsDefNb < 1) {
myLightTypeArray .Resize (0, aNbLightsMax - 1, false);
myLightParamsArray.Resize (0, aNbLightsMax - 1, false);
}
for (Standard_Integer aLightIt = 0; aLightIt < aNbLightsMax; ++aLightIt)
{
myLightTypeArray.ChangeValue (aLightIt).Type = -1;
}
if (myLightSourceState.LightSources() == NULL
|| myLightSourceState.LightSources()->IsEmpty())
{ {
theProgram->SetUniform (myContext, theProgram->SetUniform (myContext,
theProgram->GetStateLocation (OpenGl_OCC_LIGHT_SOURCE_COUNT), theProgram->GetStateLocation (OpenGl_OCC_LIGHT_SOURCE_COUNT),
0); 0);
theProgram->SetUniform (myContext, theProgram->SetUniform (myContext,
theProgram->GetStateLocation (OpenGl_OCC_LIGHT_AMBIENT), anAmbientLoc,
OpenGl_Vec4 (0.0f, 0.0f, 0.0f, 0.0f)); OpenGl_Vec4 (0.0f, 0.0f, 0.0f, 0.0f));
theProgram->SetUniform (myContext, theProgram->SetUniform (myContext,
theProgram->GetStateLocation (OpenGl_OCC_LIGHT_SOURCE_TYPES), theProgram->GetStateLocation (OpenGl_OCC_LIGHT_SOURCE_TYPES),
OpenGLMaxLights * OpenGl_ShaderLightType::NbOfVec2i(), aNbLightsMax * OpenGl_ShaderLightType::NbOfVec2i(),
myLightTypeArray[0].Packed()); myLightTypeArray.First().Packed());
return; return;
} }
@ -673,17 +685,21 @@ void OpenGl_ShaderManager::PushLightSourceState (const Handle(OpenGl_ShaderProgr
anAmbient += aLight.Color; anAmbient += aLight.Color;
continue; continue;
} }
else if (aLightsNb >= OpenGLMaxLights) else if (aLightsNb >= aNbLightsMax)
{ {
if (aNbLightsMax != 0)
{
myContext->PushMessage (GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_PORTABILITY, 0, GL_DEBUG_SEVERITY_MEDIUM,
TCollection_AsciiString("Warning: light sources limit (") + aNbLightsMax + ") has been exceeded.");
}
continue; continue;
} }
OpenGl_ShaderLightType& aLightType = myLightTypeArray[aLightsNb]; OpenGl_ShaderLightType& aLightType = myLightTypeArray.ChangeValue (aLightsNb);
OpenGl_ShaderLightParameters& aLightParams = myLightParamsArray.ChangeValue (aLightsNb);
aLightType.Type = aLight.Type; aLightType.Type = aLight.Type;
aLightType.IsHeadlight = aLight.IsHeadlight; aLightType.IsHeadlight = aLight.IsHeadlight;
aLightParams.Color = aLight.Color;
OpenGl_ShaderLightParameters& aLightParams = myLightParamsArray[aLightsNb];
aLightParams.Color = aLight.Color;
if (aLight.Type == Graphic3d_TOLS_DIRECTIONAL) if (aLight.Type == Graphic3d_TOLS_DIRECTIONAL)
{ {
aLightParams.Position = -aLight.Direction; aLightParams.Position = -aLight.Direction;
@ -715,18 +731,18 @@ void OpenGl_ShaderManager::PushLightSourceState (const Handle(OpenGl_ShaderProgr
theProgram->GetStateLocation (OpenGl_OCC_LIGHT_SOURCE_COUNT), theProgram->GetStateLocation (OpenGl_OCC_LIGHT_SOURCE_COUNT),
aLightsNb); aLightsNb);
theProgram->SetUniform (myContext, theProgram->SetUniform (myContext,
theProgram->GetStateLocation (OpenGl_OCC_LIGHT_AMBIENT), anAmbientLoc,
anAmbient); anAmbient);
theProgram->SetUniform (myContext, theProgram->SetUniform (myContext,
theProgram->GetStateLocation (OpenGl_OCC_LIGHT_SOURCE_TYPES), theProgram->GetStateLocation (OpenGl_OCC_LIGHT_SOURCE_TYPES),
OpenGLMaxLights * OpenGl_ShaderLightType::NbOfVec2i(), aNbLightsMax * OpenGl_ShaderLightType::NbOfVec2i(),
myLightTypeArray[0].Packed()); myLightTypeArray.First().Packed());
if (aLightsNb > 0) if (aLightsNb > 0)
{ {
theProgram->SetUniform (myContext, theProgram->SetUniform (myContext,
theProgram->GetStateLocation (OpenGl_OCC_LIGHT_SOURCE_PARAMS), theProgram->GetStateLocation (OpenGl_OCC_LIGHT_SOURCE_PARAMS),
aLightsNb * OpenGl_ShaderLightParameters::NbOfVec4(), aLightsNb * OpenGl_ShaderLightParameters::NbOfVec4(),
myLightParamsArray[0].Packed()); myLightParamsArray.First().Packed());
} }
} }
@ -907,8 +923,12 @@ void OpenGl_ShaderManager::PushClippingState (const Handle(OpenGl_ShaderProgram)
return; return;
} }
const Standard_Integer aNbMaxPlanes = Min (myContext->MaxClipPlanes(), THE_MAX_CLIP_PLANES); const Standard_Integer aNbMaxPlanes = myContext->MaxClipPlanes();
OpenGl_Vec4d anEquations[THE_MAX_CLIP_PLANES]; if (myClipPlaneArrayFfp.Size() < aNbMaxPlanes)
{
myClipPlaneArrayFfp.Resize (0, aNbMaxPlanes - 1, false);
}
Standard_Integer aPlaneId = 0; Standard_Integer aPlaneId = 0;
Standard_Boolean toRestoreModelView = Standard_False; Standard_Boolean toRestoreModelView = Standard_False;
for (OpenGl_ClippingIterator aPlaneIter (myContext->Clipping()); aPlaneIter.More(); aPlaneIter.Next()) for (OpenGl_ClippingIterator aPlaneIter (myContext->Clipping()); aPlaneIter.More(); aPlaneIter.Next())
@ -920,14 +940,13 @@ void OpenGl_ShaderManager::PushClippingState (const Handle(OpenGl_ShaderProgram)
} }
else if (aPlaneId >= aNbMaxPlanes) else if (aPlaneId >= aNbMaxPlanes)
{ {
myContext->PushMessage (GL_DEBUG_SOURCE_APPLICATION, myContext->PushMessage (GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_PORTABILITY, 0, GL_DEBUG_SEVERITY_MEDIUM,
GL_DEBUG_TYPE_PORTABILITY, 0, GL_DEBUG_SEVERITY_MEDIUM,
TCollection_ExtendedString("Warning: clipping planes limit (") + aNbMaxPlanes + ") has been exceeded."); TCollection_ExtendedString("Warning: clipping planes limit (") + aNbMaxPlanes + ") has been exceeded.");
break; break;
} }
const Graphic3d_ClipPlane::Equation& anEquation = aPlane->GetEquation(); const Graphic3d_ClipPlane::Equation& anEquation = aPlane->GetEquation();
OpenGl_Vec4d& aPlaneEq = anEquations[aPlaneId]; OpenGl_Vec4d& aPlaneEq = myClipPlaneArrayFfp.ChangeValue (aPlaneId);
aPlaneEq.x() = anEquation.x(); aPlaneEq.x() = anEquation.x();
aPlaneEq.y() = anEquation.y(); aPlaneEq.y() = anEquation.y();
aPlaneEq.z() = anEquation.z(); aPlaneEq.z() = anEquation.z();
@ -976,7 +995,8 @@ void OpenGl_ShaderManager::PushClippingState (const Handle(OpenGl_ShaderProgram)
return; return;
} }
const GLint aNbPlanes = Min (myContext->Clipping().NbClippingOrCappingOn(), THE_MAX_CLIP_PLANES); const Standard_Integer aNbClipPlanesMax = theProgram->NbClipPlanesMax();
const GLint aNbPlanes = Min (myContext->Clipping().NbClippingOrCappingOn(), aNbClipPlanesMax);
theProgram->SetUniform (myContext, theProgram->SetUniform (myContext,
theProgram->GetStateLocation (OpenGl_OCC_CLIP_PLANE_COUNT), theProgram->GetStateLocation (OpenGl_OCC_CLIP_PLANE_COUNT),
aNbPlanes); aNbPlanes);
@ -985,8 +1005,12 @@ void OpenGl_ShaderManager::PushClippingState (const Handle(OpenGl_ShaderProgram)
return; return;
} }
OpenGl_Vec4 anEquations[THE_MAX_CLIP_PLANES]; if (myClipPlaneArray.Size() < aNbClipPlanesMax)
GLuint aPlaneId = 0; {
myClipPlaneArray.Resize (0, aNbClipPlanesMax - 1, false);
}
Standard_Integer aPlaneId = 0;
for (OpenGl_ClippingIterator aPlaneIter (myContext->Clipping()); aPlaneIter.More(); aPlaneIter.Next()) for (OpenGl_ClippingIterator aPlaneIter (myContext->Clipping()); aPlaneIter.More(); aPlaneIter.Next())
{ {
const Handle(Graphic3d_ClipPlane)& aPlane = aPlaneIter.Value(); const Handle(Graphic3d_ClipPlane)& aPlane = aPlaneIter.Value();
@ -994,16 +1018,15 @@ void OpenGl_ShaderManager::PushClippingState (const Handle(OpenGl_ShaderProgram)
{ {
continue; continue;
} }
else if (aPlaneId >= THE_MAX_CLIP_PLANES) else if (aPlaneId >= aNbClipPlanesMax)
{ {
myContext->PushMessage (GL_DEBUG_SOURCE_APPLICATION, myContext->PushMessage (GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_PORTABILITY, 0, GL_DEBUG_SEVERITY_MEDIUM,
GL_DEBUG_TYPE_PORTABILITY, 0, GL_DEBUG_SEVERITY_MEDIUM, TCollection_AsciiString("Warning: clipping planes limit (") + aNbClipPlanesMax + ") has been exceeded.");
"Warning: clipping planes limit (8) has been exceeded.");
break; break;
} }
const Graphic3d_ClipPlane::Equation& anEquation = aPlane->GetEquation(); const Graphic3d_ClipPlane::Equation& anEquation = aPlane->GetEquation();
OpenGl_Vec4& aPlaneEq = anEquations[aPlaneId]; OpenGl_Vec4& aPlaneEq = myClipPlaneArray.ChangeValue (aPlaneId);
aPlaneEq.x() = float(anEquation.x()); aPlaneEq.x() = float(anEquation.x());
aPlaneEq.y() = float(anEquation.y()); aPlaneEq.y() = float(anEquation.y());
aPlaneEq.z() = float(anEquation.z()); aPlaneEq.z() = float(anEquation.z());
@ -1017,7 +1040,7 @@ void OpenGl_ShaderManager::PushClippingState (const Handle(OpenGl_ShaderProgram)
++aPlaneId; ++aPlaneId;
} }
theProgram->SetUniform (myContext, aLocEquations, THE_MAX_CLIP_PLANES, anEquations); theProgram->SetUniform (myContext, aLocEquations, aNbClipPlanesMax, &myClipPlaneArray.First());
} }
// ======================================================================= // =======================================================================
@ -1173,6 +1196,8 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramFont()
aProgramSrc->SetHeader ("#version 300 es"); aProgramSrc->SetHeader ("#version 300 es");
} }
#endif #endif
aProgramSrc->SetNbLightsMax (0);
aProgramSrc->SetNbClipPlanesMax (0);
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert));
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag));
TCollection_AsciiString aKey; TCollection_AsciiString aKey;
@ -1240,6 +1265,8 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramFboBlit()
aProgramSrc->SetHeader ("#version 150"); aProgramSrc->SetHeader ("#version 150");
} }
#endif #endif
aProgramSrc->SetNbLightsMax (0);
aProgramSrc->SetNbClipPlanesMax (0);
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert));
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag));
TCollection_AsciiString aKey; TCollection_AsciiString aKey;
@ -1328,6 +1355,8 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramOitCompositing (const St
#endif #endif
} }
aProgramSrc->SetNbLightsMax (0);
aProgramSrc->SetNbClipPlanesMax (0);
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert));
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag));
TCollection_AsciiString aKey; TCollection_AsciiString aKey;
@ -1463,6 +1492,8 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramFlat (Handle(OpenGl_Shad
aSrcFragExtraOut += EOL"THE_SHADER_IN vec4 VertColor;"; aSrcFragExtraOut += EOL"THE_SHADER_IN vec4 VertColor;";
aSrcFragGetColor = EOL"vec4 getColor(void) { return VertColor; }"; aSrcFragGetColor = EOL"vec4 getColor(void) { return VertColor; }";
} }
int aNbClipPlanes = 0;
if ((theBits & OpenGl_PO_ClipPlanesN) != 0) if ((theBits & OpenGl_PO_ClipPlanesN) != 0)
{ {
aSrcVertExtraOut += aSrcVertExtraOut +=
@ -1477,14 +1508,17 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramFlat (Handle(OpenGl_Shad
if ((theBits & OpenGl_PO_ClipPlanes1) != 0) if ((theBits & OpenGl_PO_ClipPlanes1) != 0)
{ {
aNbClipPlanes = 1;
aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_1; aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_1;
} }
else if ((theBits & OpenGl_PO_ClipPlanes2) != 0) else if ((theBits & OpenGl_PO_ClipPlanes2) != 0)
{ {
aNbClipPlanes = 2;
aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_2; aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_2;
} }
else else
{ {
aNbClipPlanes = Graphic3d_ShaderProgram::THE_MAX_CLIP_PLANES_DEFAULT;
aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_N; aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_N;
} }
} }
@ -1575,6 +1609,8 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramFlat (Handle(OpenGl_Shad
aProgramSrc->SetHeader ("#version 300 es"); aProgramSrc->SetHeader ("#version 300 es");
} }
#endif #endif
aProgramSrc->SetNbLightsMax (0);
aProgramSrc->SetNbClipPlanesMax (aNbClipPlanes);
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert));
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag));
@ -1625,12 +1661,14 @@ TCollection_AsciiString OpenGl_ShaderManager::pointSpriteShadingSrc (const TColl
// function : stdComputeLighting // function : stdComputeLighting
// purpose : // purpose :
// ======================================================================= // =======================================================================
TCollection_AsciiString OpenGl_ShaderManager::stdComputeLighting (const Standard_Boolean theHasVertColor) TCollection_AsciiString OpenGl_ShaderManager::stdComputeLighting (Standard_Integer& theNbLights,
Standard_Boolean theHasVertColor)
{ {
Standard_Integer aLightsMap[Graphic3d_TOLS_SPOT + 1] = { 0, 0, 0, 0 }; Standard_Integer aLightsMap[Graphic3d_TOLS_SPOT + 1] = { 0, 0, 0, 0 };
TCollection_AsciiString aLightsFunc, aLightsLoop; TCollection_AsciiString aLightsFunc, aLightsLoop;
const OpenGl_ListOfLight* aLights = myLightSourceState.LightSources(); const OpenGl_ListOfLight* aLights = myLightSourceState.LightSources();
if (aLights != NULL) if (aLights != NULL)
theNbLights = 0;
{ {
Standard_Integer anIndex = 0; Standard_Integer anIndex = 0;
for (OpenGl_ListOfLight::Iterator aLightIter (*aLights); aLightIter.More(); aLightIter.Next(), ++anIndex) for (OpenGl_ListOfLight::Iterator aLightIter (*aLights); aLightIter.More(); aLightIter.Next(), ++anIndex)
@ -1652,6 +1690,7 @@ TCollection_AsciiString OpenGl_ShaderManager::stdComputeLighting (const Standard
} }
aLightsMap[aLightIter.Value().Type] += 1; aLightsMap[aLightIter.Value().Type] += 1;
} }
theNbLights = anIndex;
const Standard_Integer aNbLoopLights = aLightsMap[Graphic3d_TOLS_DIRECTIONAL] const Standard_Integer aNbLoopLights = aLightsMap[Graphic3d_TOLS_DIRECTIONAL]
+ aLightsMap[Graphic3d_TOLS_POSITIONAL] + aLightsMap[Graphic3d_TOLS_POSITIONAL]
+ aLightsMap[Graphic3d_TOLS_SPOT]; + aLightsMap[Graphic3d_TOLS_SPOT];
@ -1762,6 +1801,7 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramGouraud (Handle(OpenGl_S
aSrcVertColor = EOL"vec4 getVertColor(void) { return occVertColor; }"; aSrcVertColor = EOL"vec4 getVertColor(void) { return occVertColor; }";
} }
int aNbClipPlanes = 0;
if ((theBits & OpenGl_PO_ClipPlanesN) != 0) if ((theBits & OpenGl_PO_ClipPlanesN) != 0)
{ {
aSrcVertExtraOut += aSrcVertExtraOut +=
@ -1776,14 +1816,17 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramGouraud (Handle(OpenGl_S
if ((theBits & OpenGl_PO_ClipPlanes1) != 0) if ((theBits & OpenGl_PO_ClipPlanes1) != 0)
{ {
aNbClipPlanes = 1;
aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_1; aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_1;
} }
else if ((theBits & OpenGl_PO_ClipPlanes2) != 0) else if ((theBits & OpenGl_PO_ClipPlanes2) != 0)
{ {
aNbClipPlanes = 2;
aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_2; aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_2;
} }
else else
{ {
aNbClipPlanes = Graphic3d_ShaderProgram::THE_MAX_CLIP_PLANES_DEFAULT;
aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_N; aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_N;
} }
} }
@ -1792,7 +1835,8 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramGouraud (Handle(OpenGl_S
aSrcFragWriteOit += THE_FRAG_write_oit_buffers; aSrcFragWriteOit += THE_FRAG_write_oit_buffers;
} }
const TCollection_AsciiString aLights = stdComputeLighting ((theBits & OpenGl_PO_VertColor) != 0); Standard_Integer aNbLights = 0;
const TCollection_AsciiString aLights = stdComputeLighting (aNbLights, (theBits & OpenGl_PO_VertColor) != 0);
aSrcVert = TCollection_AsciiString() aSrcVert = TCollection_AsciiString()
+ THE_FUNC_transformNormal + THE_FUNC_transformNormal
+ EOL + EOL
@ -1838,6 +1882,8 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramGouraud (Handle(OpenGl_S
aProgramSrc->SetHeader ("#version 300 es"); aProgramSrc->SetHeader ("#version 300 es");
} }
#endif #endif
aProgramSrc->SetNbLightsMax (aNbLights);
aProgramSrc->SetNbClipPlanesMax (aNbClipPlanes);
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert));
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag));
TCollection_AsciiString aKey; TCollection_AsciiString aKey;
@ -1914,18 +1960,22 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramPhong (Handle(OpenGl_Sha
EOL"vec4 getVertColor(void) { return VertColor; }"; EOL"vec4 getVertColor(void) { return VertColor; }";
} }
int aNbClipPlanes = 0;
if ((theBits & OpenGl_PO_ClipPlanesN) != 0) if ((theBits & OpenGl_PO_ClipPlanesN) != 0)
{ {
if ((theBits & OpenGl_PO_ClipPlanes1) != 0) if ((theBits & OpenGl_PO_ClipPlanes1) != 0)
{ {
aNbClipPlanes = 1;
aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_1; aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_1;
} }
else if ((theBits & OpenGl_PO_ClipPlanes2) != 0) else if ((theBits & OpenGl_PO_ClipPlanes2) != 0)
{ {
aNbClipPlanes = 2;
aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_2; aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_2;
} }
else else
{ {
aNbClipPlanes = Graphic3d_ShaderProgram::THE_MAX_CLIP_PLANES_DEFAULT;
aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_N; aSrcFragExtraMain += THE_FRAG_CLIP_PLANES_N;
} }
} }
@ -1955,7 +2005,8 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramPhong (Handle(OpenGl_Sha
+ EOL" gl_Position = occProjectionMatrix * occWorldViewMatrix * occModelWorldMatrix * occVertex;" + EOL" gl_Position = occProjectionMatrix * occWorldViewMatrix * occModelWorldMatrix * occVertex;"
EOL"}"; EOL"}";
const TCollection_AsciiString aLights = stdComputeLighting ((theBits & OpenGl_PO_VertColor) != 0); Standard_Integer aNbLights = 0;
const TCollection_AsciiString aLights = stdComputeLighting (aNbLights, (theBits & OpenGl_PO_VertColor) != 0);
aSrcFrag = TCollection_AsciiString() aSrcFrag = TCollection_AsciiString()
+ EOL"THE_SHADER_IN vec4 PositionWorld;" + EOL"THE_SHADER_IN vec4 PositionWorld;"
EOL"THE_SHADER_IN vec4 Position;" EOL"THE_SHADER_IN vec4 Position;"
@ -2003,6 +2054,8 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramPhong (Handle(OpenGl_Sha
} }
} }
#endif #endif
aProgramSrc->SetNbLightsMax (aNbLights);
aProgramSrc->SetNbClipPlanesMax (aNbClipPlanes);
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert));
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag));
TCollection_AsciiString aKey; TCollection_AsciiString aKey;
@ -2225,6 +2278,8 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramStereo (Handle(OpenGl_Sh
} }
#endif #endif
aProgramSrc->SetNbLightsMax (0);
aProgramSrc->SetNbClipPlanesMax (0);
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_VERTEX, aSrcVert));
aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag)); aProgramSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (Graphic3d_TOS_FRAGMENT, aSrcFrag));
TCollection_AsciiString aKey; TCollection_AsciiString aKey;

View File

@ -444,8 +444,10 @@ protected:
const Standard_Boolean theIsFlatNormal = false); const Standard_Boolean theIsFlatNormal = false);
//! Define computeLighting GLSL function depending on current lights configuration //! Define computeLighting GLSL function depending on current lights configuration
//! @param theHasVertColor flag to use getVertColor() instead of Ambient and Diffuse components of active material //! @param theNbLights [out] number of defined light sources
Standard_EXPORT TCollection_AsciiString stdComputeLighting (const Standard_Boolean theHasVertColor); //! @param theHasVertColor [in] flag to use getVertColor() instead of Ambient and Diffuse components of active material
Standard_EXPORT TCollection_AsciiString stdComputeLighting (Standard_Integer& theNbLights,
Standard_Boolean theHasVertColor);
//! Bind specified program to current context and apply state. //! Bind specified program to current context and apply state.
Standard_EXPORT Standard_Boolean bindProgramWithState (const Handle(OpenGl_ShaderProgram)& theProgram); Standard_EXPORT Standard_Boolean bindProgramWithState (const Handle(OpenGl_ShaderProgram)& theProgram);
@ -522,8 +524,10 @@ protected:
gp_XYZ myLocalOrigin; //!< local camera transformation gp_XYZ myLocalOrigin; //!< local camera transformation
Standard_Boolean myHasLocalOrigin; //!< flag indicating that local camera transformation has been set Standard_Boolean myHasLocalOrigin; //!< flag indicating that local camera transformation has been set
mutable OpenGl_ShaderLightType myLightTypeArray [OpenGLMaxLights]; mutable NCollection_Array1<OpenGl_ShaderLightType> myLightTypeArray;
mutable OpenGl_ShaderLightParameters myLightParamsArray[OpenGLMaxLights]; mutable NCollection_Array1<OpenGl_ShaderLightParameters> myLightParamsArray;
mutable NCollection_Array1<OpenGl_Vec4> myClipPlaneArray;
mutable NCollection_Array1<OpenGl_Vec4d> myClipPlaneArrayFfp;
private: private:

View File

@ -149,6 +149,8 @@ OpenGl_ShaderProgram::OpenGl_ShaderProgram (const Handle(Graphic3d_ShaderProgram
myProgramID (NO_PROGRAM), myProgramID (NO_PROGRAM),
myProxy (theProxy), myProxy (theProxy),
myShareCount(1), myShareCount(1),
myNbLightsMax (0),
myNbClipPlanesMax (0),
myHasTessShader (false) myHasTessShader (false)
{ {
memset (myCurrentState, 0, sizeof (myCurrentState)); memset (myCurrentState, 0, sizeof (myCurrentState));
@ -327,11 +329,24 @@ Standard_Boolean OpenGl_ShaderProgram::Initialize (const Handle(OpenGl_Context)&
case Graphic3d_TOS_FRAGMENT: { aHeaderType = "#define FRAGMENT_SHADER\n"; break; } case Graphic3d_TOS_FRAGMENT: { aHeaderType = "#define FRAGMENT_SHADER\n"; break; }
} }
TCollection_AsciiString aHeaderConstants;
myNbLightsMax = !myProxy.IsNull() ? myProxy->NbLightsMax() : 0;
myNbClipPlanesMax = !myProxy.IsNull() ? myProxy->NbClipPlanesMax() : 0;
if (myNbLightsMax > 0)
{
aHeaderConstants += TCollection_AsciiString("#define THE_MAX_LIGHTS ") + myNbLightsMax + "\n";
}
if (myNbClipPlanesMax > 0)
{
aHeaderConstants += TCollection_AsciiString("#define THE_MAX_CLIP_PLANES ") + myNbClipPlanesMax + "\n";
}
const TCollection_AsciiString aSource = aHeaderVer // #version - header defining GLSL version, should be first const TCollection_AsciiString aSource = aHeaderVer // #version - header defining GLSL version, should be first
+ (!aHeaderVer.IsEmpty() ? "\n" : "") + (!aHeaderVer.IsEmpty() ? "\n" : "")
+ anExtensions // #extension - list of enabled extensions, should be second + anExtensions // #extension - list of enabled extensions, should be second
+ aPrecisionHeader // precision - default precision qualifiers, should be before any code + aPrecisionHeader // precision - default precision qualifiers, should be before any code
+ aHeaderType // auxiliary macros defining a shader stage (type) + aHeaderType // auxiliary macros defining a shader stage (type)
+ aHeaderConstants
+ Shaders_Declarations_glsl // common declarations (global constants and Vertex Shader inputs) + Shaders_Declarations_glsl // common declarations (global constants and Vertex Shader inputs)
+ Shaders_DeclarationsImpl_glsl + Shaders_DeclarationsImpl_glsl
+ anIter.Value()->Source(); // the source code itself (defining main() function) + anIter.Value()->Source(); // the source code itself (defining main() function)

View File

@ -210,9 +210,19 @@ public:
return myProgramID; return myProgramID;
} }
public:
//! Return TRUE if program defines tessellation stage. //! Return TRUE if program defines tessellation stage.
Standard_Boolean HasTessellationStage() const { return myHasTessShader; } Standard_Boolean HasTessellationStage() const { return myHasTessShader; }
//! Return the length of array of light sources (THE_MAX_LIGHTS),
//! to be used for initialization occLightSources (OpenGl_OCC_LIGHT_SOURCE_PARAMS).
Standard_Integer NbLightsMax() const { return myNbLightsMax; }
//! Return the length of array of clipping planes (THE_MAX_CLIP_PLANES),
//! to be used for initialization occClipPlaneEquations (OpenGl_OCC_CLIP_PLANE_EQUATIONS).
Standard_Integer NbClipPlanesMax() const { return myNbClipPlanesMax; }
private: private:
//! Returns index of last modification of variables of specified state type. //! Returns index of last modification of variables of specified state type.
@ -554,6 +564,8 @@ protected:
OpenGl_ShaderList myShaderObjects; //!< List of attached shader objects OpenGl_ShaderList myShaderObjects; //!< List of attached shader objects
Handle(Graphic3d_ShaderProgram) myProxy; //!< Proxy shader program (from application layer) Handle(Graphic3d_ShaderProgram) myProxy; //!< Proxy shader program (from application layer)
Standard_Integer myShareCount; //!< program users count, initialized with 1 (already shared by one user) Standard_Integer myShareCount; //!< program users count, initialized with 1 (already shared by one user)
Standard_Integer myNbLightsMax; //!< length of array of light sources (THE_MAX_LIGHTS)
Standard_Integer myNbClipPlanesMax; //!< length of array of clipping planes (THE_MAX_CLIP_PLANES)
Standard_Boolean myHasTessShader; //!< flag indicating that program defines tessellation stage Standard_Boolean myHasTessShader; //!< flag indicating that program defines tessellation stage
protected: protected:

View File

@ -13,10 +13,16 @@
// Alternatively, this file may be used under the terms of Open CASCADE // Alternatively, this file may be used under the terms of Open CASCADE
// commercial license or contractual agreement. // commercial license or contractual agreement.
// This files includes definition of common uniform variables in OCCT GLSL programs //! @file Declarations.glsl
//! This files includes definition of common uniform variables in OCCT GLSL programs
#define THE_MAX_LIGHTS 8 //! @def THE_MAX_LIGHTS
#define THE_MAX_CLIP_PLANES 8 //! Specifies the length of array of lights, which is 8 by default. Defined by Shader Manager.
// #define THE_MAX_LIGHTS 8
//! @def THE_MAX_CLIP_PLANES
//! Specifies the length of array of clipping planes, which is 8 by default. Defined by Shader Manager.
// #define THE_MAX_CLIP_PLANES 8
// compatibility macros // compatibility macros
#if (__VERSION__ >= 130) #if (__VERSION__ >= 130)
@ -92,6 +98,7 @@ const int OccLightType_Spot = 3; //!< spot light source
// Light sources // Light sources
uniform vec4 occLightAmbient; //!< Cumulative ambient color uniform vec4 occLightAmbient; //!< Cumulative ambient color
#if defined(THE_MAX_LIGHTS) && (THE_MAX_LIGHTS > 0)
uniform THE_PREC_ENUM int occLightSourcesCount; //!< Total number of light sources uniform THE_PREC_ENUM int occLightSourcesCount; //!< Total number of light sources
int occLight_Type (in int theId); //!< Type of light source int occLight_Type (in int theId); //!< Type of light source
int occLight_IsHeadlight (in int theId); //!< Is light a headlight? int occLight_IsHeadlight (in int theId); //!< Is light a headlight?
@ -103,6 +110,7 @@ float occLight_ConstAttenuation (in int theId); //!< Const attenuation factor o
float occLight_LinearAttenuation (in int theId); //!< Linear attenuation factor of positional light source float occLight_LinearAttenuation (in int theId); //!< Linear attenuation factor of positional light source
float occLight_SpotCutOff (in int theId); //!< Maximum spread angle of the spot light (in radians) float occLight_SpotCutOff (in int theId); //!< Maximum spread angle of the spot light (in radians)
float occLight_SpotExponent (in int theId); //!< Attenuation of the spot light intensity (from 0 to 1) float occLight_SpotExponent (in int theId); //!< Attenuation of the spot light intensity (from 0 to 1)
#endif
// Front material properties accessors // Front material properties accessors
vec4 occFrontMaterial_Emission(void); //!< Emission color vec4 occFrontMaterial_Emission(void); //!< Emission color
@ -135,5 +143,7 @@ uniform int occOitOutput; //!< Enable bit for writing o
uniform float occOitDepthFactor; //!< Influence of the depth component to the coverage of the accumulated fragment uniform float occOitDepthFactor; //!< Influence of the depth component to the coverage of the accumulated fragment
//! Parameters of clipping planes //! Parameters of clipping planes
#if defined(THE_MAX_CLIP_PLANES) && (THE_MAX_CLIP_PLANES > 0)
uniform vec4 occClipPlaneEquations[THE_MAX_CLIP_PLANES]; uniform vec4 occClipPlaneEquations[THE_MAX_CLIP_PLANES];
uniform THE_PREC_ENUM int occClipPlaneCount; //!< Total number of clip planes uniform THE_PREC_ENUM int occClipPlaneCount; //!< Total number of clip planes
#endif

View File

@ -15,6 +15,7 @@
// This file includes implementation of common functions and properties accessors // This file includes implementation of common functions and properties accessors
#if defined(THE_MAX_LIGHTS) && (THE_MAX_LIGHTS > 0)
// arrays of light sources // arrays of light sources
uniform THE_PREC_ENUM ivec2 occLightSourcesTypes[THE_MAX_LIGHTS]; //!< packed light sources types uniform THE_PREC_ENUM ivec2 occLightSourcesTypes[THE_MAX_LIGHTS]; //!< packed light sources types
uniform vec4 occLightSources[THE_MAX_LIGHTS * 4]; //!< packed light sources parameters uniform vec4 occLightSources[THE_MAX_LIGHTS * 4]; //!< packed light sources parameters
@ -30,6 +31,7 @@ float occLight_ConstAttenuation (in int theId) { return occLightSources[theId *
float occLight_LinearAttenuation (in int theId) { return occLightSources[theId * 4 + 3].y; } float occLight_LinearAttenuation (in int theId) { return occLightSources[theId * 4 + 3].y; }
float occLight_SpotCutOff (in int theId) { return occLightSources[theId * 4 + 3].z; } float occLight_SpotCutOff (in int theId) { return occLightSources[theId * 4 + 3].z; }
float occLight_SpotExponent (in int theId) { return occLightSources[theId * 4 + 3].w; } float occLight_SpotExponent (in int theId) { return occLightSources[theId * 4 + 3].w; }
#endif
// material state // material state
uniform vec4 occFrontMaterial[5]; uniform vec4 occFrontMaterial[5];

View File

@ -18,6 +18,7 @@ static const char Shaders_DeclarationsImpl_glsl[] =
"\n" "\n"
"// This file includes implementation of common functions and properties accessors\n" "// This file includes implementation of common functions and properties accessors\n"
"\n" "\n"
"#if defined(THE_MAX_LIGHTS) && (THE_MAX_LIGHTS > 0)\n"
"// arrays of light sources\n" "// arrays of light sources\n"
"uniform THE_PREC_ENUM ivec2 occLightSourcesTypes[THE_MAX_LIGHTS]; //!< packed light sources types\n" "uniform THE_PREC_ENUM ivec2 occLightSourcesTypes[THE_MAX_LIGHTS]; //!< packed light sources types\n"
"uniform vec4 occLightSources[THE_MAX_LIGHTS * 4]; //!< packed light sources parameters\n" "uniform vec4 occLightSources[THE_MAX_LIGHTS * 4]; //!< packed light sources parameters\n"
@ -33,6 +34,7 @@ static const char Shaders_DeclarationsImpl_glsl[] =
"float occLight_LinearAttenuation (in int theId) { return occLightSources[theId * 4 + 3].y; }\n" "float occLight_LinearAttenuation (in int theId) { return occLightSources[theId * 4 + 3].y; }\n"
"float occLight_SpotCutOff (in int theId) { return occLightSources[theId * 4 + 3].z; }\n" "float occLight_SpotCutOff (in int theId) { return occLightSources[theId * 4 + 3].z; }\n"
"float occLight_SpotExponent (in int theId) { return occLightSources[theId * 4 + 3].w; }\n" "float occLight_SpotExponent (in int theId) { return occLightSources[theId * 4 + 3].w; }\n"
"#endif\n"
"\n" "\n"
"// material state\n" "// material state\n"
"uniform vec4 occFrontMaterial[5];\n" "uniform vec4 occFrontMaterial[5];\n"

View File

@ -16,10 +16,16 @@ static const char Shaders_Declarations_glsl[] =
"// Alternatively, this file may be used under the terms of Open CASCADE\n" "// Alternatively, this file may be used under the terms of Open CASCADE\n"
"// commercial license or contractual agreement.\n" "// commercial license or contractual agreement.\n"
"\n" "\n"
"// This files includes definition of common uniform variables in OCCT GLSL programs\n" "//! @file Declarations.glsl\n"
"//! This files includes definition of common uniform variables in OCCT GLSL programs\n"
"\n" "\n"
"#define THE_MAX_LIGHTS 8\n" "//! @def THE_MAX_LIGHTS\n"
"#define THE_MAX_CLIP_PLANES 8\n" "//! Specifies the length of array of lights, which is 8 by default. Defined by Shader Manager.\n"
"// #define THE_MAX_LIGHTS 8\n"
"\n"
"//! @def THE_MAX_CLIP_PLANES\n"
"//! Specifies the length of array of clipping planes, which is 8 by default. Defined by Shader Manager.\n"
"// #define THE_MAX_CLIP_PLANES 8\n"
"\n" "\n"
"// compatibility macros\n" "// compatibility macros\n"
"#if (__VERSION__ >= 130)\n" "#if (__VERSION__ >= 130)\n"
@ -95,6 +101,7 @@ static const char Shaders_Declarations_glsl[] =
"\n" "\n"
"// Light sources\n" "// Light sources\n"
"uniform vec4 occLightAmbient; //!< Cumulative ambient color\n" "uniform vec4 occLightAmbient; //!< Cumulative ambient color\n"
"#if defined(THE_MAX_LIGHTS) && (THE_MAX_LIGHTS > 0)\n"
"uniform THE_PREC_ENUM int occLightSourcesCount; //!< Total number of light sources\n" "uniform THE_PREC_ENUM int occLightSourcesCount; //!< Total number of light sources\n"
"int occLight_Type (in int theId); //!< Type of light source\n" "int occLight_Type (in int theId); //!< Type of light source\n"
"int occLight_IsHeadlight (in int theId); //!< Is light a headlight?\n" "int occLight_IsHeadlight (in int theId); //!< Is light a headlight?\n"
@ -106,6 +113,7 @@ static const char Shaders_Declarations_glsl[] =
"float occLight_LinearAttenuation (in int theId); //!< Linear attenuation factor of positional light source\n" "float occLight_LinearAttenuation (in int theId); //!< Linear attenuation factor of positional light source\n"
"float occLight_SpotCutOff (in int theId); //!< Maximum spread angle of the spot light (in radians)\n" "float occLight_SpotCutOff (in int theId); //!< Maximum spread angle of the spot light (in radians)\n"
"float occLight_SpotExponent (in int theId); //!< Attenuation of the spot light intensity (from 0 to 1)\n" "float occLight_SpotExponent (in int theId); //!< Attenuation of the spot light intensity (from 0 to 1)\n"
"#endif\n"
"\n" "\n"
"// Front material properties accessors\n" "// Front material properties accessors\n"
"vec4 occFrontMaterial_Emission(void); //!< Emission color\n" "vec4 occFrontMaterial_Emission(void); //!< Emission color\n"
@ -138,5 +146,7 @@ static const char Shaders_Declarations_glsl[] =
"uniform float occOitDepthFactor; //!< Influence of the depth component to the coverage of the accumulated fragment\n" "uniform float occOitDepthFactor; //!< Influence of the depth component to the coverage of the accumulated fragment\n"
"\n" "\n"
"//! Parameters of clipping planes\n" "//! Parameters of clipping planes\n"
"#if defined(THE_MAX_CLIP_PLANES) && (THE_MAX_CLIP_PLANES > 0)\n"
"uniform vec4 occClipPlaneEquations[THE_MAX_CLIP_PLANES];\n" "uniform vec4 occClipPlaneEquations[THE_MAX_CLIP_PLANES];\n"
"uniform THE_PREC_ENUM int occClipPlaneCount; //!< Total number of clip planes\n"; "uniform THE_PREC_ENUM int occClipPlaneCount; //!< Total number of clip planes\n"
"#endif\n";

View File

@ -53,8 +53,6 @@ void V3d_View::SetLightOn (const Handle(V3d_Light)& theLight)
{ {
if (!myActiveLights.Contains (theLight)) if (!myActiveLights.Contains (theLight))
{ {
if (myActiveLights.Extent() >= LightLimit())
throw V3d_BadValue("V3d_View::SetLightOn, too many lights");
myActiveLights.Append (theLight); myActiveLights.Append (theLight);
UpdateLights(); UpdateLights();
} }

44
tests/v3d/glsl/phong_pos2 Normal file
View File

@ -0,0 +1,44 @@
puts "========"
puts "0029283: Visualization - allow defining more than 8 light sources"
puts "========"
pload MODELING VISUALIZATION
# display objects
vclear
vclose ALL
vinit View1 -width 1024 -height 768
vcaps -ffp 0
vrenderparams -shadingModel phong
vaxo
for { set anObjIter 0 } { $anObjIter < 3 } { incr anObjIter } {
set aShiftX [expr -4 + $anObjIter * 4]
psphere s$anObjIter 0.5
vdisplay -dispMode 1 s$anObjIter
vsetlocation s$anObjIter $aShiftX 0 0
}
vfit
# define lights
set THE_LIGHTS {
{ -1 -1 -1 RED1 }
{ 1 -1 -1 YELLOW }
{ -1 1 -1 BLUE1 }
{ -1 -1 1 CYAN1 }
{ 1 1 -1 PURPLE }
{ 1 1 1 WHITE }
{ -1 1 1 HOTPINK }
{ 1 -1 1 GREEN }
{ -4 -1 0 MAGENTA1 }
{ 4 -1 0 MAGENTA3 }
}
vlight clear
for { set aLightIter 1 } { $aLightIter <= 10 } { incr aLightIter } {
set aLight [lindex $THE_LIGHTS [expr $aLightIter - 1]]
set aColor [lindex $aLight 3]
set aPos [list [lindex $aLight 0] [lindex $aLight 1] [lindex $aLight 2]]
vlight add positional pos {*}$aPos color $aColor headLight 0
vpoint v${aLightIter} {*}$aPos
vdrawtext t${aLightIter} "light${aLightIter} $aColor" -pos {*}$aPos -color $aColor
vdump $::imagedir/${::casename}_${aLightIter}.png
}