mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-03 17:56:21 +03:00
0032083: Visualization, TKOpenGl - PBR rendering is unavailable on Apple A12 Bionic (iPad)
Added "#define occLight_Index(theId) 0" workaround allowing to compile a single light source PBR shading program on OpenGL ES 2.0, as it disallows non-constant index expressions (even trivially deducable at compile-time). PBR lookup table is now automatically resized to RGBA format when RG texture format is unsupported (OpenGL ES 2.0). OpenGl_ShaderManager now allows compiling PBR shaders using WebGL 1.0 + GL_EXT_shader_texture_lod extension. PBR IBL baking GLSL program has been redisigned to use if/else to avoid non-constant index expressions. Diffuse baking implements packing float values into a temporary RGBA8 texture as a fallback solution on WebGL 1.0 implementations supporting float textures but not as FBO render targets. OpenGl_PBREnvironment now uses GL_FRAMEBUFFER instead of GL_DRAW_FRAMEBUFFER for compatibility with OpenGL ES 2.0.
This commit is contained in:
parent
776302d46b
commit
8f8fe4a97c
@ -158,6 +158,7 @@ OpenGl_Context::OpenGl_Context (const Handle(OpenGl_Caps)& theCaps)
|
||||
hasTexSRGB (Standard_False),
|
||||
hasFboSRGB (Standard_False),
|
||||
hasSRGBControl (Standard_False),
|
||||
hasFboRenderMipmap (Standard_False),
|
||||
#if defined(GL_ES_VERSION_2_0)
|
||||
hasFlatShading (OpenGl_FeatureNotAvailable),
|
||||
#else
|
||||
@ -1228,6 +1229,20 @@ void OpenGl_Context::ReadGlVersion (Standard_Integer& theGlVerMajor,
|
||||
// read numbers
|
||||
theGlVerMajor = atoi (aMajorStr);
|
||||
theGlVerMinor = atoi (aMinorStr);
|
||||
#if defined(__EMSCRIPTEN__)
|
||||
if (theGlVerMajor >= 3)
|
||||
{
|
||||
if (!toCheckVer3
|
||||
|| ::strstr (aVerStr, "WebGL 1.0") != NULL)
|
||||
{
|
||||
Message::SendWarning() << "Warning! OpenGL context reports version " << theGlVerMajor << "." << theGlVerMinor
|
||||
<< " but WebGL 2.0 was unavailable\n"
|
||||
<< "Fallback to OpenGL ES 2.0 will be used instead of reported version";
|
||||
theGlVerMajor = 2;
|
||||
theGlVerMinor = 0;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
if (theGlVerMajor <= 0)
|
||||
{
|
||||
@ -1540,6 +1555,8 @@ void OpenGl_Context::init (const Standard_Boolean theIsCoreProfile)
|
||||
|| CheckExtension ("GL_OES_rgb8_rgba8");
|
||||
hasTexSRGB = IsGlGreaterEqual (3, 0);
|
||||
hasFboSRGB = IsGlGreaterEqual (3, 0);
|
||||
hasFboRenderMipmap = IsGlGreaterEqual (3, 0)
|
||||
|| CheckExtension ("GL_OES_fbo_render_mipmap");
|
||||
hasSRGBControl = CheckExtension ("GL_EXT_sRGB_write_control");
|
||||
// NPOT textures has limited support within OpenGL ES 2.0
|
||||
// which are relaxed by OpenGL ES 3.0 or some extensions
|
||||
@ -1983,6 +2000,7 @@ void OpenGl_Context::init (const Standard_Boolean theIsCoreProfile)
|
||||
hasTexSRGB = IsGlGreaterEqual (2, 1);
|
||||
hasFboSRGB = IsGlGreaterEqual (2, 1);
|
||||
hasSRGBControl = hasFboSRGB;
|
||||
hasFboRenderMipmap = Standard_True;
|
||||
arbDrawBuffers = CheckExtension ("GL_ARB_draw_buffers");
|
||||
arbNPTW = CheckExtension ("GL_ARB_texture_non_power_of_two");
|
||||
arbTexFloat = IsGlGreaterEqual (3, 0)
|
||||
@ -3415,10 +3433,11 @@ void OpenGl_Context::init (const Standard_Boolean theIsCoreProfile)
|
||||
// check whether PBR shading model is supported
|
||||
myHasPBR = arbFBO != NULL
|
||||
&& myMaxTexCombined >= 4
|
||||
&& arbTexRG
|
||||
&& arbTexFloat
|
||||
&& (IsGlGreaterEqual (3, 0)
|
||||
#if !defined(GL_ES_VERSION_2_0)
|
||||
#if defined(GL_ES_VERSION_2_0)
|
||||
|| CheckExtension ("GL_EXT_shader_texture_lod")
|
||||
#else
|
||||
|| (IsGlGreaterEqual (2, 1) && CheckExtension ("GL_EXT_gpu_shader4"))
|
||||
#endif
|
||||
);
|
||||
|
@ -1085,6 +1085,7 @@ public: //! @name extensions
|
||||
Standard_Boolean hasTexSRGB; //!< sRGB texture formats (desktop OpenGL 2.1, OpenGL ES 3.0 or GL_EXT_texture_sRGB)
|
||||
Standard_Boolean hasFboSRGB; //!< sRGB FBO render targets (desktop OpenGL 2.1, OpenGL ES 3.0)
|
||||
Standard_Boolean hasSRGBControl; //!< sRGB write control (any desktop OpenGL, OpenGL ES + GL_EXT_sRGB_write_control extension)
|
||||
Standard_Boolean hasFboRenderMipmap; //!< FBO render target could be non-zero mimap level of texture
|
||||
OpenGl_FeatureFlag hasFlatShading; //!< Complex flag indicating support of Flat shading (Graphic3d_TOSM_FACET) (always available on desktop; on OpenGL ES - since 3.0 or as extension GL_OES_standard_derivatives)
|
||||
OpenGl_FeatureFlag hasGlslBitwiseOps; //!< GLSL supports bitwise operations; OpenGL 3.0 / OpenGL ES 3.0 (GLSL 130 / GLSL ES 300) or OpenGL 2.1 + GL_EXT_gpu_shader4
|
||||
OpenGl_FeatureFlag hasDrawBuffers; //!< Complex flag indicating support of multiple draw buffers (desktop OpenGL 2.0, OpenGL ES 3.0, GL_ARB_draw_buffers, GL_EXT_draw_buffers)
|
||||
|
@ -77,7 +77,7 @@ private:
|
||||
|
||||
void restore()
|
||||
{
|
||||
myContext->arbFBO->glBindFramebuffer (GL_DRAW_FRAMEBUFFER, myFBO);
|
||||
myContext->arbFBO->glBindFramebuffer (GL_FRAMEBUFFER, myFBO);
|
||||
myContext->BindProgram (myShaderProgram);
|
||||
myContext->ResizeViewport (myViewport);
|
||||
myContext->core11fwd->glClearColor (myClearColor.r(), myClearColor.g(), myClearColor.b(), myClearColor.a());
|
||||
@ -159,7 +159,8 @@ OpenGl_PBREnvironment::OpenGl_PBREnvironment (const Handle(OpenGl_Context)& the
|
||||
mySpecMapLevelsNumber (std::max (2u, std::min (theSpecMapLevelsNumber, std::max (1u, thePowOf2Size) + 1))),
|
||||
myFBO (OpenGl_FrameBuffer::NO_FRAMEBUFFER),
|
||||
myIsComplete (Standard_False),
|
||||
myIsNeededToBeBound (Standard_True)
|
||||
myIsNeededToBeBound (Standard_True),
|
||||
myCanRenderFloat (Standard_True)
|
||||
{
|
||||
OpenGl_PBREnvironmentSentry aSentry (theCtx);
|
||||
|
||||
@ -253,8 +254,9 @@ void OpenGl_PBREnvironment::Release (OpenGl_Context* theCtx)
|
||||
}
|
||||
myFBO = OpenGl_FrameBuffer::NO_FRAMEBUFFER;
|
||||
}
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].Release(theCtx);
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].Release (theCtx);
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_Specular] .Release (theCtx);
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseFallback].Release (theCtx);
|
||||
myVBO.Release (theCtx);
|
||||
}
|
||||
|
||||
@ -276,16 +278,35 @@ bool OpenGl_PBREnvironment::initTextures (const Handle(OpenGl_Context)& theCtx)
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_Specular] .Sampler()->Parameters()->SetTextureUnit (theCtx->PBRSpecIBLMapTexUnit());
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].Sampler()->Parameters()->SetTextureUnit (theCtx->PBRDiffIBLMapSHTexUnit());
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_Specular] .Sampler()->Parameters()->SetFilter (Graphic3d_TOTF_TRILINEAR);
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].Sampler()->Parameters()->SetFilter(Graphic3d_TOTF_NEAREST);
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].Sampler()->Parameters()->SetFilter (Graphic3d_TOTF_NEAREST);
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_Specular] .Sampler()->Parameters()->SetLevelsRange (mySpecMapLevelsNumber - 1);
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseFallback].Sampler()->Parameters()->SetFilter (Graphic3d_TOTF_NEAREST);
|
||||
|
||||
// NVIDIA's driver didn't work properly with 3 channel texture for diffuse SH coefficients so that alpha channel has been added
|
||||
return myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].Init (theCtx,
|
||||
OpenGl_TextureFormat::FindFormat (theCtx, Image_Format_RGBAF, false),
|
||||
Graphic3d_Vec2i (9, 1),
|
||||
Graphic3d_TOT_2D)
|
||||
&& myIBLMaps[OpenGl_TypeOfIBLMap_Specular].InitCubeMap (theCtx, Handle(Graphic3d_CubeMap)(),
|
||||
Standard_Size(1) << myPow2Size, Image_Format_RGB, true, false);
|
||||
if (!myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].Init (theCtx,
|
||||
OpenGl_TextureFormat::FindFormat (theCtx, Image_Format_RGBAF, false),
|
||||
Graphic3d_Vec2i (9, 1), Graphic3d_TOT_2D))
|
||||
{
|
||||
Message::SendFail() << "OpenGl_PBREnvironment, DiffuseSH texture creation failed";
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!myIBLMaps[OpenGl_TypeOfIBLMap_Specular].InitCubeMap (theCtx, Handle(Graphic3d_CubeMap)(),
|
||||
Standard_Size(1) << myPow2Size, Image_Format_RGB, true, false))
|
||||
{
|
||||
Message::SendFail() << "OpenGl_PBREnvironment, Specular texture creation failed";
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseFallback].Init (theCtx,
|
||||
OpenGl_TextureFormat::FindFormat (theCtx, Image_Format_RGBA, false),
|
||||
Graphic3d_Vec2i (10, 4), Graphic3d_TOT_2D))
|
||||
{
|
||||
Message::SendFail() << "OpenGl_PBREnvironment, DiffuseFallback texture creation failed";
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
// =======================================================================
|
||||
@ -319,28 +340,98 @@ bool OpenGl_PBREnvironment::initFBO (const Handle(OpenGl_Context)& theCtx)
|
||||
// purpose :
|
||||
// =======================================================================
|
||||
bool OpenGl_PBREnvironment::processDiffIBLMap (const Handle(OpenGl_Context)& theCtx,
|
||||
Standard_Boolean theIsDrawAction,
|
||||
Standard_Size theNbSamples)
|
||||
const BakingParams* theDrawParams)
|
||||
{
|
||||
theCtx->arbFBO->glFramebufferTexture2D (GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D,
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].TextureId(), 0);
|
||||
const Standard_Integer aViewport[4] = { 0, 0, 9, 1 };
|
||||
theCtx->ResizeViewport(aViewport);
|
||||
if (theIsDrawAction)
|
||||
const OpenGl_TypeOfIBLMap aRendMapId = myCanRenderFloat ? OpenGl_TypeOfIBLMap_DiffuseSH : OpenGl_TypeOfIBLMap_DiffuseFallback;
|
||||
Image_PixMap anImageF;
|
||||
if (!myCanRenderFloat)
|
||||
{
|
||||
theCtx->ActiveProgram()->SetUniform(theCtx, "occNbSpecIBLLevels", 0);
|
||||
theCtx->ActiveProgram()->SetUniform(theCtx, "uSamplesNum", static_cast<Standard_Integer>(theNbSamples));
|
||||
anImageF.InitZero (Image_Format_RGBAF, myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].SizeX(), myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].SizeY());
|
||||
}
|
||||
|
||||
theCtx->arbFBO->glFramebufferTexture2D (GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D,
|
||||
myIBLMaps[aRendMapId].TextureId(), 0);
|
||||
const Standard_Integer aViewport[4] = { 0, 0, 9, myCanRenderFloat ? 1 : 3 };
|
||||
theCtx->ResizeViewport(aViewport);
|
||||
if (theDrawParams != NULL)
|
||||
{
|
||||
if (!theCtx->ShaderManager()->BindPBREnvBakingProgram (aRendMapId))
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
const Handle(OpenGl_ShaderProgram)& aProg = theCtx->ActiveProgram();
|
||||
aProg->SetSampler (theCtx, "uEnvMap", theCtx->PBRSpecIBLMapTexUnit());
|
||||
aProg->SetUniform (theCtx, "uZCoeff", theDrawParams->IsZInverted ? -1 : 1);
|
||||
aProg->SetUniform (theCtx, "uYCoeff", theDrawParams->IsTopDown ? 1 : -1);
|
||||
aProg->SetUniform (theCtx, "uSamplesNum", Standard_Integer(theDrawParams->NbDiffSamples));
|
||||
|
||||
myVBO.BindAttribute (theCtx, Graphic3d_TOA_POS);
|
||||
theCtx->core11fwd->glDrawArrays (GL_TRIANGLE_STRIP, 0, 4);
|
||||
myVBO.UnbindAttribute (theCtx, Graphic3d_TOA_POS);
|
||||
|
||||
if (!myCanRenderFloat)
|
||||
{
|
||||
// unpack RGBA8 values to floats
|
||||
Image_PixMap anImageIn;
|
||||
anImageIn.InitZero (myCanRenderFloat ? Image_Format_RGBAF : Image_Format_RGBA, aViewport[2], aViewport[3]);
|
||||
theCtx->core11fwd->glReadPixels (0, 0, aViewport[2], aViewport[3],
|
||||
GL_RGBA, myCanRenderFloat ? GL_FLOAT : GL_UNSIGNED_BYTE, anImageIn.ChangeData());
|
||||
const GLenum anErr = theCtx->core11fwd->glGetError();
|
||||
if (anErr != GL_NO_ERROR)
|
||||
{
|
||||
theCtx->PushMessage (GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_ERROR, 0, GL_DEBUG_SEVERITY_HIGH,
|
||||
TCollection_AsciiString ("Unable to read PBR baking diffuse texture. Error ") + OpenGl_Context::FormatGlError (anErr));
|
||||
}
|
||||
for (Standard_Size aValIter = 0; aValIter < anImageIn.SizeX(); ++aValIter)
|
||||
{
|
||||
Graphic3d_Vec4 aVal;
|
||||
if (myCanRenderFloat)
|
||||
{
|
||||
aVal = anImageIn.Value<Graphic3d_Vec4> (0, aValIter);
|
||||
}
|
||||
else
|
||||
{
|
||||
const int32_t aPacked[3] = { anImageIn.Value<int32_t> (2, aValIter), anImageIn.Value<int32_t> (1, aValIter), anImageIn.Value<int32_t> (0, aValIter) };
|
||||
aVal[0] = aPacked[0] / 2147483647.0f;
|
||||
aVal[1] = aPacked[1] / 2147483647.0f;
|
||||
aVal[2] = aPacked[2] / 2147483647.0f;
|
||||
}
|
||||
anImageF.ChangeValue<Graphic3d_Vec4> (0, aValIter) = aVal;
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
theCtx->core11fwd->glClear (GL_COLOR_BUFFER_BIT);
|
||||
if (myCanRenderFloat)
|
||||
{
|
||||
theCtx->core11fwd->glClear (GL_COLOR_BUFFER_BIT);
|
||||
|
||||
theCtx->core11fwd->glEnable (GL_SCISSOR_TEST);
|
||||
theCtx->core11fwd->glClearColor (0.f, 0.f, 0.f, 1.f);
|
||||
theCtx->core11fwd->glScissor (1, 0, 8, 1);
|
||||
theCtx->core11fwd->glClear (GL_COLOR_BUFFER_BIT);
|
||||
theCtx->core11fwd->glEnable (GL_SCISSOR_TEST);
|
||||
theCtx->core11fwd->glClearColor (0.f, 0.f, 0.f, 1.f);
|
||||
theCtx->core11fwd->glScissor (1, 0, 8, 1);
|
||||
theCtx->core11fwd->glClear (GL_COLOR_BUFFER_BIT);
|
||||
theCtx->core11fwd->glDisable (GL_SCISSOR_TEST);
|
||||
}
|
||||
else
|
||||
{
|
||||
anImageF.ChangeValue<Graphic3d_Vec4> (0, 0) = Graphic3d_Vec4 (1.0f);
|
||||
for (Standard_Size aValIter = 1; aValIter < anImageF.SizeX(); ++aValIter)
|
||||
{
|
||||
anImageF.ChangeValue<Graphic3d_Vec4> (0, aValIter) = Graphic3d_Vec4 (0.0f, 0.0f, 0.0f, 1.0f);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (!myCanRenderFloat)
|
||||
{
|
||||
if (!myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].Init (theCtx,
|
||||
OpenGl_TextureFormat::FindFormat (theCtx, Image_Format_RGBAF, false),
|
||||
Graphic3d_Vec2i (9, 1), Graphic3d_TOT_2D, &anImageF))
|
||||
{
|
||||
Message::SendFail() << "OpenGl_PBREnvironment, DiffuseSH texture update failed";
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
@ -351,34 +442,52 @@ bool OpenGl_PBREnvironment::processDiffIBLMap (const Handle(OpenGl_Context)& the
|
||||
// purpose :
|
||||
// =======================================================================
|
||||
bool OpenGl_PBREnvironment::processSpecIBLMap (const Handle(OpenGl_Context)& theCtx,
|
||||
Standard_Boolean theIsDrawAction,
|
||||
Standard_Integer theEnvMapSize,
|
||||
Standard_Size theNbSamples,
|
||||
Standard_ShortReal theProbability)
|
||||
const BakingParams* theDrawParams)
|
||||
{
|
||||
if (theIsDrawAction)
|
||||
if (theDrawParams != NULL)
|
||||
{
|
||||
theCtx->ActiveProgram()->SetUniform (theCtx, "occNbSpecIBLLevels", Standard_Integer(mySpecMapLevelsNumber));
|
||||
theCtx->ActiveProgram()->SetUniform (theCtx, "uEnvMapSize", theEnvMapSize);
|
||||
if (!theCtx->ShaderManager()->BindPBREnvBakingProgram (OpenGl_TypeOfIBLMap_Specular))
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
const Handle(OpenGl_ShaderProgram)& aProg = theCtx->ActiveProgram();
|
||||
const float aSolidAngleSource = float(4.0 * M_PI / (6.0 * float(theDrawParams->EnvMapSize * theDrawParams->EnvMapSize)));
|
||||
aProg->SetSampler (theCtx, "uEnvMap", theCtx->PBRSpecIBLMapTexUnit());
|
||||
aProg->SetUniform (theCtx, "uZCoeff", theDrawParams->IsZInverted ? -1 : 1);
|
||||
aProg->SetUniform (theCtx, "uYCoeff", theDrawParams->IsTopDown ? 1 : -1);
|
||||
aProg->SetUniform (theCtx, "occNbSpecIBLLevels", Standard_Integer(mySpecMapLevelsNumber));
|
||||
aProg->SetUniform (theCtx, "uEnvSolidAngleSource", aSolidAngleSource);
|
||||
myVBO.BindAttribute (theCtx, Graphic3d_TOA_POS);
|
||||
}
|
||||
|
||||
const bool canRenderMipmaps = theCtx->hasFboRenderMipmap;
|
||||
const OpenGl_TextureFormat aTexFormat = OpenGl_TextureFormat::FindSizedFormat (theCtx, myIBLMaps[OpenGl_TypeOfIBLMap_Specular].SizedFormat());
|
||||
#if !defined(GL_ES_VERSION_2_0)
|
||||
const GLint anIntFormat = aTexFormat.InternalFormat();
|
||||
#else
|
||||
// ES 2.0 does not support sized formats and format conversions - them detected from data type
|
||||
const GLint anIntFormat = theCtx->IsGlGreaterEqual (3, 0) ? aTexFormat.InternalFormat() : aTexFormat.PixelFormat();
|
||||
#endif
|
||||
|
||||
for (int aLevelIter = mySpecMapLevelsNumber - 1;; --aLevelIter)
|
||||
{
|
||||
const Standard_Integer aSize = 1 << (myPow2Size - aLevelIter);
|
||||
const Standard_Integer aViewport[4] = { 0, 0, aSize, aSize };
|
||||
theCtx->ResizeViewport (aViewport);
|
||||
if (theIsDrawAction)
|
||||
if (theDrawParams != NULL)
|
||||
{
|
||||
Standard_Integer aNbSamples = static_cast<Standard_Integer>(Graphic3d_PBRMaterial::SpecIBLMapSamplesFactor (theProbability, aLevelIter / float (mySpecMapLevelsNumber - 1)) * theNbSamples);
|
||||
theCtx->ActiveProgram()->SetUniform (theCtx, "uSamplesNum", static_cast<Standard_Integer>(aNbSamples));
|
||||
const Standard_Integer aNbSamples = Standard_Integer(Graphic3d_PBRMaterial::SpecIBLMapSamplesFactor (theDrawParams->Probability,
|
||||
aLevelIter / float (mySpecMapLevelsNumber - 1)) * theDrawParams->NbSpecSamples);
|
||||
theCtx->ActiveProgram()->SetUniform (theCtx, "uSamplesNum", aNbSamples);
|
||||
theCtx->ActiveProgram()->SetUniform (theCtx, "uCurrentLevel", aLevelIter);
|
||||
}
|
||||
|
||||
for (Standard_Integer aSideIter = 0; aSideIter < 6; ++aSideIter)
|
||||
{
|
||||
theCtx->arbFBO->glFramebufferTexture2D (GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_CUBE_MAP_POSITIVE_X + aSideIter,
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_Specular].TextureId(), aLevelIter);
|
||||
if (theIsDrawAction)
|
||||
theCtx->arbFBO->glFramebufferTexture2D (GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_CUBE_MAP_POSITIVE_X + aSideIter,
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_Specular].TextureId(), canRenderMipmaps ? aLevelIter : 0);
|
||||
if (theDrawParams != NULL)
|
||||
{
|
||||
theCtx->ActiveProgram()->SetUniform(theCtx, "uCurrentSide", aSideIter);
|
||||
theCtx->core11fwd->glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
|
||||
@ -387,6 +496,20 @@ bool OpenGl_PBREnvironment::processSpecIBLMap (const Handle(OpenGl_Context)& the
|
||||
{
|
||||
theCtx->core11fwd->glClear(GL_COLOR_BUFFER_BIT);
|
||||
}
|
||||
|
||||
if (!canRenderMipmaps
|
||||
&& aLevelIter != 0)
|
||||
{
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_Specular].Bind (theCtx, Graphic3d_TextureUnit_1);
|
||||
theCtx->core20fwd->glCopyTexImage2D (GL_TEXTURE_CUBE_MAP_POSITIVE_X + aSideIter, aLevelIter, anIntFormat, 0, 0, (GLsizei )aSize, (GLsizei )aSize, 0);
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_Specular].Unbind (theCtx, Graphic3d_TextureUnit_1);
|
||||
const GLenum anErr = theCtx->core11fwd->glGetError();
|
||||
if (anErr != GL_NO_ERROR)
|
||||
{
|
||||
theCtx->PushMessage (GL_DEBUG_SOURCE_APPLICATION, GL_DEBUG_TYPE_ERROR, 0, GL_DEBUG_SEVERITY_HIGH,
|
||||
TCollection_AsciiString ("Unable to copy cubemap mipmap level. Error ") + OpenGl_Context::FormatGlError (anErr));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (aLevelIter == 0)
|
||||
@ -395,6 +518,11 @@ bool OpenGl_PBREnvironment::processSpecIBLMap (const Handle(OpenGl_Context)& the
|
||||
}
|
||||
}
|
||||
|
||||
if (theDrawParams != NULL)
|
||||
{
|
||||
myVBO.UnbindAttribute (theCtx, Graphic3d_TOA_POS);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
@ -402,23 +530,44 @@ bool OpenGl_PBREnvironment::processSpecIBLMap (const Handle(OpenGl_Context)& the
|
||||
// function : checkFBOCompletness
|
||||
// purpose :
|
||||
// =======================================================================
|
||||
bool OpenGl_PBREnvironment::checkFBOComplentess (const Handle(OpenGl_Context)& theCtx) const
|
||||
bool OpenGl_PBREnvironment::checkFBOComplentess (const Handle(OpenGl_Context)& theCtx)
|
||||
{
|
||||
theCtx->arbFBO->glBindFramebuffer (GL_DRAW_FRAMEBUFFER, myFBO);
|
||||
theCtx->arbFBO->glFramebufferTexture2D (GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D,
|
||||
myCanRenderFloat = true;
|
||||
theCtx->arbFBO->glBindFramebuffer (GL_FRAMEBUFFER, myFBO);
|
||||
theCtx->arbFBO->glFramebufferTexture2D (GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D,
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseSH].TextureId(), 0);
|
||||
if (theCtx->arbFBO->glCheckFramebufferStatus (GL_DRAW_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE)
|
||||
if (theCtx->arbFBO->glCheckFramebufferStatus (GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE)
|
||||
{
|
||||
return false;
|
||||
// Some WebGL 1.0 and OpenGL ES 2.0 implementations support float textures which cannot be used as render targets.
|
||||
// This capability could be exposed by WEBGL_color_buffer_float/EXT_color_buffer_float extensions,
|
||||
// but the simplest way is just to check FBO status.
|
||||
// The fallback solution involves rendering into RGBA8 texture with floating values packed,
|
||||
// and using glReadPixels() + conversion + texture upload of computed values.
|
||||
myCanRenderFloat = false;
|
||||
theCtx->arbFBO->glFramebufferTexture2D (GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D,
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_DiffuseFallback].TextureId(), 0);
|
||||
if (theCtx->arbFBO->glCheckFramebufferStatus (GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE)
|
||||
{
|
||||
Message::SendTrace() << "OpenGl_PBREnvironment, incomplete FBO for diffuse map";
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
for (Standard_Integer aSideIter = 0; aSideIter < 6; ++aSideIter)
|
||||
{
|
||||
for (unsigned int aLevel = 0; aLevel < mySpecMapLevelsNumber; ++aLevel)
|
||||
{
|
||||
theCtx->arbFBO->glFramebufferTexture2D (GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_CUBE_MAP_POSITIVE_X + aSideIter,
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_Specular].TextureId(), aLevel);
|
||||
if (theCtx->arbFBO->glCheckFramebufferStatus (GL_DRAW_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE)
|
||||
if (!theCtx->hasFboRenderMipmap
|
||||
&& aLevel != 0)
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
theCtx->arbFBO->glFramebufferTexture2D (GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_CUBE_MAP_POSITIVE_X + aSideIter,
|
||||
myIBLMaps[OpenGl_TypeOfIBLMap_Specular].TextureId(), aLevel);
|
||||
if (theCtx->arbFBO->glCheckFramebufferStatus (GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE)
|
||||
{
|
||||
Message::SendTrace() << "OpenGl_PBREnvironment, incomplete FBO for specular map " << aSideIter << " "<< aLevel;
|
||||
return false;
|
||||
}
|
||||
}
|
||||
@ -439,21 +588,21 @@ void OpenGl_PBREnvironment::bake (const Handle(OpenGl_Context)& theCtx,
|
||||
Standard_ShortReal theProbability)
|
||||
{
|
||||
myIsNeededToBeBound = Standard_True;
|
||||
if (!theCtx->ShaderManager()->BindPBREnvBakingProgram())
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
theEnvMap->Bind (theCtx, theCtx->PBRSpecIBLMapTexUnit());
|
||||
theCtx->ActiveProgram()->SetSampler (theCtx, "uEnvMap", theCtx->PBRSpecIBLMapTexUnit());
|
||||
theCtx->ActiveProgram()->SetUniform (theCtx, "uZCoeff", theZIsInverted ? -1 : 1);
|
||||
theCtx->ActiveProgram()->SetUniform (theCtx, "uYCoeff", theIsTopDown ? 1 : -1);
|
||||
theCtx->arbFBO->glBindFramebuffer (GL_DRAW_FRAMEBUFFER, myFBO);
|
||||
myVBO.BindAttribute (theCtx, Graphic3d_TOA_POS);
|
||||
theCtx->arbFBO->glBindFramebuffer (GL_FRAMEBUFFER, myFBO);
|
||||
|
||||
OSD_Timer aTimer;
|
||||
aTimer.Start();
|
||||
if (processSpecIBLMap (theCtx, true, theEnvMap->SizeX(), theSpecNbSamples, theProbability)
|
||||
&& processDiffIBLMap (theCtx, true, theDiffNbSamples))
|
||||
BakingParams aDrawParams;
|
||||
aDrawParams.NbSpecSamples = theSpecNbSamples;
|
||||
aDrawParams.NbDiffSamples = theDiffNbSamples;
|
||||
aDrawParams.EnvMapSize = theEnvMap->SizeX();
|
||||
aDrawParams.Probability = theProbability;
|
||||
aDrawParams.IsZInverted = theZIsInverted;
|
||||
aDrawParams.IsTopDown = theIsTopDown;
|
||||
if (processSpecIBLMap (theCtx, &aDrawParams)
|
||||
&& processDiffIBLMap (theCtx, &aDrawParams))
|
||||
{
|
||||
Message::SendTrace(TCollection_AsciiString()
|
||||
+ "IBL " + myIBLMaps[OpenGl_TypeOfIBLMap_Specular].SizeX() + "x" + myIBLMaps[OpenGl_TypeOfIBLMap_Specular].SizeY()
|
||||
@ -467,7 +616,6 @@ void OpenGl_PBREnvironment::bake (const Handle(OpenGl_Context)& theCtx,
|
||||
clear (theCtx, Graphic3d_Vec3(1.0f));
|
||||
}
|
||||
|
||||
myVBO.UnbindAttribute (theCtx, Graphic3d_TOA_POS);
|
||||
theEnvMap->Unbind (theCtx, theCtx->PBREnvLUTTexUnit());
|
||||
}
|
||||
|
||||
@ -479,9 +627,9 @@ void OpenGl_PBREnvironment::clear (const Handle(OpenGl_Context)& theCtx,
|
||||
const Graphic3d_Vec3& theColor)
|
||||
{
|
||||
myIsNeededToBeBound = Standard_True;
|
||||
theCtx->arbFBO->glBindFramebuffer (GL_DRAW_FRAMEBUFFER, myFBO);
|
||||
theCtx->arbFBO->glBindFramebuffer (GL_FRAMEBUFFER, myFBO);
|
||||
theCtx->core11fwd->glClearColor (theColor.r(), theColor.g(), theColor.b(), 1.f);
|
||||
|
||||
processSpecIBLMap (theCtx, false);
|
||||
processDiffIBLMap (theCtx, false);
|
||||
processSpecIBLMap (theCtx, NULL);
|
||||
processDiffIBLMap (theCtx, NULL);
|
||||
}
|
||||
|
@ -134,7 +134,22 @@ private:
|
||||
enum OpenGl_TypeOfIBLMap
|
||||
{
|
||||
OpenGl_TypeOfIBLMap_DiffuseSH,
|
||||
OpenGl_TypeOfIBLMap_Specular
|
||||
OpenGl_TypeOfIBLMap_Specular,
|
||||
OpenGl_TypeOfIBLMap_DiffuseFallback,
|
||||
};
|
||||
|
||||
//! Parameters for baking IBL.
|
||||
struct BakingParams
|
||||
{
|
||||
Standard_Size NbSpecSamples;
|
||||
Standard_Size NbDiffSamples;
|
||||
Standard_Integer EnvMapSize;
|
||||
Standard_ShortReal Probability;
|
||||
Standard_Boolean IsZInverted;
|
||||
Standard_Boolean IsTopDown;
|
||||
|
||||
BakingParams()
|
||||
: NbSpecSamples (0), NbDiffSamples (0), EnvMapSize (1024), Probability (1.0f), IsZInverted (false), IsTopDown (false) {}
|
||||
};
|
||||
|
||||
//! Initializes all textures.
|
||||
@ -156,23 +171,19 @@ private:
|
||||
//! @return false in case of failed baking or clearing
|
||||
//! Warning! Requires using of OpenGl_PBREnvironmentSentry.
|
||||
bool processDiffIBLMap (const Handle(OpenGl_Context)& theCtx,
|
||||
Standard_Boolean theIsDrawAction,
|
||||
Standard_Size theNbSamples = 0);
|
||||
const BakingParams* theDrawParams);
|
||||
|
||||
//! Responses for specular IBL map processing.
|
||||
//! @return false in case of failed baking or clearing
|
||||
//! Warning! Requires using of OpenGl_PBREnvironmentSentry.
|
||||
bool processSpecIBLMap (const Handle(OpenGl_Context)& theCtx,
|
||||
Standard_Boolean theIsDrawAction,
|
||||
Standard_Integer theEnvMapSize = 1024,
|
||||
Standard_Size theNbSamples = 0,
|
||||
Standard_ShortReal theProbability = 1.f);
|
||||
const BakingParams* theDrawParams);
|
||||
|
||||
//! Checks completeness of frame buffer object for all targets
|
||||
//! (all cube map sides and levels of IBL maps).
|
||||
//! @return false in case of uncompleted frame buffer object.
|
||||
//! Warning! Requires using of OpenGl_PBREnvironmentSentry.
|
||||
bool checkFBOComplentess (const Handle(OpenGl_Context)& theCtx) const;
|
||||
bool checkFBOComplentess (const Handle(OpenGl_Context)& theCtx);
|
||||
|
||||
//! Version of 'Bake' without OpenGl_PBREnvironmentSetnry.
|
||||
//! Warning! Requires using of OpenGl_PBREnvironmentSentry.
|
||||
@ -194,12 +205,13 @@ private:
|
||||
unsigned int myPow2Size; //!< size of IBL maps sides (real size can be calculated as 2^myPow2Size)
|
||||
unsigned int mySpecMapLevelsNumber; //!< number of mipmap levels used in specular IBL map
|
||||
|
||||
OpenGl_Texture myIBLMaps[2]; //!< IBL maps
|
||||
OpenGl_Texture myIBLMaps[3]; //!< IBL maps
|
||||
OpenGl_VertexBuffer myVBO; //!< vertex buffer object of screen rectangular
|
||||
GLuint myFBO; //!< frame buffer object to generate or clear IBL maps
|
||||
|
||||
Standard_Boolean myIsComplete; //!< completeness of PBR environment
|
||||
Standard_Boolean myIsNeededToBeBound; //!< indicates whether IBL map's textures have to be bound or it is not obligate
|
||||
Standard_Boolean myCanRenderFloat; //!< indicates whether driver supports rendering into floating point texture or not
|
||||
|
||||
};
|
||||
|
||||
|
@ -1764,10 +1764,17 @@ int OpenGl_ShaderManager::defaultGlslVersion (const Handle(Graphic3d_ShaderProgr
|
||||
}
|
||||
else
|
||||
{
|
||||
if (theProgram->IsPBR()
|
||||
&& myContext->IsGlGreaterEqual (3, 0))
|
||||
if (theProgram->IsPBR())
|
||||
{
|
||||
theProgram->SetHeader ("#version 300 es");
|
||||
if (myContext->IsGlGreaterEqual (3, 0))
|
||||
{
|
||||
theProgram->SetHeader ("#version 300 es");
|
||||
}
|
||||
else if (myContext->CheckExtension ("GL_EXT_shader_texture_lod"))
|
||||
{
|
||||
theProgram->SetHeader ("#extension GL_EXT_shader_texture_lod : enable\n"
|
||||
"#define textureCubeLod textureCubeLodEXT");
|
||||
}
|
||||
}
|
||||
if ((theBits & OpenGl_PO_WriteOit) != 0
|
||||
|| (theBits & OpenGl_PO_OitDepthPeeling) != 0
|
||||
@ -3165,8 +3172,9 @@ Standard_Boolean OpenGl_ShaderManager::prepareStdProgramBoundBox()
|
||||
// function : preparePBREnvBakingProgram
|
||||
// purpose :
|
||||
// =======================================================================
|
||||
Standard_Boolean OpenGl_ShaderManager::preparePBREnvBakingProgram()
|
||||
Standard_Boolean OpenGl_ShaderManager::preparePBREnvBakingProgram (Standard_Integer theIndex)
|
||||
{
|
||||
Standard_ASSERT_RAISE (theIndex >= 0 && theIndex <= 2,"");
|
||||
Handle(Graphic3d_ShaderProgram) aProgramSrc = new Graphic3d_ShaderProgram();
|
||||
OpenGl_ShaderObject::ShaderVariableList aUniforms, aStageInOuts;
|
||||
|
||||
@ -3177,16 +3185,27 @@ Standard_Boolean OpenGl_ShaderManager::preparePBREnvBakingProgram()
|
||||
TCollection_AsciiString aSrcFrag = TCollection_AsciiString()
|
||||
+ THE_FUNC_cubemap_vector_transform
|
||||
+ Shaders_PBRDistribution_glsl
|
||||
+ ((theIndex == 0 || theIndex == 2) ? "\n#define THE_TO_BAKE_DIFFUSE\n" : "\n#define THE_TO_BAKE_SPECULAR\n")
|
||||
+ (theIndex == 2 ? "\n#define THE_TO_PACK_FLOAT\n" : "")
|
||||
+ Shaders_PBREnvBaking_fs;
|
||||
|
||||
// constant array definition requires OpenGL 2.1+ or OpenGL ES 3.0+
|
||||
#if defined(GL_ES_VERSION_2_0)
|
||||
aProgramSrc->SetHeader ("#version 300 es");
|
||||
if (myContext->IsGlGreaterEqual (3, 0))
|
||||
{
|
||||
aProgramSrc->SetHeader ("#version 300 es");
|
||||
}
|
||||
else if (myContext->CheckExtension ("GL_EXT_shader_texture_lod"))
|
||||
{
|
||||
aProgramSrc->SetHeader ("#extension GL_EXT_shader_texture_lod : enable\n"
|
||||
"#define textureCubeLod textureCubeLodEXT");
|
||||
}
|
||||
#else
|
||||
aProgramSrc->SetHeader ("#version 120");
|
||||
#endif
|
||||
|
||||
defaultGlslVersion (aProgramSrc, "pbr_env_baking", 0);
|
||||
static const char* THE_BAKE_NAMES[3] = { "pbr_env_baking_diffuse", "pbr_env_baking_specular", "pbr_env_baking_difffallback" };
|
||||
defaultGlslVersion (aProgramSrc, THE_BAKE_NAMES[theIndex], 0);
|
||||
aProgramSrc->SetDefaultSampler (false);
|
||||
aProgramSrc->SetNbLightsMax (0);
|
||||
aProgramSrc->SetNbShadowMaps (0);
|
||||
@ -3195,12 +3214,28 @@ Standard_Boolean OpenGl_ShaderManager::preparePBREnvBakingProgram()
|
||||
aProgramSrc->AttachShader (OpenGl_ShaderObject::CreateFromSource (aSrcVert, Graphic3d_TOS_VERTEX, aUniforms, aStageInOuts));
|
||||
aProgramSrc->AttachShader (OpenGl_ShaderObject::CreateFromSource (aSrcFrag, Graphic3d_TOS_FRAGMENT, aUniforms, aStageInOuts));
|
||||
TCollection_AsciiString aKey;
|
||||
if (!Create (aProgramSrc, aKey, myPBREnvBakingProgram))
|
||||
if (!Create (aProgramSrc, aKey, myPBREnvBakingProgram[theIndex]))
|
||||
{
|
||||
myPBREnvBakingProgram = new OpenGl_ShaderProgram(); // just mark as invalid
|
||||
myPBREnvBakingProgram[theIndex] = new OpenGl_ShaderProgram(); // just mark as invalid
|
||||
return Standard_False;
|
||||
}
|
||||
|
||||
if (theIndex == 0
|
||||
|| theIndex == 2)
|
||||
{
|
||||
// workaround for old GLSL - load constants as uniform
|
||||
myContext->BindProgram (myPBREnvBakingProgram[theIndex]);
|
||||
const float aSHBasisFuncCoeffs[9] =
|
||||
{
|
||||
0.282095f * 0.282095f, 0.488603f * 0.488603f, 0.488603f * 0.488603f, 0.488603f * 0.488603f,
|
||||
1.092548f * 1.092548f, 1.092548f * 1.092548f, 1.092548f * 1.092548f, 0.315392f * 0.315392f, 0.546274f * 0.546274f
|
||||
};
|
||||
const float aSHCosCoeffs[9] = { 3.141593f, 2.094395f, 2.094395f, 2.094395f, 0.785398f, 0.785398f, 0.785398f, 0.785398f, 0.785398f };
|
||||
myPBREnvBakingProgram[theIndex]->SetUniform (myContext, myPBREnvBakingProgram[theIndex]->GetUniformLocation (myContext, "aSHBasisFuncCoeffs"), 9, aSHBasisFuncCoeffs);
|
||||
myPBREnvBakingProgram[theIndex]->SetUniform (myContext, myPBREnvBakingProgram[theIndex]->GetUniformLocation (myContext, "aSHCosCoeffs"), 9, aSHCosCoeffs);
|
||||
myContext->BindProgram (NULL);
|
||||
}
|
||||
|
||||
return Standard_True;
|
||||
}
|
||||
|
||||
|
@ -277,13 +277,13 @@ public:
|
||||
const Handle(OpenGl_VertexBuffer)& BoundBoxVertBuffer() const { return myBoundBoxVertBuffer; }
|
||||
|
||||
//! Bind program for IBL maps generation in PBR pipeline.
|
||||
Standard_Boolean BindPBREnvBakingProgram()
|
||||
Standard_Boolean BindPBREnvBakingProgram (Standard_Integer theIndex)
|
||||
{
|
||||
if (myPBREnvBakingProgram.IsNull())
|
||||
if (myPBREnvBakingProgram[theIndex].IsNull())
|
||||
{
|
||||
preparePBREnvBakingProgram();
|
||||
preparePBREnvBakingProgram (theIndex);
|
||||
}
|
||||
return myContext->BindProgram (myPBREnvBakingProgram);
|
||||
return myContext->BindProgram (myPBREnvBakingProgram[theIndex]);
|
||||
}
|
||||
|
||||
//! Generates shader program to render environment cubemap as background.
|
||||
@ -814,7 +814,7 @@ protected:
|
||||
Standard_Integer theBits);
|
||||
|
||||
//! Prepare GLSL source for IBL generation used in PBR pipeline.
|
||||
Standard_EXPORT Standard_Boolean preparePBREnvBakingProgram();
|
||||
Standard_EXPORT Standard_Boolean preparePBREnvBakingProgram (Standard_Integer theIndex);
|
||||
|
||||
//! Checks whether one of PBR shading models is set as default model.
|
||||
Standard_Boolean IsPbrAllowed() const { return myShadingModel == Graphic3d_TOSM_PBR
|
||||
@ -883,7 +883,7 @@ protected:
|
||||
Handle(OpenGl_ShaderProgram) myOitDepthPeelingFlushProgram[2]; //!< standard program for OIT Depth Peeling flush (default and MSAA)
|
||||
OpenGl_MapOfShaderPrograms myMapOfLightPrograms; //!< map of lighting programs depending on lights configuration
|
||||
|
||||
Handle(OpenGl_ShaderProgram) myPBREnvBakingProgram;//!< program for IBL maps generation used in PBR pipeline
|
||||
Handle(OpenGl_ShaderProgram) myPBREnvBakingProgram[3]; //!< programs for IBL maps generation used in PBR pipeline (0 for Diffuse; 1 for Specular; 2 for fallback)
|
||||
Handle(Graphic3d_ShaderProgram) myBgCubeMapProgram; //!< program for background cubemap rendering
|
||||
|
||||
Handle(OpenGl_ShaderProgram) myStereoPrograms[Graphic3d_StereoMode_NB]; //!< standard stereo programs
|
||||
|
@ -16,7 +16,7 @@
|
||||
|
||||
#include <OpenGl_ArbFBO.hxx>
|
||||
#include <OpenGl_Context.hxx>
|
||||
#include <OpenGl_GlCore32.hxx>
|
||||
#include <OpenGl_GlCore45.hxx>
|
||||
#include <OpenGl_Sampler.hxx>
|
||||
#include <Graphic3d_TextureParams.hxx>
|
||||
#include <TCollection_ExtendedString.hxx>
|
||||
@ -1235,3 +1235,72 @@ Standard_Size OpenGl_Texture::EstimatedDataSize() const
|
||||
}
|
||||
return aSize;
|
||||
}
|
||||
|
||||
// =======================================================================
|
||||
// function : ImageDump
|
||||
// purpose :
|
||||
// =======================================================================
|
||||
bool OpenGl_Texture::ImageDump (Image_PixMap& theImage,
|
||||
const Handle(OpenGl_Context)& theCtx,
|
||||
Graphic3d_TextureUnit theTexUnit,
|
||||
Standard_Integer theLevel,
|
||||
Standard_Integer theCubeSide) const
|
||||
{
|
||||
#if !defined(GL_ES_VERSION_2_0)
|
||||
const OpenGl_TextureFormat aFormat = OpenGl_TextureFormat::FindSizedFormat (theCtx, mySizedFormat);
|
||||
if (theCtx.IsNull()
|
||||
|| !IsValid()
|
||||
|| theLevel < 0
|
||||
|| !aFormat.IsValid()
|
||||
|| aFormat.ImageFormat() == Image_Format_UNKNOWN
|
||||
|| (myTarget == GL_TEXTURE_CUBE_MAP
|
||||
&& (theCubeSide < 0 || theCubeSide > 5)))
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
GLenum aTarget = myTarget;
|
||||
Graphic3d_Vec2i aSize (mySizeX, mySizeY);
|
||||
if (myTarget == GL_TEXTURE_CUBE_MAP)
|
||||
{
|
||||
aTarget = GL_TEXTURE_CUBE_MAP_POSITIVE_X + theCubeSide;
|
||||
}
|
||||
for (Standard_Integer aMipIter = 0; aMipIter < theLevel; ++aMipIter)
|
||||
{
|
||||
aSize /= 2;
|
||||
if (aSize.x() == 0) { aSize.x() = 1; }
|
||||
if (aSize.y() == 0) { aSize.y() = 1; }
|
||||
}
|
||||
if (!theImage.InitTrash (aFormat.ImageFormat(), aSize.x(), aSize.y()))
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
const GLint anAligment = Min (GLint(theImage.MaxRowAligmentBytes()), 8); // limit to 8 bytes for OpenGL
|
||||
theCtx->core11fwd->glPixelStorei (GL_PACK_ALIGNMENT, anAligment);
|
||||
theCtx->core11fwd->glPixelStorei (GL_PACK_ROW_LENGTH, 0);
|
||||
// glGetTextureImage() allows avoiding to binding texture id, but apparently requires clean FBO binding state...
|
||||
//if (theCtx->core45 != NULL) { theCtx->core45->glGetTextureImage (myTextureId, theLevel, aFormat.PixelFormat(), aFormat.DataType(), (GLsizei )theImage.SizeBytes(), theImage.ChangeData()); } else
|
||||
{
|
||||
Bind (theCtx, theTexUnit);
|
||||
theCtx->core11fwd->glGetTexImage (aTarget, theLevel, aFormat.PixelFormat(), aFormat.DataType(), theImage.ChangeData());
|
||||
Unbind (theCtx, theTexUnit);
|
||||
}
|
||||
if (theImage.Format() != aFormat.ImageFormat())
|
||||
{
|
||||
Image_PixMap::SwapRgbaBgra (theImage);
|
||||
}
|
||||
|
||||
const bool hasErrors = theCtx->ResetErrors (true);
|
||||
theCtx->core11fwd->glPixelStorei (GL_PACK_ALIGNMENT, 1);
|
||||
return !hasErrors;
|
||||
#else
|
||||
// glGetTexImage() is unavailable in OpenGL ES
|
||||
(void )theImage;
|
||||
(void )theCtx;
|
||||
(void )theTexUnit;
|
||||
(void )theLevel;
|
||||
(void )theCubeSide;
|
||||
return false;
|
||||
#endif
|
||||
}
|
||||
|
@ -191,6 +191,19 @@ public:
|
||||
//! Returns TRUE for point sprite texture.
|
||||
virtual bool IsPointSprite() const { return false; }
|
||||
|
||||
//! Auxiliary method for making an image dump from texture data.
|
||||
//! @param theImage [out] result image data (will be overridden)
|
||||
//! @param theCtx [in] active GL context
|
||||
//! @param theTexUnit [in] texture slot to use
|
||||
//! @param theLevel [in] mipmap level to dump
|
||||
//! @param theCubeSide [in] cubemap side to dump within [0, 5] range
|
||||
//! @return FALSE on error
|
||||
Standard_EXPORT bool ImageDump (Image_PixMap& theImage,
|
||||
const Handle(OpenGl_Context)& theCtx,
|
||||
Graphic3d_TextureUnit theTexUnit,
|
||||
Standard_Integer theLevel = 0,
|
||||
Standard_Integer theCubeSide = 0) const;
|
||||
|
||||
public:
|
||||
|
||||
Standard_DEPRECATED("Deprecated method, OpenGl_TextureFormat::FindFormat() should be used instead")
|
||||
|
@ -122,6 +122,7 @@ OpenGl_TextureFormat OpenGl_TextureFormat::FindFormat (const Handle(OpenGl_Conte
|
||||
bool theIsColorMap)
|
||||
{
|
||||
OpenGl_TextureFormat aFormat;
|
||||
aFormat.SetImageFormat (theFormat);
|
||||
switch (theFormat)
|
||||
{
|
||||
case Image_Format_GrayF:
|
||||
@ -414,6 +415,7 @@ OpenGl_TextureFormat OpenGl_TextureFormat::FindSizedFormat (const Handle(OpenGl_
|
||||
aFormat.SetInternalFormat (theSizedFormat);
|
||||
aFormat.SetPixelFormat (GL_RGBA);
|
||||
aFormat.SetDataType (GL_FLOAT);
|
||||
aFormat.SetImageFormat (Image_Format_RGBAF);
|
||||
return aFormat;
|
||||
}
|
||||
case GL_R32F:
|
||||
@ -422,6 +424,7 @@ OpenGl_TextureFormat OpenGl_TextureFormat::FindSizedFormat (const Handle(OpenGl_
|
||||
aFormat.SetInternalFormat (theSizedFormat);
|
||||
aFormat.SetPixelFormat (GL_RED);
|
||||
aFormat.SetDataType (GL_FLOAT);
|
||||
aFormat.SetImageFormat (Image_Format_GrayF);
|
||||
return aFormat;
|
||||
}
|
||||
case GL_RG32F:
|
||||
@ -430,6 +433,7 @@ OpenGl_TextureFormat OpenGl_TextureFormat::FindSizedFormat (const Handle(OpenGl_
|
||||
aFormat.SetInternalFormat (theSizedFormat);
|
||||
aFormat.SetPixelFormat (GL_RG);
|
||||
aFormat.SetDataType (GL_FLOAT);
|
||||
aFormat.SetImageFormat (Image_Format_RGF);
|
||||
return aFormat;
|
||||
}
|
||||
case GL_RGBA16F:
|
||||
@ -438,6 +442,7 @@ OpenGl_TextureFormat OpenGl_TextureFormat::FindSizedFormat (const Handle(OpenGl_
|
||||
aFormat.SetInternalFormat (theSizedFormat);
|
||||
aFormat.SetPixelFormat (GL_RGBA);
|
||||
aFormat.SetDataType (GL_HALF_FLOAT);
|
||||
aFormat.SetImageFormat (Image_Format_RGBAF);
|
||||
if (theCtx->hasHalfFloatBuffer == OpenGl_FeatureInExtensions)
|
||||
{
|
||||
#if defined(GL_ES_VERSION_2_0)
|
||||
@ -454,6 +459,7 @@ OpenGl_TextureFormat OpenGl_TextureFormat::FindSizedFormat (const Handle(OpenGl_
|
||||
aFormat.SetInternalFormat (theSizedFormat);
|
||||
aFormat.SetPixelFormat (GL_RED);
|
||||
aFormat.SetDataType (GL_HALF_FLOAT);
|
||||
aFormat.SetImageFormat (Image_Format_GrayF);
|
||||
if (theCtx->hasHalfFloatBuffer == OpenGl_FeatureInExtensions)
|
||||
{
|
||||
#if defined(GL_ES_VERSION_2_0)
|
||||
@ -472,6 +478,7 @@ OpenGl_TextureFormat OpenGl_TextureFormat::FindSizedFormat (const Handle(OpenGl_
|
||||
aFormat.SetInternalFormat (theSizedFormat);
|
||||
aFormat.SetPixelFormat (GL_RGBA);
|
||||
aFormat.SetDataType (GL_UNSIGNED_BYTE);
|
||||
aFormat.SetImageFormat (Image_Format_RGBA);
|
||||
if (theSizedFormat == GL_SRGB8_ALPHA8
|
||||
&& !theCtx->ToRenderSRGB())
|
||||
{
|
||||
@ -487,6 +494,7 @@ OpenGl_TextureFormat OpenGl_TextureFormat::FindSizedFormat (const Handle(OpenGl_
|
||||
aFormat.SetInternalFormat (theSizedFormat);
|
||||
aFormat.SetPixelFormat (GL_RGB);
|
||||
aFormat.SetDataType (GL_UNSIGNED_BYTE);
|
||||
aFormat.SetImageFormat (Image_Format_RGB);
|
||||
if (theSizedFormat == GL_SRGB8
|
||||
&& !theCtx->ToRenderSRGB())
|
||||
{
|
||||
|
@ -66,7 +66,7 @@ public:
|
||||
public:
|
||||
|
||||
//! Empty constructor (invalid texture format).
|
||||
OpenGl_TextureFormat() : myInternalFormat (0), myPixelFormat (0), myDataType (0), myNbComponents (0) {}
|
||||
OpenGl_TextureFormat() : myImageFormat (Image_Format_UNKNOWN), myInternalFormat (0), myPixelFormat (0), myDataType (0), myNbComponents (0) {}
|
||||
|
||||
//! Return TRUE if format is defined.
|
||||
bool IsValid() const
|
||||
@ -107,6 +107,12 @@ public:
|
||||
|| myInternalFormat == GL_SRGB8_ALPHA8;
|
||||
}
|
||||
|
||||
//! Returns image format (best match or Image_Format_UNKNOWN if no suitable fit).
|
||||
Image_Format ImageFormat() const { return myImageFormat; }
|
||||
|
||||
//! Sets image format.
|
||||
void SetImageFormat (Image_Format theFormat) { myImageFormat = theFormat; }
|
||||
|
||||
public:
|
||||
|
||||
//! Returns OpenGL internal format of the pixel data (example: GL_R32F).
|
||||
@ -117,6 +123,7 @@ public:
|
||||
|
||||
private:
|
||||
|
||||
Image_Format myImageFormat; //!< image format
|
||||
GLint myInternalFormat; //!< OpenGL internal format of the pixel data
|
||||
GLenum myPixelFormat; //!< OpenGL pixel format
|
||||
GLint myDataType; //!< OpenGL data type of input pixel data
|
||||
|
@ -1194,14 +1194,34 @@ bool OpenGl_View::prepareFrameBuffers (Graphic3d_Camera::Projection& theProj)
|
||||
aParams->SetTextureUnit (aCtx->PBREnvLUTTexUnit());
|
||||
anEnvLUT = new OpenGl_Texture(THE_SHARED_ENV_LUT_KEY, aParams);
|
||||
Handle(Image_PixMap) aPixMap = new Image_PixMap();
|
||||
aPixMap->InitWrapper (Image_Format_RGF, (Standard_Byte*)Textures_EnvLUT, Textures_EnvLUTSize, Textures_EnvLUTSize);
|
||||
if (aCtx->arbTexRG)
|
||||
{
|
||||
aPixMap->InitWrapper (Image_Format_RGF, (Standard_Byte*)Textures_EnvLUT, Textures_EnvLUTSize, Textures_EnvLUTSize);
|
||||
}
|
||||
else
|
||||
{
|
||||
Image_PixMap aPixMapRG;
|
||||
aPixMapRG.InitWrapper (Image_Format_RGF, (Standard_Byte*)Textures_EnvLUT, Textures_EnvLUTSize, Textures_EnvLUTSize);
|
||||
aPixMap->InitZero (Image_Format_RGBAF, Textures_EnvLUTSize, Textures_EnvLUTSize);
|
||||
for (Standard_Size aRowIter = 0; aRowIter < aPixMapRG.SizeY(); ++aRowIter)
|
||||
{
|
||||
for (Standard_Size aColIter = 0; aColIter < aPixMapRG.SizeX(); ++aColIter)
|
||||
{
|
||||
const Image_ColorRGF& aPixelRG = aPixMapRG.Value<Image_ColorRGF> (aRowIter, aColIter);
|
||||
Image_ColorRGBAF& aPixelRGBA = aPixMap->ChangeValue<Image_ColorRGBAF> (aRowIter, aColIter);
|
||||
aPixelRGBA.r() = aPixelRG.r();
|
||||
aPixelRGBA.g() = aPixelRG.g();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
OpenGl_TextureFormat aTexFormat = OpenGl_TextureFormat::FindFormat (aCtx, aPixMap->Format(), false);
|
||||
#if defined(GL_ES_VERSION_2_0)
|
||||
// GL_RG32F is not texture-filterable format on OpenGL ES without OES_texture_float_linear extension.
|
||||
// GL_RG16F is texture-filterable since OpenGL ES 3.0 and can be initialized from 32-bit floats.
|
||||
// Note that it is expected that GL_RG16F has enough precision for this table, so that it can be used also on desktop OpenGL.
|
||||
//if (!aCtx->hasTexFloatLinear)
|
||||
aTexFormat.SetInternalFormat (GL_RG16F);
|
||||
aTexFormat.SetInternalFormat (aCtx->arbTexRG ? GL_RG16F : GL_RGBA16F);
|
||||
#endif
|
||||
if (!aTexFormat.IsValid()
|
||||
|| !anEnvLUT->Init (aCtx, aTexFormat, Graphic3d_Vec2i((Standard_Integer)Textures_EnvLUTSize), Graphic3d_TOT_2D, aPixMap.get()))
|
||||
|
@ -143,44 +143,49 @@ const int OccLightType_Spot = 3; //!< spot light source
|
||||
// Light sources
|
||||
uniform vec4 occLightAmbient; //!< Cumulative ambient color
|
||||
#if defined(THE_MAX_LIGHTS) && (THE_MAX_LIGHTS > 0)
|
||||
#if (THE_MAX_LIGHTS > 1)
|
||||
#define occLight_Index(theId) theId
|
||||
#else
|
||||
#define occLight_Index(theId) 0
|
||||
#endif
|
||||
uniform THE_PREC_ENUM int occLightSourcesCount; //!< Total number of light sources
|
||||
|
||||
//! Type of light source, int (see OccLightType enum).
|
||||
#define occLight_Type(theId) occLightSourcesTypes[theId]
|
||||
#define occLight_Type(theId) occLightSourcesTypes[occLight_Index(theId)]
|
||||
|
||||
//! Specular intensity (equals to diffuse), vec3.
|
||||
#define occLight_Specular(theId) occLightSources[theId * 4 + 0].rgb
|
||||
#define occLight_Specular(theId) occLightSources[occLight_Index(theId) * 4 + 0].rgb
|
||||
|
||||
//! Intensity of light source (>= 0), float.
|
||||
#define occLight_Intensity(theId) occLightSources[theId * 4 + 0].a
|
||||
#define occLight_Intensity(theId) occLightSources[occLight_Index(theId) * 4 + 0].a
|
||||
|
||||
//! Is light a headlight, bool? DEPRECATED method.
|
||||
#define occLight_IsHeadlight(theId) false
|
||||
|
||||
//! Position of specified light source or direction of directional light source, vec3.
|
||||
#define occLight_Position(theId) occLightSources[theId * 4 + 1].xyz
|
||||
#define occLight_Position(theId) occLightSources[occLight_Index(theId) * 4 + 1].xyz
|
||||
|
||||
//! Direction of specified spot light source, vec3.
|
||||
#define occLight_SpotDirection(theId) occLightSources[theId * 4 + 2].xyz
|
||||
#define occLight_SpotDirection(theId) occLightSources[occLight_Index(theId) * 4 + 2].xyz
|
||||
|
||||
//! Range on which point light source (positional or spot) can affect (>= 0), float.
|
||||
#define occLight_Range(theId) occLightSources[theId * 4 + 2].w
|
||||
#define occLight_Range(theId) occLightSources[occLight_Index(theId) * 4 + 2].w
|
||||
|
||||
//! Maximum spread angle of the spot light (in radians), float.
|
||||
#define occLight_SpotCutOff(theId) occLightSources[theId * 4 + 3].z
|
||||
#define occLight_SpotCutOff(theId) occLightSources[occLight_Index(theId) * 4 + 3].z
|
||||
|
||||
//! Attenuation of the spot light intensity (from 0 to 1), float.
|
||||
#define occLight_SpotExponent(theId) occLightSources[theId * 4 + 3].w
|
||||
#define occLight_SpotExponent(theId) occLightSources[occLight_Index(theId) * 4 + 3].w
|
||||
|
||||
#if !defined(THE_IS_PBR)
|
||||
//! Diffuse intensity (equals to Specular), vec3.
|
||||
#define occLight_Diffuse(theId) occLightSources[theId * 4 + 0].rgb
|
||||
#define occLight_Diffuse(theId) occLightSources[occLight_Index(theId) * 4 + 0].rgb
|
||||
|
||||
//! Const attenuation factor of positional light source, float.
|
||||
#define occLight_ConstAttenuation(theId) occLightSources[theId * 4 + 3].x
|
||||
#define occLight_ConstAttenuation(theId) occLightSources[occLight_Index(theId) * 4 + 3].x
|
||||
|
||||
//! Linear attenuation factor of positional light source, float.
|
||||
#define occLight_LinearAttenuation(theId) occLightSources[theId * 4 + 3].y
|
||||
#define occLight_LinearAttenuation(theId) occLightSources[occLight_Index(theId) * 4 + 3].y
|
||||
#endif
|
||||
#endif
|
||||
|
||||
|
@ -14,7 +14,7 @@ float occDirectionalLightShadow (in sampler2D theShadow,
|
||||
in int theId,
|
||||
in vec3 theNormal)
|
||||
{
|
||||
vec4 aPosLightSpace = PosLightSpace[theId];
|
||||
vec4 aPosLightSpace = PosLightSpace[occLight_Index(theId)];
|
||||
vec3 aLightDir = vec3 (occWorldViewMatrix * vec4 (occLight_Position (theId), 0.0));
|
||||
vec3 aProjCoords = (aPosLightSpace.xyz / aPosLightSpace.w) * 0.5 + vec3 (0.5);
|
||||
float aCurrentDepth = aProjCoords.z;
|
||||
|
@ -1,12 +1,22 @@
|
||||
THE_SHADER_IN vec3 ViewDirection; //!< direction of fetching from environment cubemap
|
||||
|
||||
#if (__VERSION__ >= 120)
|
||||
uniform int uSamplesNum; //!< number of samples in Monte-Carlo integration
|
||||
uniform int uCurrentLevel; //!< current level of specular IBL map (ignored in case of diffuse map's processing)
|
||||
uniform int uEnvMapSize; //!< one edge's size of source environtment map's zero mipmap level
|
||||
uniform int uYCoeff; //!< coefficient of Y controlling horizontal flip of cubemap
|
||||
uniform int uZCoeff; //!< coefficient of Z controlling vertical flip of cubemap
|
||||
#else
|
||||
const int uSamplesNum = 256;
|
||||
#endif
|
||||
uniform samplerCube uEnvMap; //!< source of baking (environment cubemap)
|
||||
|
||||
#ifdef THE_TO_BAKE_DIFFUSE
|
||||
uniform int uYCoeff; //!< coefficient of Y controlling horizontal flip of cubemap
|
||||
uniform int uZCoeff; //!< coefficient of Z controlling vertical flip of cubemap
|
||||
#endif
|
||||
|
||||
#ifdef THE_TO_BAKE_SPECULAR
|
||||
uniform int uCurrentLevel; //!< current level of specular IBL map (ignored in case of diffuse map's processing)
|
||||
uniform float uEnvSolidAngleSource; //!< source solid angle sample computed from one edge's size of source environment map's zero mipmap level
|
||||
#endif
|
||||
|
||||
//! Returns coordinates of point theNumber from Hammersley point set having size theSize.
|
||||
vec2 hammersley (in int theNumber,
|
||||
in int theSize)
|
||||
@ -18,7 +28,7 @@ vec2 hammersley (in int theNumber,
|
||||
{
|
||||
if (aNumber > 0)
|
||||
{
|
||||
aVanDerCorput += float(aNumber % 2) / float(aDenominator);
|
||||
aVanDerCorput += mod(float(aNumber), 2.0) / float(aDenominator);
|
||||
aNumber /= 2;
|
||||
aDenominator *= 2;
|
||||
}
|
||||
@ -62,6 +72,8 @@ vec3 fromTangentSpace (in vec3 theVector,
|
||||
return anX * theVector.x + anY * theVector.y + theNormal * theVector.z;
|
||||
}
|
||||
|
||||
#ifdef THE_TO_BAKE_DIFFUSE
|
||||
#if (__VERSION__ >= 120)
|
||||
const float aSHBasisFuncCoeffs[9] = float[9]
|
||||
(
|
||||
0.282095 * 0.282095,
|
||||
@ -74,7 +86,6 @@ const float aSHBasisFuncCoeffs[9] = float[9]
|
||||
0.315392 * 0.315392,
|
||||
0.546274 * 0.546274
|
||||
);
|
||||
|
||||
const float aSHCosCoeffs[9] = float[9]
|
||||
(
|
||||
3.141593,
|
||||
@ -87,76 +98,129 @@ const float aSHCosCoeffs[9] = float[9]
|
||||
0.785398,
|
||||
0.785398
|
||||
);
|
||||
#else
|
||||
uniform float aSHBasisFuncCoeffs[9];
|
||||
uniform float aSHCosCoeffs[9];
|
||||
#endif
|
||||
|
||||
//! Bakes diffuse IBL map's spherical harmonics coefficients.
|
||||
vec3 bakeDiffuseSH()
|
||||
{
|
||||
int anIndex = int(gl_FragCoord.x);
|
||||
vec3 aResult = vec3 (0.0);
|
||||
int anId = int(gl_FragCoord.x);
|
||||
float aCoef;
|
||||
#if (__VERSION__ >= 120)
|
||||
aCoef = aSHCosCoeffs[anId] * aSHBasisFuncCoeffs[anId];
|
||||
#else
|
||||
if (anId == 0) { aCoef = aSHCosCoeffs[0] * aSHBasisFuncCoeffs[0]; }
|
||||
else if (anId == 1) { aCoef = aSHCosCoeffs[1] * aSHBasisFuncCoeffs[1]; }
|
||||
else if (anId == 2) { aCoef = aSHCosCoeffs[2] * aSHBasisFuncCoeffs[2]; }
|
||||
else if (anId == 3) { aCoef = aSHCosCoeffs[3] * aSHBasisFuncCoeffs[3]; }
|
||||
else if (anId == 4) { aCoef = aSHCosCoeffs[4] * aSHBasisFuncCoeffs[4]; }
|
||||
else if (anId == 5) { aCoef = aSHCosCoeffs[5] * aSHBasisFuncCoeffs[5]; }
|
||||
else if (anId == 6) { aCoef = aSHCosCoeffs[6] * aSHBasisFuncCoeffs[6]; }
|
||||
else if (anId == 7) { aCoef = aSHCosCoeffs[7] * aSHBasisFuncCoeffs[7]; }
|
||||
else { aCoef = aSHCosCoeffs[8] * aSHBasisFuncCoeffs[8]; }
|
||||
#endif
|
||||
vec3 aRes = vec3 (0.0);
|
||||
for (int aSampleIter = 0; aSampleIter < uSamplesNum; ++aSampleIter)
|
||||
{
|
||||
vec2 aHammersleyPoint = hammersley (aSampleIter, uSamplesNum);
|
||||
vec3 aDirection = sphereUniformSample (aHammersleyPoint);
|
||||
vec3 aDir = sphereUniformSample (aHammersleyPoint);
|
||||
|
||||
vec3 aValue = occTextureCube (uEnvMap, cubemapVectorTransform (aDirection, uYCoeff, uZCoeff)).rgb;
|
||||
vec3 aVal = occTextureCube (uEnvMap, cubemapVectorTransform (aDir, uYCoeff, uZCoeff)).rgb;
|
||||
#if (__VERSION__ >= 120)
|
||||
float aFunc[9];
|
||||
aFunc[0] = 1.0;
|
||||
|
||||
float aBasisFunc[9];
|
||||
aBasisFunc[0] = 1.0;
|
||||
aFunc[1] = aDir.x;
|
||||
aFunc[2] = aDir.y;
|
||||
aFunc[3] = aDir.z;
|
||||
|
||||
aBasisFunc[1] = aDirection.x;
|
||||
aBasisFunc[2] = aDirection.y;
|
||||
aBasisFunc[3] = aDirection.z;
|
||||
aFunc[4] = aDir.x * aDir.z;
|
||||
aFunc[5] = aDir.y * aDir.z;
|
||||
aFunc[6] = aDir.x * aDir.y;
|
||||
|
||||
aBasisFunc[4] = aDirection.x * aDirection.z;
|
||||
aBasisFunc[5] = aDirection.y * aDirection.z;
|
||||
aBasisFunc[6] = aDirection.x * aDirection.y;
|
||||
aFunc[7] = 3.0 * aDir.z * aDir.z - 1.0;
|
||||
aFunc[8] = aDir.x * aDir.x - aDir.y * aDir.y;
|
||||
|
||||
aBasisFunc[7] = 3.0 * aDirection.z * aDirection.z - 1.0;
|
||||
aBasisFunc[8] = aDirection.x * aDirection.x - aDirection.y * aDirection.y;
|
||||
|
||||
aResult += aValue * aBasisFunc[anIndex];
|
||||
aRes += aVal * aFunc[anId];
|
||||
#else
|
||||
if (anId == 0) { aRes += aVal * 1.0; }
|
||||
else if (anId == 1) { aRes += aVal * aDir.x; }
|
||||
else if (anId == 2) { aRes += aVal * aDir.y; }
|
||||
else if (anId == 3) { aRes += aVal * aDir.z; }
|
||||
else if (anId == 4) { aRes += aVal * (aDir.x * aDir.z); }
|
||||
else if (anId == 5) { aRes += aVal * (aDir.y * aDir.z); }
|
||||
else if (anId == 6) { aRes += aVal * (aDir.x * aDir.y); }
|
||||
else if (anId == 7) { aRes += aVal * (3.0 * aDir.z * aDir.z - 1.0); }
|
||||
else { aRes += aVal * (aDir.x * aDir.x - aDir.y * aDir.y); }
|
||||
#endif
|
||||
}
|
||||
|
||||
aResult *= 4.0 * aSHCosCoeffs[anIndex] * aSHBasisFuncCoeffs[anIndex] / float(uSamplesNum);
|
||||
return aResult;
|
||||
return 4.0 * aRes * aCoef / float(uSamplesNum);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef THE_TO_BAKE_SPECULAR
|
||||
//! Computes a single sample for specular IBL map.
|
||||
vec4 specularMapSample (in vec3 theNormal,
|
||||
in float theRoughness,
|
||||
in int theNumber,
|
||||
in int theSize)
|
||||
{
|
||||
vec2 aHammersleyPoint = hammersley (theNumber, theSize);
|
||||
vec3 aHalf = importanceSample (aHammersleyPoint, occRoughness (theRoughness));
|
||||
float aHdotV = aHalf.z;
|
||||
aHalf = fromTangentSpace (aHalf, theNormal);
|
||||
vec3 aLight = -reflect (theNormal, aHalf);
|
||||
float aNdotL = dot (aLight, theNormal);
|
||||
if (aNdotL > 0.0)
|
||||
{
|
||||
float aSolidAngleSample = 1.0 / (float(theSize) * (occPBRDistribution (aHdotV, theRoughness) * 0.25 + 0.0001) + 0.0001);
|
||||
float aLod = (theRoughness == 0.0) ? 0.0 : 0.5 * log2 (aSolidAngleSample / uEnvSolidAngleSource);
|
||||
return vec4 (occTextureCubeLod (uEnvMap, aLight, aLod).rgb * aNdotL, aNdotL);
|
||||
}
|
||||
return vec4 (0.0);
|
||||
}
|
||||
|
||||
//! Bakes specular IBL map.
|
||||
vec3 bakeSpecularMap (in vec3 theNormal,
|
||||
in float theRoughness)
|
||||
{
|
||||
vec3 aResult = vec3(0.0);
|
||||
float aWeightSum = 0.0;
|
||||
int aSamplesNum = (theRoughness == 0.0) ? 1 : uSamplesNum;
|
||||
float aSolidAngleSource = 4.0 * PI / (6.0 * float(uEnvMapSize * uEnvMapSize));
|
||||
for (int aSampleIter = 0; aSampleIter < aSamplesNum; ++aSampleIter)
|
||||
vec4 aResult = vec4(0.0);
|
||||
if (theRoughness == 0.0)
|
||||
{
|
||||
vec2 aHammersleyPoint = hammersley (aSampleIter, aSamplesNum);
|
||||
vec3 aHalf = importanceSample (aHammersleyPoint, occRoughness (theRoughness));
|
||||
float aHdotV = aHalf.z;
|
||||
aHalf = fromTangentSpace (aHalf, theNormal);
|
||||
vec3 aLight = -reflect (theNormal, aHalf);
|
||||
float aNdotL = dot (aLight, theNormal);
|
||||
if (aNdotL > 0.0)
|
||||
aResult = specularMapSample (theNormal, theRoughness, 0, 1);
|
||||
}
|
||||
else
|
||||
{
|
||||
for (int aSampleIter = 0; aSampleIter < uSamplesNum; ++aSampleIter)
|
||||
{
|
||||
float aSolidAngleSample = 1.0 / (float(aSamplesNum) * (occPBRDistribution (aHdotV, theRoughness) * 0.25 + 0.0001) + 0.0001);
|
||||
float aLod = (theRoughness == 0.0) ? 0.0 : 0.5 * log2 (aSolidAngleSample / aSolidAngleSource);
|
||||
aResult += occTextureCubeLod (uEnvMap, aLight, aLod).rgb * aNdotL;
|
||||
aWeightSum += aNdotL;
|
||||
aResult += specularMapSample (theNormal, theRoughness, aSampleIter, uSamplesNum);
|
||||
}
|
||||
}
|
||||
return aResult / aWeightSum;
|
||||
return aResult.xyz / aResult.w;
|
||||
}
|
||||
#endif
|
||||
|
||||
void main()
|
||||
{
|
||||
vec3 aViewDirection = normalize (ViewDirection);
|
||||
if (occNbSpecIBLLevels == 0)
|
||||
{
|
||||
occSetFragColor (vec4 (bakeDiffuseSH (), 1.0));
|
||||
}
|
||||
else
|
||||
{
|
||||
occSetFragColor (vec4 (bakeSpecularMap (aViewDirection, float(uCurrentLevel) / float(occNbSpecIBLLevels - 1)), 1.0));
|
||||
}
|
||||
#ifdef THE_TO_BAKE_DIFFUSE
|
||||
vec4 aRes = vec4 (bakeDiffuseSH(), 1.0);
|
||||
#ifdef THE_TO_PACK_FLOAT
|
||||
int aCompIndex = int(gl_FragCoord.y);
|
||||
float aComp = aCompIndex == 0 ? aRes.x : (aCompIndex == 1 ? aRes.y : aRes.z);
|
||||
int aFixedPrec = int(aComp * 2147483647.0);
|
||||
int aFixedDiv1 = aFixedPrec / 256;
|
||||
int aFixedDiv2 = aFixedDiv1 / 256;
|
||||
int aFixedDiv3 = aFixedDiv2 / 256;
|
||||
vec4 aPacked = vec4(float(aFixedPrec), float(aFixedDiv1), float(aFixedDiv2), float(aFixedDiv3));
|
||||
aRes = fract (aPacked * (1.0 / 256.0));
|
||||
#endif
|
||||
occFragColor = aRes;
|
||||
#else
|
||||
float aRoughness = float(uCurrentLevel) / float(occNbSpecIBLLevels - 1);
|
||||
occFragColor = vec4 (bakeSpecularMap (aViewDirection, aRoughness), 1.0);
|
||||
#endif
|
||||
}
|
||||
|
@ -4,32 +4,52 @@ uniform int uCurrentSide; //!< current side of cubemap
|
||||
uniform int uYCoeff; //!< coefficient of Y controlling horizontal flip of cubemap
|
||||
uniform int uZCoeff; //!< coefficient of Z controlling vertical flip of cubemap
|
||||
|
||||
const mat2 cubemapDirectionMatrices[6] = mat2[]
|
||||
(
|
||||
mat2 ( 0, -1, -1, 0),
|
||||
mat2 ( 0, 1, -1, 0),
|
||||
mat2 ( 0, 1, 1, 0),
|
||||
mat2 ( 0, 1, -1, 0),
|
||||
mat2 ( 1, 0, 0, -1),
|
||||
mat2 (-1, 0, 0, -1)
|
||||
);
|
||||
|
||||
//! Generates environment map fetching direction considering current index of side.
|
||||
vec3 cubemapBakingViewDirection (in int theSide,
|
||||
in vec2 theScreenCoord)
|
||||
{
|
||||
int anAxis = theSide / 2;
|
||||
vec3 aDirection = vec3(0.0);
|
||||
aDirection[anAxis] = float(-(int(theSide) % 2) * 2 + 1);
|
||||
theScreenCoord = cubemapDirectionMatrices[theSide] * theScreenCoord;
|
||||
aDirection[(anAxis + 1) % 3] = theScreenCoord.x;
|
||||
aDirection[(anAxis + 2) % 3] = theScreenCoord.y;
|
||||
return aDirection;
|
||||
}
|
||||
|
||||
void main()
|
||||
{
|
||||
ViewDirection = cubemapBakingViewDirection (uCurrentSide, occVertex.xy);
|
||||
ViewDirection = cubemapVectorTransform (ViewDirection, uYCoeff, uZCoeff);
|
||||
vec3 aDir;
|
||||
vec2 aCoord;
|
||||
if (uCurrentSide == 0)
|
||||
{
|
||||
aCoord = mat2( 0,-1,-1, 0) * occVertex.xy;
|
||||
aDir.x = 1.0;
|
||||
aDir.y = aCoord.x;
|
||||
aDir.z = aCoord.y;
|
||||
}
|
||||
else if (uCurrentSide == 1)
|
||||
{
|
||||
aCoord = mat2( 0, 1,-1, 0) * occVertex.xy;
|
||||
aDir.x = -1.0;
|
||||
aDir.y = aCoord.x;
|
||||
aDir.z = aCoord.y;
|
||||
}
|
||||
else if (uCurrentSide == 2)
|
||||
{
|
||||
aCoord = mat2( 0, 1, 1, 0) * occVertex.xy;
|
||||
aDir.x = aCoord.y;
|
||||
aDir.y = 1.0;
|
||||
aDir.z = aCoord.x;
|
||||
}
|
||||
else if (uCurrentSide == 3)
|
||||
{
|
||||
aCoord = mat2( 0, 1,-1, 0) * occVertex.xy;
|
||||
aDir.x = aCoord.y;
|
||||
aDir.y = -1.0;
|
||||
aDir.z = aCoord.x;
|
||||
}
|
||||
else if (uCurrentSide == 4)
|
||||
{
|
||||
aCoord = mat2( 1, 0, 0,-1) * occVertex.xy;
|
||||
aDir.x = aCoord.x;
|
||||
aDir.y = aCoord.y;
|
||||
aDir.z = 1.0;
|
||||
}
|
||||
else //if (uCurrentSide == 5)
|
||||
{
|
||||
aCoord = mat2(-1, 0, 0,-1) * occVertex.xy;
|
||||
aDir.x = aCoord.x;
|
||||
aDir.y = aCoord.y;
|
||||
aDir.z = -1.0;
|
||||
}
|
||||
ViewDirection = cubemapVectorTransform (aDir, uYCoeff, uZCoeff);
|
||||
gl_Position = vec4 (occVertex.xy, 0.0, 1.0);
|
||||
}
|
||||
|
@ -146,44 +146,49 @@ static const char Shaders_Declarations_glsl[] =
|
||||
"// Light sources\n"
|
||||
"uniform vec4 occLightAmbient; //!< Cumulative ambient color\n"
|
||||
"#if defined(THE_MAX_LIGHTS) && (THE_MAX_LIGHTS > 0)\n"
|
||||
"#if (THE_MAX_LIGHTS > 1)\n"
|
||||
" #define occLight_Index(theId) theId\n"
|
||||
"#else\n"
|
||||
" #define occLight_Index(theId) 0\n"
|
||||
"#endif\n"
|
||||
"uniform THE_PREC_ENUM int occLightSourcesCount; //!< Total number of light sources\n"
|
||||
"\n"
|
||||
"//! Type of light source, int (see OccLightType enum).\n"
|
||||
"#define occLight_Type(theId) occLightSourcesTypes[theId]\n"
|
||||
"#define occLight_Type(theId) occLightSourcesTypes[occLight_Index(theId)]\n"
|
||||
"\n"
|
||||
"//! Specular intensity (equals to diffuse), vec3.\n"
|
||||
"#define occLight_Specular(theId) occLightSources[theId * 4 + 0].rgb\n"
|
||||
"#define occLight_Specular(theId) occLightSources[occLight_Index(theId) * 4 + 0].rgb\n"
|
||||
"\n"
|
||||
"//! Intensity of light source (>= 0), float.\n"
|
||||
"#define occLight_Intensity(theId) occLightSources[theId * 4 + 0].a\n"
|
||||
"#define occLight_Intensity(theId) occLightSources[occLight_Index(theId) * 4 + 0].a\n"
|
||||
"\n"
|
||||
"//! Is light a headlight, bool? DEPRECATED method.\n"
|
||||
"#define occLight_IsHeadlight(theId) false\n"
|
||||
"\n"
|
||||
"//! Position of specified light source or direction of directional light source, vec3.\n"
|
||||
"#define occLight_Position(theId) occLightSources[theId * 4 + 1].xyz\n"
|
||||
"#define occLight_Position(theId) occLightSources[occLight_Index(theId) * 4 + 1].xyz\n"
|
||||
"\n"
|
||||
"//! Direction of specified spot light source, vec3.\n"
|
||||
"#define occLight_SpotDirection(theId) occLightSources[theId * 4 + 2].xyz\n"
|
||||
"#define occLight_SpotDirection(theId) occLightSources[occLight_Index(theId) * 4 + 2].xyz\n"
|
||||
"\n"
|
||||
"//! Range on which point light source (positional or spot) can affect (>= 0), float.\n"
|
||||
"#define occLight_Range(theId) occLightSources[theId * 4 + 2].w\n"
|
||||
"#define occLight_Range(theId) occLightSources[occLight_Index(theId) * 4 + 2].w\n"
|
||||
"\n"
|
||||
"//! Maximum spread angle of the spot light (in radians), float.\n"
|
||||
"#define occLight_SpotCutOff(theId) occLightSources[theId * 4 + 3].z\n"
|
||||
"#define occLight_SpotCutOff(theId) occLightSources[occLight_Index(theId) * 4 + 3].z\n"
|
||||
"\n"
|
||||
"//! Attenuation of the spot light intensity (from 0 to 1), float.\n"
|
||||
"#define occLight_SpotExponent(theId) occLightSources[theId * 4 + 3].w\n"
|
||||
"#define occLight_SpotExponent(theId) occLightSources[occLight_Index(theId) * 4 + 3].w\n"
|
||||
"\n"
|
||||
"#if !defined(THE_IS_PBR)\n"
|
||||
"//! Diffuse intensity (equals to Specular), vec3.\n"
|
||||
"#define occLight_Diffuse(theId) occLightSources[theId * 4 + 0].rgb\n"
|
||||
"#define occLight_Diffuse(theId) occLightSources[occLight_Index(theId) * 4 + 0].rgb\n"
|
||||
"\n"
|
||||
"//! Const attenuation factor of positional light source, float.\n"
|
||||
"#define occLight_ConstAttenuation(theId) occLightSources[theId * 4 + 3].x\n"
|
||||
"#define occLight_ConstAttenuation(theId) occLightSources[occLight_Index(theId) * 4 + 3].x\n"
|
||||
"\n"
|
||||
"//! Linear attenuation factor of positional light source, float.\n"
|
||||
"#define occLight_LinearAttenuation(theId) occLightSources[theId * 4 + 3].y\n"
|
||||
"#define occLight_LinearAttenuation(theId) occLightSources[occLight_Index(theId) * 4 + 3].y\n"
|
||||
"#endif\n"
|
||||
"#endif\n"
|
||||
"\n"
|
||||
|
@ -17,7 +17,7 @@ static const char Shaders_DirectionalLightShadow_glsl[] =
|
||||
" in int theId,\n"
|
||||
" in vec3 theNormal)\n"
|
||||
"{\n"
|
||||
" vec4 aPosLightSpace = PosLightSpace[theId];\n"
|
||||
" vec4 aPosLightSpace = PosLightSpace[occLight_Index(theId)];\n"
|
||||
" vec3 aLightDir = vec3 (occWorldViewMatrix * vec4 (occLight_Position (theId), 0.0));\n"
|
||||
" vec3 aProjCoords = (aPosLightSpace.xyz / aPosLightSpace.w) * 0.5 + vec3 (0.5);\n"
|
||||
" float aCurrentDepth = aProjCoords.z;\n"
|
||||
|
@ -3,13 +3,23 @@
|
||||
static const char Shaders_PBREnvBaking_fs[] =
|
||||
"THE_SHADER_IN vec3 ViewDirection; //!< direction of fetching from environment cubemap\n"
|
||||
"\n"
|
||||
"#if (__VERSION__ >= 120)\n"
|
||||
"uniform int uSamplesNum; //!< number of samples in Monte-Carlo integration\n"
|
||||
"uniform int uCurrentLevel; //!< current level of specular IBL map (ignored in case of diffuse map's processing)\n"
|
||||
"uniform int uEnvMapSize; //!< one edge's size of source environtment map's zero mipmap level\n"
|
||||
"uniform int uYCoeff; //!< coefficient of Y controlling horizontal flip of cubemap\n"
|
||||
"uniform int uZCoeff; //!< coefficient of Z controlling vertical flip of cubemap\n"
|
||||
"#else\n"
|
||||
"const int uSamplesNum = 256;\n"
|
||||
"#endif\n"
|
||||
"uniform samplerCube uEnvMap; //!< source of baking (environment cubemap)\n"
|
||||
"\n"
|
||||
"#ifdef THE_TO_BAKE_DIFFUSE\n"
|
||||
"uniform int uYCoeff; //!< coefficient of Y controlling horizontal flip of cubemap\n"
|
||||
"uniform int uZCoeff; //!< coefficient of Z controlling vertical flip of cubemap\n"
|
||||
"#endif\n"
|
||||
"\n"
|
||||
"#ifdef THE_TO_BAKE_SPECULAR\n"
|
||||
"uniform int uCurrentLevel; //!< current level of specular IBL map (ignored in case of diffuse map's processing)\n"
|
||||
"uniform float uEnvSolidAngleSource; //!< source solid angle sample computed from one edge's size of source environment map's zero mipmap level\n"
|
||||
"#endif\n"
|
||||
"\n"
|
||||
"//! Returns coordinates of point theNumber from Hammersley point set having size theSize.\n"
|
||||
"vec2 hammersley (in int theNumber,\n"
|
||||
" in int theSize)\n"
|
||||
@ -21,7 +31,7 @@ static const char Shaders_PBREnvBaking_fs[] =
|
||||
" {\n"
|
||||
" if (aNumber > 0)\n"
|
||||
" {\n"
|
||||
" aVanDerCorput += float(aNumber % 2) / float(aDenominator);\n"
|
||||
" aVanDerCorput += mod(float(aNumber), 2.0) / float(aDenominator);\n"
|
||||
" aNumber /= 2;\n"
|
||||
" aDenominator *= 2;\n"
|
||||
" }\n"
|
||||
@ -65,6 +75,8 @@ static const char Shaders_PBREnvBaking_fs[] =
|
||||
" return anX * theVector.x + anY * theVector.y + theNormal * theVector.z;\n"
|
||||
"}\n"
|
||||
"\n"
|
||||
"#ifdef THE_TO_BAKE_DIFFUSE\n"
|
||||
"#if (__VERSION__ >= 120)\n"
|
||||
"const float aSHBasisFuncCoeffs[9] = float[9]\n"
|
||||
"(\n"
|
||||
" 0.282095 * 0.282095,\n"
|
||||
@ -77,7 +89,6 @@ static const char Shaders_PBREnvBaking_fs[] =
|
||||
" 0.315392 * 0.315392,\n"
|
||||
" 0.546274 * 0.546274\n"
|
||||
");\n"
|
||||
"\n"
|
||||
"const float aSHCosCoeffs[9] = float[9]\n"
|
||||
"(\n"
|
||||
" 3.141593,\n"
|
||||
@ -90,76 +101,129 @@ static const char Shaders_PBREnvBaking_fs[] =
|
||||
" 0.785398,\n"
|
||||
" 0.785398\n"
|
||||
");\n"
|
||||
"#else\n"
|
||||
"uniform float aSHBasisFuncCoeffs[9];\n"
|
||||
"uniform float aSHCosCoeffs[9];\n"
|
||||
"#endif\n"
|
||||
"\n"
|
||||
"//! Bakes diffuse IBL map's spherical harmonics coefficients.\n"
|
||||
"vec3 bakeDiffuseSH()\n"
|
||||
"{\n"
|
||||
" int anIndex = int(gl_FragCoord.x);\n"
|
||||
" vec3 aResult = vec3 (0.0);\n"
|
||||
" int anId = int(gl_FragCoord.x);\n"
|
||||
" float aCoef;\n"
|
||||
"#if (__VERSION__ >= 120)\n"
|
||||
" aCoef = aSHCosCoeffs[anId] * aSHBasisFuncCoeffs[anId];\n"
|
||||
"#else\n"
|
||||
" if (anId == 0) { aCoef = aSHCosCoeffs[0] * aSHBasisFuncCoeffs[0]; }\n"
|
||||
" else if (anId == 1) { aCoef = aSHCosCoeffs[1] * aSHBasisFuncCoeffs[1]; }\n"
|
||||
" else if (anId == 2) { aCoef = aSHCosCoeffs[2] * aSHBasisFuncCoeffs[2]; }\n"
|
||||
" else if (anId == 3) { aCoef = aSHCosCoeffs[3] * aSHBasisFuncCoeffs[3]; }\n"
|
||||
" else if (anId == 4) { aCoef = aSHCosCoeffs[4] * aSHBasisFuncCoeffs[4]; }\n"
|
||||
" else if (anId == 5) { aCoef = aSHCosCoeffs[5] * aSHBasisFuncCoeffs[5]; }\n"
|
||||
" else if (anId == 6) { aCoef = aSHCosCoeffs[6] * aSHBasisFuncCoeffs[6]; }\n"
|
||||
" else if (anId == 7) { aCoef = aSHCosCoeffs[7] * aSHBasisFuncCoeffs[7]; }\n"
|
||||
" else { aCoef = aSHCosCoeffs[8] * aSHBasisFuncCoeffs[8]; }\n"
|
||||
"#endif\n"
|
||||
" vec3 aRes = vec3 (0.0);\n"
|
||||
" for (int aSampleIter = 0; aSampleIter < uSamplesNum; ++aSampleIter)\n"
|
||||
" {\n"
|
||||
" vec2 aHammersleyPoint = hammersley (aSampleIter, uSamplesNum);\n"
|
||||
" vec3 aDirection = sphereUniformSample (aHammersleyPoint);\n"
|
||||
" vec3 aDir = sphereUniformSample (aHammersleyPoint);\n"
|
||||
"\n"
|
||||
" vec3 aValue = occTextureCube (uEnvMap, cubemapVectorTransform (aDirection, uYCoeff, uZCoeff)).rgb;\n"
|
||||
" vec3 aVal = occTextureCube (uEnvMap, cubemapVectorTransform (aDir, uYCoeff, uZCoeff)).rgb;\n"
|
||||
" #if (__VERSION__ >= 120)\n"
|
||||
" float aFunc[9];\n"
|
||||
" aFunc[0] = 1.0;\n"
|
||||
"\n"
|
||||
" float aBasisFunc[9];\n"
|
||||
" aBasisFunc[0] = 1.0;\n"
|
||||
" aFunc[1] = aDir.x;\n"
|
||||
" aFunc[2] = aDir.y;\n"
|
||||
" aFunc[3] = aDir.z;\n"
|
||||
"\n"
|
||||
" aBasisFunc[1] = aDirection.x;\n"
|
||||
" aBasisFunc[2] = aDirection.y;\n"
|
||||
" aBasisFunc[3] = aDirection.z;\n"
|
||||
" aFunc[4] = aDir.x * aDir.z;\n"
|
||||
" aFunc[5] = aDir.y * aDir.z;\n"
|
||||
" aFunc[6] = aDir.x * aDir.y;\n"
|
||||
"\n"
|
||||
" aBasisFunc[4] = aDirection.x * aDirection.z;\n"
|
||||
" aBasisFunc[5] = aDirection.y * aDirection.z;\n"
|
||||
" aBasisFunc[6] = aDirection.x * aDirection.y;\n"
|
||||
" aFunc[7] = 3.0 * aDir.z * aDir.z - 1.0;\n"
|
||||
" aFunc[8] = aDir.x * aDir.x - aDir.y * aDir.y;\n"
|
||||
"\n"
|
||||
" aBasisFunc[7] = 3.0 * aDirection.z * aDirection.z - 1.0;\n"
|
||||
" aBasisFunc[8] = aDirection.x * aDirection.x - aDirection.y * aDirection.y;\n"
|
||||
"\n"
|
||||
" aResult += aValue * aBasisFunc[anIndex];\n"
|
||||
" aRes += aVal * aFunc[anId];\n"
|
||||
" #else\n"
|
||||
" if (anId == 0) { aRes += aVal * 1.0; }\n"
|
||||
" else if (anId == 1) { aRes += aVal * aDir.x; }\n"
|
||||
" else if (anId == 2) { aRes += aVal * aDir.y; }\n"
|
||||
" else if (anId == 3) { aRes += aVal * aDir.z; }\n"
|
||||
" else if (anId == 4) { aRes += aVal * (aDir.x * aDir.z); }\n"
|
||||
" else if (anId == 5) { aRes += aVal * (aDir.y * aDir.z); }\n"
|
||||
" else if (anId == 6) { aRes += aVal * (aDir.x * aDir.y); }\n"
|
||||
" else if (anId == 7) { aRes += aVal * (3.0 * aDir.z * aDir.z - 1.0); }\n"
|
||||
" else { aRes += aVal * (aDir.x * aDir.x - aDir.y * aDir.y); }\n"
|
||||
" #endif\n"
|
||||
" }\n"
|
||||
"\n"
|
||||
" aResult *= 4.0 * aSHCosCoeffs[anIndex] * aSHBasisFuncCoeffs[anIndex] / float(uSamplesNum);\n"
|
||||
" return aResult;\n"
|
||||
" return 4.0 * aRes * aCoef / float(uSamplesNum);\n"
|
||||
"}\n"
|
||||
"#endif\n"
|
||||
"\n"
|
||||
"#ifdef THE_TO_BAKE_SPECULAR\n"
|
||||
"//! Computes a single sample for specular IBL map.\n"
|
||||
"vec4 specularMapSample (in vec3 theNormal,\n"
|
||||
" in float theRoughness,\n"
|
||||
" in int theNumber,\n"
|
||||
" in int theSize)\n"
|
||||
"{\n"
|
||||
" vec2 aHammersleyPoint = hammersley (theNumber, theSize);\n"
|
||||
" vec3 aHalf = importanceSample (aHammersleyPoint, occRoughness (theRoughness));\n"
|
||||
" float aHdotV = aHalf.z;\n"
|
||||
" aHalf = fromTangentSpace (aHalf, theNormal);\n"
|
||||
" vec3 aLight = -reflect (theNormal, aHalf);\n"
|
||||
" float aNdotL = dot (aLight, theNormal);\n"
|
||||
" if (aNdotL > 0.0)\n"
|
||||
" {\n"
|
||||
" float aSolidAngleSample = 1.0 / (float(theSize) * (occPBRDistribution (aHdotV, theRoughness) * 0.25 + 0.0001) + 0.0001);\n"
|
||||
" float aLod = (theRoughness == 0.0) ? 0.0 : 0.5 * log2 (aSolidAngleSample / uEnvSolidAngleSource);\n"
|
||||
" return vec4 (occTextureCubeLod (uEnvMap, aLight, aLod).rgb * aNdotL, aNdotL);\n"
|
||||
" }\n"
|
||||
" return vec4 (0.0);\n"
|
||||
"}\n"
|
||||
"\n"
|
||||
"//! Bakes specular IBL map.\n"
|
||||
"vec3 bakeSpecularMap (in vec3 theNormal,\n"
|
||||
" in float theRoughness)\n"
|
||||
"{\n"
|
||||
" vec3 aResult = vec3(0.0);\n"
|
||||
" float aWeightSum = 0.0;\n"
|
||||
" int aSamplesNum = (theRoughness == 0.0) ? 1 : uSamplesNum;\n"
|
||||
" float aSolidAngleSource = 4.0 * PI / (6.0 * float(uEnvMapSize * uEnvMapSize));\n"
|
||||
" for (int aSampleIter = 0; aSampleIter < aSamplesNum; ++aSampleIter)\n"
|
||||
" vec4 aResult = vec4(0.0);\n"
|
||||
" if (theRoughness == 0.0)\n"
|
||||
" {\n"
|
||||
" vec2 aHammersleyPoint = hammersley (aSampleIter, aSamplesNum);\n"
|
||||
" vec3 aHalf = importanceSample (aHammersleyPoint, occRoughness (theRoughness));\n"
|
||||
" float aHdotV = aHalf.z;\n"
|
||||
" aHalf = fromTangentSpace (aHalf, theNormal);\n"
|
||||
" vec3 aLight = -reflect (theNormal, aHalf);\n"
|
||||
" float aNdotL = dot (aLight, theNormal);\n"
|
||||
" if (aNdotL > 0.0)\n"
|
||||
" aResult = specularMapSample (theNormal, theRoughness, 0, 1);\n"
|
||||
" }\n"
|
||||
" else\n"
|
||||
" {\n"
|
||||
" for (int aSampleIter = 0; aSampleIter < uSamplesNum; ++aSampleIter)\n"
|
||||
" {\n"
|
||||
" float aSolidAngleSample = 1.0 / (float(aSamplesNum) * (occPBRDistribution (aHdotV, theRoughness) * 0.25 + 0.0001) + 0.0001);\n"
|
||||
" float aLod = (theRoughness == 0.0) ? 0.0 : 0.5 * log2 (aSolidAngleSample / aSolidAngleSource);\n"
|
||||
" aResult += occTextureCubeLod (uEnvMap, aLight, aLod).rgb * aNdotL;\n"
|
||||
" aWeightSum += aNdotL;\n"
|
||||
" aResult += specularMapSample (theNormal, theRoughness, aSampleIter, uSamplesNum);\n"
|
||||
" }\n"
|
||||
" }\n"
|
||||
" return aResult / aWeightSum;\n"
|
||||
" return aResult.xyz / aResult.w;\n"
|
||||
"}\n"
|
||||
"#endif\n"
|
||||
"\n"
|
||||
"void main()\n"
|
||||
"{\n"
|
||||
" vec3 aViewDirection = normalize (ViewDirection);\n"
|
||||
" if (occNbSpecIBLLevels == 0)\n"
|
||||
" {\n"
|
||||
" occSetFragColor (vec4 (bakeDiffuseSH (), 1.0));\n"
|
||||
" }\n"
|
||||
" else\n"
|
||||
" {\n"
|
||||
" occSetFragColor (vec4 (bakeSpecularMap (aViewDirection, float(uCurrentLevel) / float(occNbSpecIBLLevels - 1)), 1.0));\n"
|
||||
" }\n"
|
||||
"#ifdef THE_TO_BAKE_DIFFUSE\n"
|
||||
" vec4 aRes = vec4 (bakeDiffuseSH(), 1.0);\n"
|
||||
"#ifdef THE_TO_PACK_FLOAT\n"
|
||||
" int aCompIndex = int(gl_FragCoord.y);\n"
|
||||
" float aComp = aCompIndex == 0 ? aRes.x : (aCompIndex == 1 ? aRes.y : aRes.z);\n"
|
||||
" int aFixedPrec = int(aComp * 2147483647.0);\n"
|
||||
" int aFixedDiv1 = aFixedPrec / 256;\n"
|
||||
" int aFixedDiv2 = aFixedDiv1 / 256;\n"
|
||||
" int aFixedDiv3 = aFixedDiv2 / 256;\n"
|
||||
" vec4 aPacked = vec4(float(aFixedPrec), float(aFixedDiv1), float(aFixedDiv2), float(aFixedDiv3));\n"
|
||||
" aRes = fract (aPacked * (1.0 / 256.0));\n"
|
||||
"#endif\n"
|
||||
" occFragColor = aRes;\n"
|
||||
"#else\n"
|
||||
" float aRoughness = float(uCurrentLevel) / float(occNbSpecIBLLevels - 1);\n"
|
||||
" occFragColor = vec4 (bakeSpecularMap (aViewDirection, aRoughness), 1.0);\n"
|
||||
"#endif\n"
|
||||
"}\n";
|
||||
|
@ -7,32 +7,52 @@ static const char Shaders_PBREnvBaking_vs[] =
|
||||
"uniform int uYCoeff; //!< coefficient of Y controlling horizontal flip of cubemap\n"
|
||||
"uniform int uZCoeff; //!< coefficient of Z controlling vertical flip of cubemap\n"
|
||||
"\n"
|
||||
"const mat2 cubemapDirectionMatrices[6] = mat2[]\n"
|
||||
"(\n"
|
||||
" mat2 ( 0, -1, -1, 0),\n"
|
||||
" mat2 ( 0, 1, -1, 0),\n"
|
||||
" mat2 ( 0, 1, 1, 0),\n"
|
||||
" mat2 ( 0, 1, -1, 0),\n"
|
||||
" mat2 ( 1, 0, 0, -1),\n"
|
||||
" mat2 (-1, 0, 0, -1)\n"
|
||||
");\n"
|
||||
"\n"
|
||||
"//! Generates environment map fetching direction considering current index of side.\n"
|
||||
"vec3 cubemapBakingViewDirection (in int theSide,\n"
|
||||
" in vec2 theScreenCoord)\n"
|
||||
"{\n"
|
||||
" int anAxis = theSide / 2;\n"
|
||||
" vec3 aDirection = vec3(0.0);\n"
|
||||
" aDirection[anAxis] = float(-(int(theSide) % 2) * 2 + 1);\n"
|
||||
" theScreenCoord = cubemapDirectionMatrices[theSide] * theScreenCoord;\n"
|
||||
" aDirection[(anAxis + 1) % 3] = theScreenCoord.x;\n"
|
||||
" aDirection[(anAxis + 2) % 3] = theScreenCoord.y;\n"
|
||||
" return aDirection;\n"
|
||||
"}\n"
|
||||
"\n"
|
||||
"void main()\n"
|
||||
"{\n"
|
||||
" ViewDirection = cubemapBakingViewDirection (uCurrentSide, occVertex.xy);\n"
|
||||
" ViewDirection = cubemapVectorTransform (ViewDirection, uYCoeff, uZCoeff);\n"
|
||||
" vec3 aDir;\n"
|
||||
" vec2 aCoord;\n"
|
||||
" if (uCurrentSide == 0)\n"
|
||||
" {\n"
|
||||
" aCoord = mat2( 0,-1,-1, 0) * occVertex.xy;\n"
|
||||
" aDir.x = 1.0;\n"
|
||||
" aDir.y = aCoord.x;\n"
|
||||
" aDir.z = aCoord.y;\n"
|
||||
" }\n"
|
||||
" else if (uCurrentSide == 1)\n"
|
||||
" {\n"
|
||||
" aCoord = mat2( 0, 1,-1, 0) * occVertex.xy;\n"
|
||||
" aDir.x = -1.0;\n"
|
||||
" aDir.y = aCoord.x;\n"
|
||||
" aDir.z = aCoord.y;\n"
|
||||
" }\n"
|
||||
" else if (uCurrentSide == 2)\n"
|
||||
" {\n"
|
||||
" aCoord = mat2( 0, 1, 1, 0) * occVertex.xy;\n"
|
||||
" aDir.x = aCoord.y;\n"
|
||||
" aDir.y = 1.0;\n"
|
||||
" aDir.z = aCoord.x;\n"
|
||||
" }\n"
|
||||
" else if (uCurrentSide == 3)\n"
|
||||
" {\n"
|
||||
" aCoord = mat2( 0, 1,-1, 0) * occVertex.xy;\n"
|
||||
" aDir.x = aCoord.y;\n"
|
||||
" aDir.y = -1.0;\n"
|
||||
" aDir.z = aCoord.x;\n"
|
||||
" }\n"
|
||||
" else if (uCurrentSide == 4)\n"
|
||||
" {\n"
|
||||
" aCoord = mat2( 1, 0, 0,-1) * occVertex.xy;\n"
|
||||
" aDir.x = aCoord.x;\n"
|
||||
" aDir.y = aCoord.y;\n"
|
||||
" aDir.z = 1.0;\n"
|
||||
" }\n"
|
||||
" else //if (uCurrentSide == 5)\n"
|
||||
" {\n"
|
||||
" aCoord = mat2(-1, 0, 0,-1) * occVertex.xy;\n"
|
||||
" aDir.x = aCoord.x;\n"
|
||||
" aDir.y = aCoord.y;\n"
|
||||
" aDir.z = -1.0;\n"
|
||||
" }\n"
|
||||
" ViewDirection = cubemapVectorTransform (aDir, uYCoeff, uZCoeff);\n"
|
||||
" gl_Position = vec4 (occVertex.xy, 0.0, 1.0);\n"
|
||||
"}\n";
|
||||
|
Loading…
x
Reference in New Issue
Block a user