rendering omni shadow in one pass.

 通常而言, 对于不支持TextureArray 的设备(D3D9 是完全不支持的), 渲染omni light shadow 通常的做法是.

shadow generation:

for (int face = 0; face <6; ++face)
{
         push shadow render target(2D Texture)
        setup frustum's projection matrix 
        draw frustum's shadow caster.
}


render shadow :


draw light volume [Sphere]
for (int face = 0; face <6; ++face) { render shadow frustum volume[Cone or frustum ]. }

在D3D11中, 基于TextureArray 可以方便地实现one-pass 渲染. 这就引发了一个问题, 那就是我长久以来不愿意抛弃的D3D9, 或者那些不支持GL_EXT_texture_array的设备能不能做到呢?

2天的尝试后, 发现是可行的. 先抛出结果:

上图中是Hardware PCF 渲染出来的.

 

思路:

1. 对任一需要产生阴影的点光源, 渲染shadow map 的时候, 打包到一张纹理中. 最多是需要6张,因此, 我用了一个3X2的分割.  [这里有一个问题, 点光源不是在所有方向都会产生阴影. 很多时候, 点光源往往只有1-2个Frustum会有shadow caster, 所以这里固定分割,会导致浪费]

2. 设置viewport 到3X2个分割的区域, 分别渲染6个Frustum 的shadow map. 

 

在着色阶段.(下面是基于延时渲染的) 将阴影计算和光照放到一个shader 中, 在渲染灯光包围球的时候, 同时完成. 

几个关键的要素:

  • World -> Shadow 变换. 
  • Projected Texture TexCoord . 
  • Fast Shadow map uv address.

传统做法中, 不需要考虑这些. World->Shadow 变换, 无非就是 mul(WorldToShadowMatrix, WorldPos); 然后 tex2Dproj(ShadowMap, ShadowMapCoord); //tex2Dlod

但是我们这里必须要进行变换. 

 首先最关键的因素是如何将一个pixel 对应的World Position 变换到shadow map 空间? 或者说, 如何确定一个World Position 对应哪个方向上的Frsutum ???? 这里可以武断地确定:

 

float3 WorldPos; // 
float3 LightPos;

float3 LightSpacePos = WorldPos - LightPos;
float3 AbsLSP = abs(LightSpacePos );
float Det = max(AbsLSP .x,max(AbsLSP .y,AbsLSP .z)); 

暂时先不考虑到底是哪个Frsutum. 总之, 在任意Frsutum 中, 距离最大的那个肯定就是在LightView Space 中的距离.  很好, 拥有这个, 我们已经距离结果很近了. 起码, 我们确定了在shadow map 空间中的Z. 

看看XY. 在上面, 我们获得了LightSpacePos.  要快速映射到一个3X2 2D纹理中的纹理坐标, 呵呵, CUBEMAP. 很自然的方式~~~~这里面有一张图最好了, 等我先发完再来补. 

关于构建CUBEMAP 的算法, 其实就是根据渲染阴影时候的视口选择,是一样的. 后面再补. 

 

posted @ 2014-08-28 16:13  访问异常  阅读(354)  评论(0编辑  收藏  举报