65 lines
2.8 KiB
Markdown
65 lines
2.8 KiB
Markdown
---
|
||
title: Lighting阶段的ShadowMap逻辑
|
||
date: 2023-04-10 14:45:45
|
||
excerpt:
|
||
tags:
|
||
rating: ⭐
|
||
---
|
||
|
||
## PS Parameter
|
||
`FDeferredLightPS::FParameters GetDeferredLightPSParameters()`
|
||
|
||
- LightAttenuationTexture = ShadowMaskTexture ? ShadowMaskTexture : WhiteDummy;
|
||
- One pass projection
|
||
- VirtualShadowMap = VirtualShadowMapUniformBuffer;
|
||
- VirtualShadowMapId = VirtualShadowMapId;
|
||
- ShadowMaskBits = ShadowMaskBits ? ShadowMaskBits : GSystemTextures.GetZeroUIntDummy(GraphBuilder);
|
||
|
||
最后输出
|
||
- SceneColorTexture
|
||
- SceneDepthTexture
|
||
|
||
## Lighting阶段中VSM 渲染函数
|
||
RenderDeferredShadowProjections(),输出`ScreenShadowMaskTexture`。
|
||
|
||
## 问题所在
|
||
LightAttenuation
|
||
- x:整个场景的方向光阴影。
|
||
- y:整个场景的方向光SSS阴影。
|
||
- z:Light function + per-object shadows
|
||
- w:per-object SSS shadowing in w
|
||
|
||
zw用于非方向光或者移动端渲染路径。
|
||
```c++
|
||
if (LightData.bRadialLight || SHADING_PATH_MOBILE)
|
||
{
|
||
// Remapping the light attenuation buffer (see ShadowRendering.cpp)
|
||
|
||
Shadow.SurfaceShadow = LightAttenuation.z * StaticShadowing;
|
||
// SSS uses a separate shadowing term that allows light to penetrate the surface
|
||
//@todo - how to do static shadowing of SSS correctly? Shadow.TransmissionShadow = LightAttenuation.w * StaticShadowing;
|
||
|
||
Shadow.TransmissionThickness = LightAttenuation.w;
|
||
}
|
||
else
|
||
{
|
||
// Remapping the light attenuation buffer (see ShadowRendering.cpp)
|
||
// Also fix up the fade between dynamic and static shadows // to work with plane splits rather than spheres.
|
||
float DynamicShadowFraction = DistanceFromCameraFade(SceneDepth, LightData);
|
||
// For a directional light, fade between static shadowing and the whole scene dynamic shadowing based on distance + per object shadows
|
||
Shadow.SurfaceShadow = lerp(LightAttenuation.x, StaticShadowing, DynamicShadowFraction);
|
||
// Fade between SSS dynamic shadowing and static shadowing based on distance
|
||
Shadow.TransmissionShadow = min(lerp(LightAttenuation.y, StaticShadowing, DynamicShadowFraction), LightAttenuation.w);
|
||
|
||
Shadow.SurfaceShadow *= LightAttenuation.z;
|
||
Shadow.TransmissionShadow *= LightAttenuation.z;
|
||
|
||
// Need this min or backscattering will leak when in shadow which cast by non perobject shadow(Only for directional light)
|
||
Shadow.TransmissionThickness = min(LightAttenuation.y, LightAttenuation.w);
|
||
}
|
||
```
|
||
|
||
|
||
**DeferredLightingCommon.ush**与**DeferredLightPixelShaders.usf**中的**LightAttenuation**即为ShadowMap数据。具体可以参考**GetLightAttenuationFromShadow()**,可以看得出EPIC之后打算通过抖动来消除VSM的阴影锯齿。
|
||
|
||
主要的阴影效果由**LightAttenuation**与ShadingModels.ush中的NoL提供。我们只需要调整自阴影也就是z通道,其他的整个场景的阴影。 |