游戏人生

不积跬步,无以至千里;不积小流,无以成江海。

My Links

Blog Stats

引擎设计跟踪(九.14.3.3) Deferred shading的一些小细节

1.ambient light

之前的shader里面, 方向光会加上ambient 的计算. 但是如果没有方向光, 就没有ambient. 这是把全局方向光改为点光源之后发现的, 因为透明物体的forward shading是正常的, 其他的全是黑的.

解决方法: 如果没有方向光, 就专门画一个单独quad来计算ambient lighting.

 1     //GBuffer color
 2     float2 colorUV = UV * colorUVSpace.zw + colorUVSpace.xy;
 3     float4 albedo = tex2D(colorBuffer, colorUV);
 4 #if defined(AMBIENT)
 5     return light_ambient*albedo;
 6 #endif
 7     //GBuffer normal
 8     float2 normalUV = UV * normalUVSpace.zw + normalUVSpace.xy;
 9     float4 normal = tex2D(normalBuffer, normalUV);
10     float3 worldNormal = normalize(expand_vector(normal).xyz);
11 #if defined(DIRECTIONAL)
12     float4 diffuse = light_ambient;

 

2.uv flip

这个是因为DX9的纹理坐标系和OGL的相反, 而他们的屏幕坐标系却一样所导致. OGL的屏幕坐标系和UV坐标系都是向右X为正, 向上Y为正轴, 而DX9的屏幕坐标系和OGL一样, 但是UV坐标系相反, 所以render target中的内容是和正常使用贴图相比是Y反转的.

在vertex shader里面flip一下屏幕坐标就可以了.

IGraphicsManager有一个关于图像朝向的标记, 不同的API是不一样的, 这个flip的值通过CPU端计算再传入shader. 来保证API兼容.

1             scalar flip = IGraphicsResourceManager::getSingleton().getGraphicsConfig().TextureDir == IMGO_TOP_DOWN ? -1.0f : 1.0f;
2             mViewSize[3] = flip;

另外, 因为现在所有光源都使用了quad, 所以可以在vertex shader 里面计算uv. 之前point light和spot light使用的volume, 因为要做perspective correction 之后才能在pixel shader里面做perspective divide( /w), 现在因为是平面, 不需要再pixel shader里面计算, vertex shader里面 /w 就可以了.

 

3. half pixel/texel offset

这个也是为了兼容性: 目前只有DX9需要这个trick, DX10+以及OGL都不需要.  现在在IGraphicsResourceManager里面加了一个flag, runtime做判断, 根据API的设置来确定是否要计算, 这样可以兼容所有的API.

1                 scalar fHalfPixelOffsetX = 0;
2                 scalar fHalfPixelOffsetY = 0;
3                 if (IGraphicsResourceManager::getSingleton().getGraphicsConfig().HalfTexel2Pixel)
4                 {
5                     fHalfPixelOffsetX = 0.5f / (scalar)width;
6                     fHalfPixelOffsetY = 0.5f / (scalar)height;
7                 }        

 

posted on 2016-04-03 18:45  crazii  阅读(...)  评论(... 编辑 收藏