From 72773a806c89205cdda9fb2fcf068d4f7894635b Mon Sep 17 00:00:00 2001 From: songchengjiang Date: Wed, 27 Aug 2014 19:12:49 +0800 Subject: [PATCH 1/3] decompose Light Struct --- cocos/renderer/CCGLProgram.cpp | 47 +++++++------------ cocos/renderer/ccShader_3D_ColorNormal.frag | 16 +++---- .../renderer/ccShader_3D_ColorNormalTex.frag | 16 +++---- .../ccShader_3D_PositionNormalTex.vert | 10 ++-- 4 files changed, 37 insertions(+), 52 deletions(-) diff --git a/cocos/renderer/CCGLProgram.cpp b/cocos/renderer/CCGLProgram.cpp index 7a3547a2aa..cb8b0c2d54 100644 --- a/cocos/renderer/CCGLProgram.cpp +++ b/cocos/renderer/CCGLProgram.cpp @@ -445,33 +445,18 @@ bool GLProgram::compileShader(GLuint * shader, GLenum type, const GLchar* source GLchar lightStruct[] = { "#if CC_MAX_DIRECTIONAL_LIGHT_NUM \n" - "struct DirectionalLightSource \n" - "{ \n" - " vec3 color; \n" - " vec3 direction; \n" - "}; \n" - "uniform DirectionalLightSource CC_DirLightSource[CC_MAX_DIRECTIONAL_LIGHT_NUM]; \n" - "uniform int CC_EnabledDirLightNum; \n" + "uniform vec3 CC_DirLightSourceColor[CC_MAX_DIRECTIONAL_LIGHT_NUM]; \n" + "uniform vec3 CC_DirLightSourceDirection[CC_MAX_DIRECTIONAL_LIGHT_NUM]; \n" "#endif \n" "#if CC_MAX_POINT_LIGHT_NUM \n" - "struct PointLightSource \n" - "{ \n" - " vec3 color; \n" - " vec4 position; \n" - "}; \n" - "uniform PointLightSource CC_PointLightSource[CC_MAX_POINT_LIGHT_NUM]; \n" - "uniform int CC_EnabledPointLightNum; \n" + "uniform vec3 CC_PointLightSourceColor[CC_MAX_POINT_LIGHT_NUM]; \n" + "uniform vec4 CC_PointLightSourcePosition[CC_MAX_POINT_LIGHT_NUM]; \n" "#endif \n" "#if CC_MAX_SPOT_LIGHT_NUM \n" - "struct SpotLightSource \n" - "{ \n" - " vec3 color; \n" - " vec3 position; \n" - " vec3 direction; \n" - " vec3 params; \n" - "}; \n" - "uniform SpotLightSource CC_SpotLightSource[CC_MAX_SPOT_LIGHT_NUM]; \n" - "uniform int CC_EnabledSpotLightNum; \n" + "uniform vec3 CC_SpotLightSourceColor[CC_MAX_SPOT_LIGHT_NUM]; \n" + "uniform vec3 CC_SpotLightSourcePosition[CC_MAX_SPOT_LIGHT_NUM]; \n" + "uniform vec3 CC_SpotLightSourceDirection[CC_MAX_SPOT_LIGHT_NUM]; \n" + "uniform vec3 CC_SpotLightSourceParams[CC_MAX_SPOT_LIGHT_NUM]; \n" "#endif \n" "uniform vec4 CC_AmbientColor; \n" }; @@ -1008,9 +993,9 @@ void GLProgram::setUniformsForBuiltins(const Mat4 &matrixMV) CCASSERT(enabledDirLightNum < CC_MAX_DIRECTIONAL_LIGHT_NUM, ""); Vec3 dir = light->getWorldDirection(); dir.normalize(); - sprintf(str, "CC_DirLightSource[%d].%s", enabledDirLightNum, "color"); + sprintf(str, "CC_DirLightSourceColor[%d]", enabledDirLightNum); setUniformLocationWith3f(glGetUniformLocation(_program, str), col.r / 255.0f, col.g / 255.0f, col.b / 255.0f); - sprintf(str, "CC_DirLightSource[%d].%s", enabledDirLightNum, "direction"); + sprintf(str, "CC_DirLightSourceDirection[%d]", enabledDirLightNum); setUniformLocationWith3f(glGetUniformLocation(_program, str), dir.x, dir.y, dir.z); ++enabledDirLightNum; } @@ -1019,9 +1004,9 @@ void GLProgram::setUniformsForBuiltins(const Mat4 &matrixMV) { CCASSERT(enabledPointLightNum < CC_MAX_POINT_LIGHT_NUM, ""); Mat4 mat= light->getNodeToWorldTransform(); - sprintf(str, "CC_PointLightSource[%d].%s", enabledPointLightNum, "color"); + sprintf(str, "CC_PointLightSourceColor[%d]", enabledPointLightNum); setUniformLocationWith3f(glGetUniformLocation(_program, str), col.r / 255.0f, col.g / 255.0f, col.b / 255.0f); - sprintf(str, "CC_PointLightSource[%d].%s", enabledPointLightNum, "position"); + sprintf(str, "CC_PointLightSourcePosition[%d]", enabledPointLightNum); setUniformLocationWith4f(glGetUniformLocation(_program, str), mat.m[12], mat.m[13], mat.m[14], 1.0f / light->getRange()); ++enabledPointLightNum; } @@ -1031,13 +1016,13 @@ void GLProgram::setUniformsForBuiltins(const Mat4 &matrixMV) Vec3 dir = light->getWorldDirection(); dir.normalize(); Mat4 mat= light->getNodeToWorldTransform(); - sprintf(str, "CC_SpotLightSource[%d].%s", enabledSpotLightNum, "color"); + sprintf(str, "CC_SpotLightSourceColor[%d]", enabledSpotLightNum); setUniformLocationWith3f(glGetUniformLocation(_program, str), col.r / 255.0f, col.g / 255.0f, col.b / 255.0f); - sprintf(str, "CC_SpotLightSource[%d].%s", enabledSpotLightNum, "position"); + sprintf(str, "CC_SpotLightSourcePosition[%d]", enabledSpotLightNum); setUniformLocationWith3f(glGetUniformLocation(_program, str), mat.m[12], mat.m[13], mat.m[14]); - sprintf(str, "CC_SpotLightSource[%d].%s", enabledSpotLightNum, "direction"); + sprintf(str, "CC_SpotLightSourceDirection[%d]", enabledSpotLightNum); setUniformLocationWith3f(glGetUniformLocation(_program, str), dir.x, dir.y, dir.z); - sprintf(str, "CC_SpotLightSource[%d].%s", enabledSpotLightNum, "params"); + sprintf(str, "CC_SpotLightSourceParams[%d]", enabledSpotLightNum); setUniformLocationWith3f(glGetUniformLocation(_program, str), cosf(light->getInnerAngle()), cosf(light->getOuterAngle()), 1.0f / light->getRange()); ++enabledSpotLightNum; } diff --git a/cocos/renderer/ccShader_3D_ColorNormal.frag b/cocos/renderer/ccShader_3D_ColorNormal.frag index efc14d8dd9..6ad1d0ebf8 100644 --- a/cocos/renderer/ccShader_3D_ColorNormal.frag +++ b/cocos/renderer/ccShader_3D_ColorNormal.frag @@ -31,9 +31,9 @@ uniform vec4 u_color; void PointLight(int n, vec3 normal, inout vec4 intensity) { vec3 dir = v_vertexToPointLightDirection[n]; - vec3 ldir = dir * CC_PointLightSource[n].position.w; + vec3 ldir = dir * CC_PointLightSourcePosition[n].w; float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - intensity.xyz += CC_PointLightSource[n].color * max(0.0, dot(normalize(dir), normal)) * attenuation; + intensity.xyz += CC_PointLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * attenuation; intensity.w = 1.0; } \n#endif\n @@ -41,7 +41,7 @@ void PointLight(int n, vec3 normal, inout vec4 intensity) \n#if CC_MAX_DIRECTIONAL_LIGHT_NUM\n void DirectionalLight(int n, vec3 normal, inout vec4 intensity) { - intensity.xyz += CC_DirLightSource[n].color * max(0.0, dot(-CC_DirLightSource[n].direction, normal)); + intensity.xyz += CC_DirLightSourceColor[n] * max(0.0, dot(-CC_DirLightSourceDirection[n], normal)); intensity.w = 1.0; } \n#endif\n @@ -50,13 +50,13 @@ void DirectionalLight(int n, vec3 normal, inout vec4 intensity) void SpotLight(int n, vec3 normal, inout vec4 intensity) { vec3 dir = v_vertexToSpotLightDirection[n]; - vec3 ldir = dir * CC_SpotLightSource[n].params.z; + vec3 ldir = dir * CC_SpotLightSourceParams[n].z; float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - float spotDot = dot(normalize(dir), -CC_SpotLightSource[n].direction); - float innerCos = CC_SpotLightSource[n].params.x; - float outerCos = CC_SpotLightSource[n].params.y; + float spotDot = dot(normalize(dir), -CC_SpotLightSourceDirection[n]); + float innerCos = CC_SpotLightSourceParams[n].x; + float outerCos = CC_SpotLightSourceParams[n].y; float factor = smoothstep(outerCos, innerCos, spotDot); - intensity.xyz += CC_SpotLightSource[n].color * max(0.0, dot(normalize(dir), normal)) * factor * attenuation; + intensity.xyz += CC_SpotLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * factor * attenuation; intensity.w = 1.0; } \n#endif\n diff --git a/cocos/renderer/ccShader_3D_ColorNormalTex.frag b/cocos/renderer/ccShader_3D_ColorNormalTex.frag index d36e7f4fbd..2d74c3aff7 100644 --- a/cocos/renderer/ccShader_3D_ColorNormalTex.frag +++ b/cocos/renderer/ccShader_3D_ColorNormalTex.frag @@ -31,9 +31,9 @@ uniform vec4 u_color; void PointLight(int n, vec3 normal, inout vec4 intensity) { vec3 dir = v_vertexToPointLightDirection[n]; - vec3 ldir = dir * CC_PointLightSource[n].position.w; + vec3 ldir = dir * CC_PointLightSourcePosition[n].w; float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - intensity.xyz += CC_PointLightSource[n].color * max(0.0, dot(normalize(dir), normal)) * attenuation; + intensity.xyz += CC_PointLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * attenuation; intensity.w = 1.0; } \n#endif\n @@ -41,7 +41,7 @@ void PointLight(int n, vec3 normal, inout vec4 intensity) \n#if CC_MAX_DIRECTIONAL_LIGHT_NUM\n void DirectionalLight(int n, vec3 normal, inout vec4 intensity) { - intensity.xyz += CC_DirLightSource[n].color * max(0.0, dot(-CC_DirLightSource[n].direction, normal)); + intensity.xyz += CC_DirLightSourceColor[n] * max(0.0, dot(-CC_DirLightSourceDirection[n], normal)); intensity.w = 1.0; } \n#endif\n @@ -50,13 +50,13 @@ void DirectionalLight(int n, vec3 normal, inout vec4 intensity) void SpotLight(int n, vec3 normal, inout vec4 intensity) { vec3 dir = v_vertexToSpotLightDirection[n]; - vec3 ldir = dir * CC_SpotLightSource[n].params.z; + vec3 ldir = dir * CC_SpotLightSourceParams[n].z; float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - float spotDot = dot(normalize(dir), -CC_SpotLightSource[n].direction); - float innerCos = CC_SpotLightSource[n].params.x; - float outerCos = CC_SpotLightSource[n].params.y; + float spotDot = dot(normalize(dir), -CC_SpotLightSourceDirection[n]); + float innerCos = CC_SpotLightSourceParams[n].x; + float outerCos = CC_SpotLightSourceParams[n].y; float factor = smoothstep(outerCos, innerCos, spotDot); - intensity.xyz += CC_SpotLightSource[n].color * max(0.0, dot(normalize(dir), normal)) * factor * attenuation; + intensity.xyz += CC_SpotLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * factor * attenuation; intensity.w = 1.0; } \n#endif\n diff --git a/cocos/renderer/ccShader_3D_PositionNormalTex.vert b/cocos/renderer/ccShader_3D_PositionNormalTex.vert index 4b1f489d22..6d19baaa82 100644 --- a/cocos/renderer/ccShader_3D_PositionNormalTex.vert +++ b/cocos/renderer/ccShader_3D_PositionNormalTex.vert @@ -20,14 +20,14 @@ void main(void) \n#if (CC_MAX_POINT_LIGHT_NUM > 0)\n for (int i = 0; i < CC_MAX_POINT_LIGHT_NUM; ++i) { - v_vertexToPointLightDirection[i] = CC_PointLightSource[i].position.xyz - ePosition.xyz; + v_vertexToPointLightDirection[i] = CC_PointLightSourcePosition[i].xyz - ePosition.xyz; } \n#endif\n \n#if (CC_MAX_SPOT_LIGHT_NUM > 0)\n for (int i = 0; i < CC_MAX_SPOT_LIGHT_NUM; ++i) { - v_vertexToSpotLightDirection[i] = CC_SpotLightSource[i].position.xyz - ePosition.xyz; + v_vertexToSpotLightDirection[i] = CC_SpotLightSourcePosition[i] - ePosition.xyz; } \n#endif\n @@ -113,7 +113,7 @@ void getPositionAndNormal(out vec4 position, out vec3 normal) position.w = p.w; \n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM || CC_MAX_POINT_LIGHT_NUM || CC_MAX_SPOT_LIGHT_NUM)\n - vec3 n = vec4(a_normal, 0.0); + vec4 n = vec4(a_normal, 0.0); normal.x = dot(n, matrixPalette1); normal.y = dot(n, matrixPalette2); normal.z = dot(n, matrixPalette3); @@ -130,14 +130,14 @@ void main() \n#if (CC_MAX_POINT_LIGHT_NUM > 0)\n for (int i = 0; i < CC_MAX_POINT_LIGHT_NUM; ++i) { - v_vertexToPointLightDirection[i] = CC_PointLightSource[i].position.xyz- ePosition.xyz; + v_vertexToPointLightDirection[i] = CC_PointLightSourcePosition[i].xyz- ePosition.xyz; } \n#endif\n \n#if (CC_MAX_SPOT_LIGHT_NUM > 0)\n for (int i = 0; i < CC_MAX_SPOT_LIGHT_NUM; ++i) { - v_vertexToSpotLightDirection[i] = CC_SpotLightSource[i].position.xyz - ePosition.xyz; + v_vertexToSpotLightDirection[i] = CC_SpotLightSourcePosition[i] - ePosition.xyz; } \n#endif\n From 26dc7de8027357e229b852edf67e58d18cdc9623 Mon Sep 17 00:00:00 2001 From: songchengjiang Date: Wed, 27 Aug 2014 21:03:52 +0800 Subject: [PATCH 2/3] Repaired Light Shader BUG On ios --- cocos/renderer/CCGLProgram.cpp | 8 +-- cocos/renderer/ccShader_3D_ColorNormal.frag | 65 ++++++------------- .../renderer/ccShader_3D_ColorNormalTex.frag | 65 ++++++------------- tests/cpp-tests/Classes/controller.cpp | 1 - 4 files changed, 46 insertions(+), 93 deletions(-) diff --git a/cocos/renderer/CCGLProgram.cpp b/cocos/renderer/CCGLProgram.cpp index cb8b0c2d54..8db0c5904a 100644 --- a/cocos/renderer/CCGLProgram.cpp +++ b/cocos/renderer/CCGLProgram.cpp @@ -449,8 +449,8 @@ bool GLProgram::compileShader(GLuint * shader, GLenum type, const GLchar* source "uniform vec3 CC_DirLightSourceDirection[CC_MAX_DIRECTIONAL_LIGHT_NUM]; \n" "#endif \n" "#if CC_MAX_POINT_LIGHT_NUM \n" - "uniform vec3 CC_PointLightSourceColor[CC_MAX_POINT_LIGHT_NUM]; \n" - "uniform vec4 CC_PointLightSourcePosition[CC_MAX_POINT_LIGHT_NUM]; \n" + "uniform vec4 CC_PointLightSourceColor[CC_MAX_POINT_LIGHT_NUM]; \n" + "uniform vec3 CC_PointLightSourcePosition[CC_MAX_POINT_LIGHT_NUM]; \n" "#endif \n" "#if CC_MAX_SPOT_LIGHT_NUM \n" "uniform vec3 CC_SpotLightSourceColor[CC_MAX_SPOT_LIGHT_NUM]; \n" @@ -1005,9 +1005,9 @@ void GLProgram::setUniformsForBuiltins(const Mat4 &matrixMV) CCASSERT(enabledPointLightNum < CC_MAX_POINT_LIGHT_NUM, ""); Mat4 mat= light->getNodeToWorldTransform(); sprintf(str, "CC_PointLightSourceColor[%d]", enabledPointLightNum); - setUniformLocationWith3f(glGetUniformLocation(_program, str), col.r / 255.0f, col.g / 255.0f, col.b / 255.0f); + setUniformLocationWith4f(glGetUniformLocation(_program, str), col.r / 255.0f, col.g / 255.0f, col.b / 255.0f, 1.0f / light->getRange()); sprintf(str, "CC_PointLightSourcePosition[%d]", enabledPointLightNum); - setUniformLocationWith4f(glGetUniformLocation(_program, str), mat.m[12], mat.m[13], mat.m[14], 1.0f / light->getRange()); + setUniformLocationWith3f(glGetUniformLocation(_program, str), mat.m[12], mat.m[13], mat.m[14]); ++enabledPointLightNum; } else diff --git a/cocos/renderer/ccShader_3D_ColorNormal.frag b/cocos/renderer/ccShader_3D_ColorNormal.frag index 6ad1d0ebf8..55749f6d3c 100644 --- a/cocos/renderer/ccShader_3D_ColorNormal.frag +++ b/cocos/renderer/ccShader_3D_ColorNormal.frag @@ -27,70 +27,47 @@ varying vec3 v_normal; uniform vec4 u_color; -\n#if CC_MAX_POINT_LIGHT_NUM\n -void PointLight(int n, vec3 normal, inout vec4 intensity) -{ - vec3 dir = v_vertexToPointLightDirection[n]; - vec3 ldir = dir * CC_PointLightSourcePosition[n].w; - float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - intensity.xyz += CC_PointLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * attenuation; - intensity.w = 1.0; -} -\n#endif\n - -\n#if CC_MAX_DIRECTIONAL_LIGHT_NUM\n -void DirectionalLight(int n, vec3 normal, inout vec4 intensity) -{ - intensity.xyz += CC_DirLightSourceColor[n] * max(0.0, dot(-CC_DirLightSourceDirection[n], normal)); - intensity.w = 1.0; -} -\n#endif\n - -\n#if CC_MAX_SPOT_LIGHT_NUM\n -void SpotLight(int n, vec3 normal, inout vec4 intensity) -{ - vec3 dir = v_vertexToSpotLightDirection[n]; - vec3 ldir = dir * CC_SpotLightSourceParams[n].z; - float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - float spotDot = dot(normalize(dir), -CC_SpotLightSourceDirection[n]); - float innerCos = CC_SpotLightSourceParams[n].x; - float outerCos = CC_SpotLightSourceParams[n].y; - float factor = smoothstep(outerCos, innerCos, spotDot); - intensity.xyz += CC_SpotLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * factor * attenuation; - intensity.w = 1.0; -} -\n#endif\n - void main(void) { - vec4 intensity = vec4(0.0); + vec4 intensity = vec4(0.0, 0.0, 0.0, 1.0); \n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM || CC_MAX_POINT_LIGHT_NUM || CC_MAX_SPOT_LIGHT_NUM)\n vec3 normal = normalize(v_normal); \n#endif\n -\n#if CC_MAX_DIRECTIONAL_LIGHT_NUM\n +\n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM > 0)\n - for (int i = 0; i < CC_MAX_DIRECTIONAL_LIGHT_NUM; ++i) + for (int n = 0; n < CC_MAX_DIRECTIONAL_LIGHT_NUM; ++n) { - DirectionalLight(i, normal, intensity); + intensity.xyz += CC_DirLightSourceColor[n] * max(0.0, dot(-CC_DirLightSourceDirection[n], normal)); } \n#endif\n -\n#if CC_MAX_POINT_LIGHT_NUM\n +\n#if (CC_MAX_POINT_LIGHT_NUM > 0)\n - for (int i = 0; i < CC_MAX_POINT_LIGHT_NUM; ++i) + for (int n = 0; n < CC_MAX_POINT_LIGHT_NUM; ++n) { - PointLight(i, normal, intensity); + vec3 dir = v_vertexToPointLightDirection[n]; + vec3 ldir = dir * CC_PointLightSourceColor[n].w; + float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); + intensity.xyz += CC_PointLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * attenuation; } \n#endif\n -\n#if CC_MAX_SPOT_LIGHT_NUM\n +\n#if (CC_MAX_SPOT_LIGHT_NUM > 0)\n - for (int i = 0; i < CC_MAX_SPOT_LIGHT_NUM; ++i) + for (int n = 0; n < CC_MAX_SPOT_LIGHT_NUM; ++n) { - SpotLight(i, normal, intensity); + vec3 dir = v_vertexToSpotLightDirection[n]; + vec3 ldir = dir * CC_SpotLightSourceParams[n].z; + float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); + dir = normalize(dir); + float spotDot = dot(normalize(dir), -CC_SpotLightSourceDirection[n]); + float innerCos = CC_SpotLightSourceParams[n].x; + float outerCos = CC_SpotLightSourceParams[n].y; + float factor = smoothstep(outerCos, innerCos, spotDot); + intensity.xyz += CC_SpotLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * factor * attenuation; } \n#endif\n diff --git a/cocos/renderer/ccShader_3D_ColorNormalTex.frag b/cocos/renderer/ccShader_3D_ColorNormalTex.frag index 2d74c3aff7..3c4fabcf97 100644 --- a/cocos/renderer/ccShader_3D_ColorNormalTex.frag +++ b/cocos/renderer/ccShader_3D_ColorNormalTex.frag @@ -27,70 +27,47 @@ varying vec3 v_normal; uniform vec4 u_color; -\n#if CC_MAX_POINT_LIGHT_NUM\n -void PointLight(int n, vec3 normal, inout vec4 intensity) -{ - vec3 dir = v_vertexToPointLightDirection[n]; - vec3 ldir = dir * CC_PointLightSourcePosition[n].w; - float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - intensity.xyz += CC_PointLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * attenuation; - intensity.w = 1.0; -} -\n#endif\n - -\n#if CC_MAX_DIRECTIONAL_LIGHT_NUM\n -void DirectionalLight(int n, vec3 normal, inout vec4 intensity) -{ - intensity.xyz += CC_DirLightSourceColor[n] * max(0.0, dot(-CC_DirLightSourceDirection[n], normal)); - intensity.w = 1.0; -} -\n#endif\n - -\n#if CC_MAX_SPOT_LIGHT_NUM\n -void SpotLight(int n, vec3 normal, inout vec4 intensity) -{ - vec3 dir = v_vertexToSpotLightDirection[n]; - vec3 ldir = dir * CC_SpotLightSourceParams[n].z; - float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - float spotDot = dot(normalize(dir), -CC_SpotLightSourceDirection[n]); - float innerCos = CC_SpotLightSourceParams[n].x; - float outerCos = CC_SpotLightSourceParams[n].y; - float factor = smoothstep(outerCos, innerCos, spotDot); - intensity.xyz += CC_SpotLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * factor * attenuation; - intensity.w = 1.0; -} -\n#endif\n - void main(void) { - vec4 intensity = vec4(0.0); + vec4 intensity = vec4(0.0, 0.0, 0.0, 1.0); \n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM || CC_MAX_POINT_LIGHT_NUM || CC_MAX_SPOT_LIGHT_NUM)\n vec3 normal = normalize(v_normal); \n#endif\n -\n#if CC_MAX_DIRECTIONAL_LIGHT_NUM\n +\n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM > 0)\n - for (int i = 0; i < CC_MAX_DIRECTIONAL_LIGHT_NUM; ++i) + for (int n = 0; n < CC_MAX_DIRECTIONAL_LIGHT_NUM; ++n) { - DirectionalLight(i, normal, intensity); + intensity.xyz += CC_DirLightSourceColor[n] * max(0.0, dot(-CC_DirLightSourceDirection[n], normal)); } \n#endif\n -\n#if CC_MAX_POINT_LIGHT_NUM\n +\n#if (CC_MAX_POINT_LIGHT_NUM > 0)\n - for (int i = 0; i < CC_MAX_POINT_LIGHT_NUM; ++i) + for (int n = 0; n < CC_MAX_POINT_LIGHT_NUM; ++n) { - PointLight(i, normal, intensity); + vec3 dir = v_vertexToPointLightDirection[n]; + vec3 ldir = dir * CC_PointLightSourceColor[n].w; + float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); + intensity.xyz += CC_PointLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * attenuation; } \n#endif\n -\n#if CC_MAX_SPOT_LIGHT_NUM\n +\n#if (CC_MAX_SPOT_LIGHT_NUM > 0)\n - for (int i = 0; i < CC_MAX_SPOT_LIGHT_NUM; ++i) + for (int n = 0; n < CC_MAX_SPOT_LIGHT_NUM; ++n) { - SpotLight(i, normal, intensity); + vec3 dir = v_vertexToSpotLightDirection[n]; + vec3 ldir = dir * CC_SpotLightSourceParams[n].z; + float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); + dir = normalize(dir); + float spotDot = dot(normalize(dir), -CC_SpotLightSourceDirection[n]); + float innerCos = CC_SpotLightSourceParams[n].x; + float outerCos = CC_SpotLightSourceParams[n].y; + float factor = smoothstep(outerCos, innerCos, spotDot); + intensity.xyz += CC_SpotLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * factor * attenuation; } \n#endif\n diff --git a/tests/cpp-tests/Classes/controller.cpp b/tests/cpp-tests/Classes/controller.cpp index 830f78b943..89585f73f2 100644 --- a/tests/cpp-tests/Classes/controller.cpp +++ b/tests/cpp-tests/Classes/controller.cpp @@ -78,7 +78,6 @@ Controller g_aTestNames[] = { { "Node: Spine", []() { return new SpineTestScene(); } }, { "Node: Sprite", [](){return new SpriteTestScene(); } }, { "Node: Sprite3D", [](){ return new Sprite3DTestScene(); }}, - { "Node: LightTest", [](){ return new LightTestScene(); }}, { "Node: TileMap", [](){return new TileMapTestScene(); } }, #if CC_TARGET_PLATFORM != CC_PLATFORM_WP8 { "Node: FastTileMap", [](){return new TileMapTestSceneNew(); } }, From 365c57bc2bcf7613d3cdd56043edef65d1990b9e Mon Sep 17 00:00:00 2001 From: songchengjiang Date: Wed, 27 Aug 2014 23:40:49 +0800 Subject: [PATCH 3/3] replaced Light Shader by optimized Light Shader --- cocos/renderer/CCGLProgram.cpp | 25 +++++--- cocos/renderer/ccShader_3D_ColorNormal.frag | 60 ++++++++++-------- .../renderer/ccShader_3D_ColorNormalTex.frag | 63 +++++++++++-------- .../ccShader_3D_PositionNormalTex.vert | 6 +- 4 files changed, 92 insertions(+), 62 deletions(-) diff --git a/cocos/renderer/CCGLProgram.cpp b/cocos/renderer/CCGLProgram.cpp index 8db0c5904a..6941241989 100644 --- a/cocos/renderer/CCGLProgram.cpp +++ b/cocos/renderer/CCGLProgram.cpp @@ -449,14 +449,17 @@ bool GLProgram::compileShader(GLuint * shader, GLenum type, const GLchar* source "uniform vec3 CC_DirLightSourceDirection[CC_MAX_DIRECTIONAL_LIGHT_NUM]; \n" "#endif \n" "#if CC_MAX_POINT_LIGHT_NUM \n" - "uniform vec4 CC_PointLightSourceColor[CC_MAX_POINT_LIGHT_NUM]; \n" + "uniform vec3 CC_PointLightSourceColor[CC_MAX_POINT_LIGHT_NUM]; \n" "uniform vec3 CC_PointLightSourcePosition[CC_MAX_POINT_LIGHT_NUM]; \n" + "uniform float CC_PointLightSourceRangeInverse[CC_MAX_POINT_LIGHT_NUM]; \n" "#endif \n" "#if CC_MAX_SPOT_LIGHT_NUM \n" "uniform vec3 CC_SpotLightSourceColor[CC_MAX_SPOT_LIGHT_NUM]; \n" "uniform vec3 CC_SpotLightSourcePosition[CC_MAX_SPOT_LIGHT_NUM]; \n" "uniform vec3 CC_SpotLightSourceDirection[CC_MAX_SPOT_LIGHT_NUM]; \n" - "uniform vec3 CC_SpotLightSourceParams[CC_MAX_SPOT_LIGHT_NUM]; \n" + "uniform float CC_SpotLightSourceInnerAngleCos[CC_MAX_SPOT_LIGHT_NUM]; \n" + "uniform float CC_SpotLightSourceOuterAngleCos[CC_MAX_SPOT_LIGHT_NUM]; \n" + "uniform float CC_SpotLightSourceRangeInverse[CC_MAX_SPOT_LIGHT_NUM]; \n" "#endif \n" "uniform vec4 CC_AmbientColor; \n" }; @@ -556,9 +559,9 @@ void GLProgram::updateUniforms() // _builtInUniforms[UNIFORM_ENABLED_SPOT_LIGHT_NUM] != -1 // ); - _flags.usesLights = CC_MAX_DIRECTIONAL_LIGHT_NUM || - CC_MAX_POINT_LIGHT_NUM || - CC_MAX_SPOT_LIGHT_NUM; + _flags.usesLights = (0 < CC_MAX_DIRECTIONAL_LIGHT_NUM) || + (0 < CC_MAX_POINT_LIGHT_NUM) || + (0 < CC_MAX_SPOT_LIGHT_NUM); _flags.usesP = _builtInUniforms[UNIFORM_P_MATRIX] != -1; _flags.usesMV = _builtInUniforms[UNIFORM_MV_MATRIX] != -1; _flags.usesMVP = _builtInUniforms[UNIFORM_MVP_MATRIX] != -1; @@ -1005,9 +1008,11 @@ void GLProgram::setUniformsForBuiltins(const Mat4 &matrixMV) CCASSERT(enabledPointLightNum < CC_MAX_POINT_LIGHT_NUM, ""); Mat4 mat= light->getNodeToWorldTransform(); sprintf(str, "CC_PointLightSourceColor[%d]", enabledPointLightNum); - setUniformLocationWith4f(glGetUniformLocation(_program, str), col.r / 255.0f, col.g / 255.0f, col.b / 255.0f, 1.0f / light->getRange()); + setUniformLocationWith3f(glGetUniformLocation(_program, str), col.r / 255.0f, col.g / 255.0f, col.b / 255.0f); sprintf(str, "CC_PointLightSourcePosition[%d]", enabledPointLightNum); setUniformLocationWith3f(glGetUniformLocation(_program, str), mat.m[12], mat.m[13], mat.m[14]); + sprintf(str, "CC_PointLightSourceRangeInverse[%d]", enabledPointLightNum); + setUniformLocationWith1f(glGetUniformLocation(_program, str), 1.0f / light->getRange()); ++enabledPointLightNum; } else @@ -1022,8 +1027,12 @@ void GLProgram::setUniformsForBuiltins(const Mat4 &matrixMV) setUniformLocationWith3f(glGetUniformLocation(_program, str), mat.m[12], mat.m[13], mat.m[14]); sprintf(str, "CC_SpotLightSourceDirection[%d]", enabledSpotLightNum); setUniformLocationWith3f(glGetUniformLocation(_program, str), dir.x, dir.y, dir.z); - sprintf(str, "CC_SpotLightSourceParams[%d]", enabledSpotLightNum); - setUniformLocationWith3f(glGetUniformLocation(_program, str), cosf(light->getInnerAngle()), cosf(light->getOuterAngle()), 1.0f / light->getRange()); + sprintf(str, "CC_SpotLightSourceInnerAngleCos[%d]", enabledSpotLightNum); + setUniformLocationWith1f(glGetUniformLocation(_program, str), cosf(light->getInnerAngle())); + sprintf(str, "CC_SpotLightSourceOuterAngleCos[%d]", enabledSpotLightNum); + setUniformLocationWith1f(glGetUniformLocation(_program, str), cosf(light->getOuterAngle())); + sprintf(str, "CC_SpotLightSourceRangeInverse[%d]", enabledSpotLightNum); + setUniformLocationWith1f(glGetUniformLocation(_program, str), 1.0f / light->getRange()); ++enabledSpotLightNum; } } diff --git a/cocos/renderer/ccShader_3D_ColorNormal.frag b/cocos/renderer/ccShader_3D_ColorNormal.frag index 55749f6d3c..bec8b94dfe 100644 --- a/cocos/renderer/ccShader_3D_ColorNormal.frag +++ b/cocos/renderer/ccShader_3D_ColorNormal.frag @@ -27,53 +27,63 @@ varying vec3 v_normal; uniform vec4 u_color; +vec3 computeLighting(vec3 normalVector, vec3 lightDirection, vec3 lightColor, float attenuation) +{ + float diffuse = max(dot(normalVector, lightDirection), 0.0); + vec3 diffuseColor = lightColor * diffuse * attenuation; + + return diffuseColor; +} + void main(void) { - vec4 intensity = vec4(0.0, 0.0, 0.0, 1.0); -\n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM || CC_MAX_POINT_LIGHT_NUM || CC_MAX_SPOT_LIGHT_NUM)\n +\n#if ((CC_MAX_DIRECTIONAL_LIGHT_NUM > 0) || (CC_MAX_POINT_LIGHT_NUM > 0) || (CC_MAX_SPOT_LIGHT_NUM > 0))\n vec3 normal = normalize(v_normal); \n#endif\n + vec4 combinedColor = CC_AmbientColor; + + // Directional light contribution \n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM > 0)\n - - for (int n = 0; n < CC_MAX_DIRECTIONAL_LIGHT_NUM; ++n) + for (int i = 0; i < CC_MAX_DIRECTIONAL_LIGHT_NUM; ++i) { - intensity.xyz += CC_DirLightSourceColor[n] * max(0.0, dot(-CC_DirLightSourceDirection[n], normal)); + vec3 lightDirection = normalize(CC_DirLightSourceDirection[i] * 2.0); + combinedColor.xyz += computeLighting(normal, -lightDirection, CC_DirLightSourceColor[i], 1.0); } - \n#endif\n + // Point light contribution \n#if (CC_MAX_POINT_LIGHT_NUM > 0)\n - - for (int n = 0; n < CC_MAX_POINT_LIGHT_NUM; ++n) + for (int i = 0; i < CC_MAX_POINT_LIGHT_NUM; ++i) { - vec3 dir = v_vertexToPointLightDirection[n]; - vec3 ldir = dir * CC_PointLightSourceColor[n].w; + vec3 ldir = v_vertexToPointLightDirection[i] * CC_PointLightSourceRangeInverse[i]; float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - intensity.xyz += CC_PointLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * attenuation; + combinedColor.xyz += computeLighting(normal, normalize(v_vertexToPointLightDirection[i]), CC_PointLightSourceColor[i], attenuation); } - \n#endif\n + // Spot light contribution \n#if (CC_MAX_SPOT_LIGHT_NUM > 0)\n - - for (int n = 0; n < CC_MAX_SPOT_LIGHT_NUM; ++n) + for (int i = 0; i < CC_MAX_SPOT_LIGHT_NUM; ++i) { - vec3 dir = v_vertexToSpotLightDirection[n]; - vec3 ldir = dir * CC_SpotLightSourceParams[n].z; + // Compute range attenuation + vec3 ldir = v_vertexToSpotLightDirection[i] * CC_SpotLightSourceRangeInverse[i]; float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - dir = normalize(dir); - float spotDot = dot(normalize(dir), -CC_SpotLightSourceDirection[n]); - float innerCos = CC_SpotLightSourceParams[n].x; - float outerCos = CC_SpotLightSourceParams[n].y; - float factor = smoothstep(outerCos, innerCos, spotDot); - intensity.xyz += CC_SpotLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * factor * attenuation; - } + vec3 vertexToSpotLightDirection = normalize(v_vertexToSpotLightDirection[i]); + vec3 spotLightDirection = normalize(CC_SpotLightSourceDirection[i] * 2.0); + + // "-lightDirection" is used because light direction points in opposite direction to spot direction. + float spotCurrentAngleCos = dot(spotLightDirection, -vertexToSpotLightDirection); + + // Apply spot attenuation + attenuation *= smoothstep(CC_SpotLightSourceOuterAngleCos[i], CC_SpotLightSourceInnerAngleCos[i], spotCurrentAngleCos); + combinedColor.xyz += computeLighting(normal, vertexToSpotLightDirection, CC_SpotLightSourceColor[i], attenuation); + } \n#endif\n -\n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM || CC_MAX_POINT_LIGHT_NUM || CC_MAX_SPOT_LIGHT_NUM)\n - gl_FragColor = u_color * (CC_AmbientColor + intensity); +\n#if ((CC_MAX_DIRECTIONAL_LIGHT_NUM > 0) || (CC_MAX_POINT_LIGHT_NUM > 0) || (CC_MAX_SPOT_LIGHT_NUM > 0))\n + gl_FragColor = u_color * combinedColor; \n#else\n gl_FragColor = u_color; \n#endif\n diff --git a/cocos/renderer/ccShader_3D_ColorNormalTex.frag b/cocos/renderer/ccShader_3D_ColorNormalTex.frag index 3c4fabcf97..a614905d13 100644 --- a/cocos/renderer/ccShader_3D_ColorNormalTex.frag +++ b/cocos/renderer/ccShader_3D_ColorNormalTex.frag @@ -27,55 +27,66 @@ varying vec3 v_normal; uniform vec4 u_color; +vec3 computeLighting(vec3 normalVector, vec3 lightDirection, vec3 lightColor, float attenuation) +{ + float diffuse = max(dot(normalVector, lightDirection), 0.0); + vec3 diffuseColor = lightColor * diffuse * attenuation; + + return diffuseColor; +} + void main(void) { - vec4 intensity = vec4(0.0, 0.0, 0.0, 1.0); -\n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM || CC_MAX_POINT_LIGHT_NUM || CC_MAX_SPOT_LIGHT_NUM)\n +\n#if ((CC_MAX_DIRECTIONAL_LIGHT_NUM > 0) || (CC_MAX_POINT_LIGHT_NUM > 0) || (CC_MAX_SPOT_LIGHT_NUM > 0))\n vec3 normal = normalize(v_normal); \n#endif\n + vec4 combinedColor = CC_AmbientColor; + + // Directional light contribution \n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM > 0)\n - - for (int n = 0; n < CC_MAX_DIRECTIONAL_LIGHT_NUM; ++n) + for (int i = 0; i < CC_MAX_DIRECTIONAL_LIGHT_NUM; ++i) { - intensity.xyz += CC_DirLightSourceColor[n] * max(0.0, dot(-CC_DirLightSourceDirection[n], normal)); + vec3 lightDirection = normalize(CC_DirLightSourceDirection[i] * 2.0); + combinedColor.xyz += computeLighting(normal, -lightDirection, CC_DirLightSourceColor[i], 1.0); } - \n#endif\n + // Point light contribution \n#if (CC_MAX_POINT_LIGHT_NUM > 0)\n - - for (int n = 0; n < CC_MAX_POINT_LIGHT_NUM; ++n) + for (int i = 0; i < CC_MAX_POINT_LIGHT_NUM; ++i) { - vec3 dir = v_vertexToPointLightDirection[n]; - vec3 ldir = dir * CC_PointLightSourceColor[n].w; + vec3 ldir = v_vertexToPointLightDirection[i] * CC_PointLightSourceRangeInverse[i]; float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - intensity.xyz += CC_PointLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * attenuation; + combinedColor.xyz += computeLighting(normal, normalize(v_vertexToPointLightDirection[i]), CC_PointLightSourceColor[i], attenuation); } - \n#endif\n + // Spot light contribution \n#if (CC_MAX_SPOT_LIGHT_NUM > 0)\n - - for (int n = 0; n < CC_MAX_SPOT_LIGHT_NUM; ++n) + for (int i = 0; i < CC_MAX_SPOT_LIGHT_NUM; ++i) { - vec3 dir = v_vertexToSpotLightDirection[n]; - vec3 ldir = dir * CC_SpotLightSourceParams[n].z; + // Compute range attenuation + vec3 ldir = v_vertexToSpotLightDirection[i] * CC_SpotLightSourceRangeInverse[i]; float attenuation = clamp(1.0 - dot(ldir, ldir), 0.0, 1.0); - dir = normalize(dir); - float spotDot = dot(normalize(dir), -CC_SpotLightSourceDirection[n]); - float innerCos = CC_SpotLightSourceParams[n].x; - float outerCos = CC_SpotLightSourceParams[n].y; - float factor = smoothstep(outerCos, innerCos, spotDot); - intensity.xyz += CC_SpotLightSourceColor[n] * max(0.0, dot(normalize(dir), normal)) * factor * attenuation; - } + vec3 vertexToSpotLightDirection = normalize(v_vertexToSpotLightDirection[i]); + vec3 spotLightDirection = normalize(CC_SpotLightSourceDirection[i] * 2.0); + + // "-lightDirection" is used because light direction points in opposite direction to spot direction. + float spotCurrentAngleCos = dot(spotLightDirection, -vertexToSpotLightDirection); + + // Apply spot attenuation + attenuation *= smoothstep(CC_SpotLightSourceOuterAngleCos[i], CC_SpotLightSourceInnerAngleCos[i], spotCurrentAngleCos); + combinedColor.xyz += computeLighting(normal, vertexToSpotLightDirection, CC_SpotLightSourceColor[i], attenuation); + } \n#endif\n -\n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM || CC_MAX_POINT_LIGHT_NUM || CC_MAX_SPOT_LIGHT_NUM)\n - gl_FragColor = texture2D(CC_Texture0, TextureCoordOut) * u_color * (CC_AmbientColor + intensity); + +\n#if ((CC_MAX_DIRECTIONAL_LIGHT_NUM > 0) || (CC_MAX_POINT_LIGHT_NUM > 0) || (CC_MAX_SPOT_LIGHT_NUM > 0))\n + gl_FragColor = texture2D(CC_Texture0, TextureCoordOut) * u_color * combinedColor; \n#else\n - gl_FragColor = texture2D(CC_Texture0, TextureCoordOut) * u_color; + gl_FragColor = texture2D(CC_Texture0, TextureCoordOut) * u_color; \n#endif\n } diff --git a/cocos/renderer/ccShader_3D_PositionNormalTex.vert b/cocos/renderer/ccShader_3D_PositionNormalTex.vert index 6d19baaa82..80bce316ea 100644 --- a/cocos/renderer/ccShader_3D_PositionNormalTex.vert +++ b/cocos/renderer/ccShader_3D_PositionNormalTex.vert @@ -31,7 +31,7 @@ void main(void) } \n#endif\n -\n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM || CC_MAX_POINT_LIGHT_NUM || CC_MAX_SPOT_LIGHT_NUM)\n +\n#if ((CC_MAX_DIRECTIONAL_LIGHT_NUM > 0) || (CC_MAX_POINT_LIGHT_NUM > 0) || (CC_MAX_SPOT_LIGHT_NUM > 0))\n v_normal = CC_NormalMatrix * a_normal; \n#endif\n @@ -112,7 +112,7 @@ void getPositionAndNormal(out vec4 position, out vec3 normal) position.z = dot(p, matrixPalette3); position.w = p.w; -\n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM || CC_MAX_POINT_LIGHT_NUM || CC_MAX_SPOT_LIGHT_NUM)\n +\n#if ((CC_MAX_DIRECTIONAL_LIGHT_NUM > 0) || (CC_MAX_POINT_LIGHT_NUM > 0) || (CC_MAX_SPOT_LIGHT_NUM > 0))\n vec4 n = vec4(a_normal, 0.0); normal.x = dot(n, matrixPalette1); normal.y = dot(n, matrixPalette2); @@ -141,7 +141,7 @@ void main() } \n#endif\n -\n#if (CC_MAX_DIRECTIONAL_LIGHT_NUM || CC_MAX_POINT_LIGHT_NUM || CC_MAX_SPOT_LIGHT_NUM)\n +\n#if ((CC_MAX_DIRECTIONAL_LIGHT_NUM > 0) || (CC_MAX_POINT_LIGHT_NUM > 0) || (CC_MAX_SPOT_LIGHT_NUM > 0))\n v_normal = CC_NormalMatrix * normal; \n#endif\n