【Unity3D】屏幕深度和法线纹理简介

1 前言

1)深度纹理和法线纹理的含义

​ 深度纹理本质是一张图片,图片中每个像素反应了屏幕中该像素位置对应的顶点 z 值相反数(观察坐标系),之所以用 "反应了" 而不是 "等于"(或 "对应" ),因为深度纹理中颜色的值域是 [0, 1],而顶点 z 值相反数不一定在该区间,另外顶点 z 值相反数与深度纹理不是线性关系(透视投影引起的)。

​ 法线纹理本质也是一张图片,图片中每个像素的 R、G 值对应该点法线向量的 x、y 值(观察空间),z 值通过公式 z = sqrt(x * x + y * y) 计算得到。

2)深度纹理和法线纹理的作用

​ 深度纹理和法线纹理可以用于边缘检测特效、全局雾化特效、激光雷达特效等场景。

2 顶点映射

1)顶点映射过程

​ 在空间和变换中,讲述了顶点是如何从模型空间逐步映射到屏幕上的像素点。这里再简单描述下顶点映射过程,模型中的顶点先后经历了模型变换、观察变换、投影变换、齐次除法(或透视除法)、屏幕映射,逐步映射到屏幕空间。

2)顶点映射的线性阶段和非线性阶段

​ 光栅化发生在投影变换和齐次除法(或透视除法)之间,它是顶点映射的一个重大转折点,主要体现在以下两点:

  • 光栅化对三角形内部进行线性插值,使得光栅化后的顶点数远大于光栅化前的顶点数;
  • 光栅化及之前的变换都是线性变换,光栅化之后进行了齐次除法(或透视除法),使得顶点映射不再保有线性性质(这主要是相机的透视效果引起的)。

3)顶点映射各阶段值域

​ 在空间和变换中,介绍了每次变换后顶点的各个分量的值域,这里再简单描述下:经过投影变换后,顶点 x、y、z 坐标都映射在区间 [-w, w];经过齐次除法后,顶点 x、y、z 坐标都映射在区间 [-1, 1](DirectX 平台上 z 坐标映射在区间 [0, 1]),该空间被称为归一化的设备空间(Normalized Device Coordinates, NDC);经过屏幕映射后,x、y 坐标分别映射在区间 [0, pixelWidth]、 [0, pixelHeight],z 坐标保持不变。

4)NDC 到深度纹理

​ 归一化的设备空间中顶点坐标 z 分量值域是 [-1, 1],而颜色 R、G、B、A 分量值域都是 [0,1],因此需要进行以下映射,其中 z 为 NDC 空间中顶点坐标 z 值,c 为映射的深度纹理的 R 通道值。

3 深度纹理和法线纹理的来源

1)前向渲染生成深度和法线纹理

​ 当使用前向渲染(Forward Rendering)路径时,Unity 会选取所有不透明物体(RenderType 为 Opaque,Queue 为 Background、Geometry 或 AlphaTest,即 Queue <= 2500)生成深度和法线纹理。对于深度纹理,Unity 使用着色器替换技术,在 FallBack 中寻找 LightMode 为 ShadowCast 的 Pass 进行阴影投射(详见阴影原理及应用),同时生成深度纹理;对于法线纹理,Unity 底层会使用一个单独的 Pass 把整个场景再渲染一遍,生成法线纹理(Camera-DepthNormalTexture.shader)。

2)延时渲染生成深度和法线纹理

​ 当使用延时渲染(Deferred Rendering)路径时,Unity 会将深度和法线信息渲染到 G-buffer(Geometric Buffer,几何缓冲区)中。

3)深度&法线纹理编码

​ 用户可以设置只生成深度纹理还是生成深度&法线纹理(深度和法线信息编码在一张纹理中),当设置深度&法线纹理时,Unity 会创建一张和屏幕分辨率相同、精度为 32 位(每个通道 8 位)的纹理,其中观察空间下的法线信息会被编码进 R、G 通道,深度信息会被编码进 B 和 A 通道。

4 深度值和法线向量的获取

4.1 设置深度纹理模式

​ 在 C# 脚本中可以设置深度纹理模式,DepthTextureMode.Depth 模式下会生成一张深度纹理,在 Shader 中可以通过 _CameraDepthTexture 变量获取;DepthTextureMode.DepthNormals 模式下会生成一张深度&法线纹理,在 Shader 中可以通过 _CameraDepthNormalsTexture 变量获取。

cs 复制代码
camera.depthTextureMode = DepthTextureMode.None; // 不渲染深度纹理和法线纹理
camera.depthTextureMode = DepthTextureMode.Depth; // 渲染深度纹理
camera.depthTextureMode = DepthTextureMode.DepthNormals; // 渲染深度&法线纹理
// 渲染两张纹理, 一张深度纹理, 一张深度&法线纹理
camera.depthTextureMode |= DepthTextureMode.Depth;
camera.depthTextureMode |= DepthTextureMode.DepthNormals;

​ 在 Inspector 面板相机组件的最下方可以看到设置的属性,如下:

4.2 从 _CameraDepthTexture 中获取深度

​ 如果生成了深度纹理,深度纹理会保存在内置变量 _CameraDepthTexture 中。

1)深度纹理采样

cpp 复制代码
// 非线性的深度(即计算的深度值与实际深度值不是线性关系)
float depth = SAMPLE_DEPTH_TEXTURE(_CameraDepthTexture, i.uv); // tex2D(_CameraDepthTexture, i.uv).r
// 观察空间中的线性的深度, 值域: [Near, Far], 公式: 1.0 / (_ZBufferParams.z * depth + _ZBufferParams.w)
float linearDepth = LinearEyeDepth(depth);
// 观察空间中的线性且归一化的深度, 值域: [0, 1], 公式: 1.0 / (_ZBufferParams.x * depth + _ZBufferParams.y)
float linear01Depth = Linear01Depth(depth);

​ SAMPLE_DEPTH_TEXTURE 内部使用 tex2D 进行采样,类似的宏还有 SAMPLE_DEPTH_TEXTURE_PROJ、SAMPLE_DEPTH_TEXTURE_LOD,它们在 HLSLSupport.cgin 文件中有定义,如下:

cpp 复制代码
# define SAMPLE_DEPTH_TEXTURE(sampler, uv) (tex2D(sampler, uv).r)
# define SAMPLE_DEPTH_TEXTURE_PROJ(sampler, uv) (tex2Dproj(sampler, uv).r)
# define SAMPLE_DEPTH_TEXTURE_LOD(sampler, uv) (tex2Dlod(sampler, uv).r)

float4 tex2Dproj(sampler2D s, in float3 t) { return tex2D(s, t.xy / t.z); }
float4 tex2Dproj(sampler2D s, in float4 t) { return tex2D(s, t.xy / t.w); }
float4 tex2Dlod(sampler2D x, in float4 t) { return x.t.SampleLevel(x.s, t.xy, t.w); }

​ 说明: SAMPLE_DEPTH_TEXTURE 得到的深度不是线性的深度,即 SAMPLE_DEPTH_TEXTURE 返回的深度值与实践的深度值不是线性关系。

2)LinearEyeDepth 函数源码分析

​ LinearEyeDepth 函数源码(见 UnityCG.cgin 文件)如下,_ZBufferParams.z = (Near - Far) / (Near · Far),_ZBufferParams.w = 1 / Near(_ZBufferParams 为内置变量,详见→Shader常量、变量、结构体、函数,Near、Far 分别为近裁剪平面和远裁剪平面离相机的距离),因此 LinearEyeDepth 内部实现等价于注释部分。

cs 复制代码
// 公式: Near * Far / ((Near - Far) * z + Far), 值域: [Near, Far]
inline float LinearEyeDepth(float z)
{ // 观察空间中的线性的深度, z为纹理采样的非线性的深度
    return 1.0 / (_ZBufferParams.z * z + _ZBufferParams.w);
}

​ 通过第 2 节顶点映射过程,我们可以得出以下方程组关系,其中,z1 为观察空间中顶点坐标 z 值,z2、w2 分别为裁剪空间中顶点坐标 z 值和 w 值,z3 为归一化的设备空间(NDC)中顶点坐标 z 值,z4 为纹理空间中顶点坐标 z 值,depth 为观察空间中顶点的深度值,公式 1 和公式 2 由空间和变换中透视投影得到,公式 3 是齐次除法(或透视除法)(z3 值域为 [-1, 1]),公式 4 是归一化处理(z4 值域为 [0, 1]),公式 5 是将深度值取正(观察空间中顶点坐标都是负值,取反后使得深度值为正)。

​ 进一步计算得到 z4 与 depth 的关系如下:

​ 计算反函数得到 depth 与 z4 的关系如下,结果与代码中注释一致。

3)Linear01Depth 函数源码分析

​ Linear01Depth 函数源码(见 UnityCG.cgin 文件)如下,_ZBufferParams.x = (Near - Far) / Near,_ZBufferParams.y = Far / Near (_ZBufferParams 为内置变量,详见→Shader常量、变量、结构体、函数,Near、Far 分别为近裁剪平面和远裁剪平面离相机的距离),因此 Linear01Depth 内部实现等价于注释部分。

cpp 复制代码
// 公式: Near / ((Near - Far) * z + Far), 值域: [0, 1]
inline float Linear01Depth(float z)
{ // 观察空间中的线性且归一化的深度, z为纹理采样的非线性的深度
    return 1.0 / (_ZBufferParams.x * z + _ZBufferParams.y);
}

​ 本节继续沿用 2)中变量,假设归一化的线性深度为 depth01,depth 的值域为 [Near, Far],depth01 的值域为 [0, 1],因此 depth01 = (depth - Near) / (Far - Near),由于 Near 一般取值较小(Unity 中默认值为 0.3)、Far 取值较大(Unity 中默认值为 1000),depth01 和 depth 的关系可以简化为:depth01 = depth / Far,进一步计算得到 depth 与 z4 的关系如下,结果与代码中注释一致。

​ 说明:Linear01Depth 归一化的结果是一个近似结果,即值域并不是 [0,1],而是 [Near / Far, 1],由于 Near 一般取值较小(Unity 中默认值为 0.3)、Far 取值较大(Unity 中默认值为 1000),Near / Far 近似为 0。

4.3 从 _CameraDepthNormalsTexture 中获取深度和法线

​ 如果生成了深度&法线纹理,深度&法线纹理会保存在 _CameraDepthNormalsTexture 中。

1)深度&法线纹理采样

cs 复制代码
inline void DecodeDepthNormal(float4 enc, out float depth, out float3 normal)
{ // 深度&法线采样, enc为tex2D采样结果, depth、normal为解码后的深度和法线
    depth = DecodeFloatRG(enc.zw); // 观察空间中的线性且归一化的深度
    normal = DecodeViewNormalStereo(enc); // 观察空间中的法线向量
}

2)DecodeFloatRG 函数源码

cpp 复制代码
inline float DecodeFloatRG(float2 enc)
{
    float2 kDecodeDot = float2(1.0, 1 / 255.0);
    return dot(enc, kDecodeDot);
}

3)DecodeViewNormalStereo 函数源码

cpp 复制代码
inline float3 DecodeViewNormalStereo(float4 enc4)
{
    float kScale = 1.7777;
    float3 nn = enc4.xyz * float3(2 * kScale, 2 * kScale, 0) + float3(-kScale, -kScale, 1);
    float g = 2.0 / dot(nn.xyz, nn.xyz);
    float3 n;
    n.xy = g * nn.xy;
    n.z = g - 1;
    return n;
}

5 查看深度纹理和法线纹理

​ 为了不让深度值映射到一个比较小的区域(接近 0 或接近 1),使得深度纹理图呈现黑色或白色,需要调整远裁剪平面的值。

5.1 帧调试器查看深度纹理和法线纹理

1)设置深度纹理模式

​ DepthNormalTest.cs

cs 复制代码
using UnityEngine;

[ExecuteInEditMode] // 编辑态可以查看脚本运行效果
public class DepthNormalTest : MonoBehaviour {
    private void OnEnable() {
        GetComponent<Camera>().depthTextureMode |= DepthTextureMode.Depth;
        GetComponent<Camera>().depthTextureMode |= DepthTextureMode.DepthNormals;
    }
}

​ 说明:DepthNormalTest 脚本组件需要挂在相机上。

​ 场景渲染如下:

2)查看深度纹理

​ 通过 Window → Analysis → Frame Debug 打开帧调试器,单击 Enable 按钮开始调试,如下:

​ 深度纹理如下:

3)查看深度&法线纹理

​ 在帧调试器中调整帧渲染事件,找到最后一个渲染目标为 Camera DepthNormalsTexture 的事件,显示深度&法线纹理如下:

5.2 代码查看线性的深度和法线纹理

1)设置深度纹理模式和材质的 Shader

​ LinearDepthNormalsTexture.cs

cs 复制代码
using UnityEngine;

[ExecuteInEditMode] // 编辑态可以查看脚本运行效果
[RequireComponent(typeof(Camera))] // 需要相机组件
public class LinearDepthNormalsTexture : MonoBehaviour {
    private Material material = null; // 材质

    private void Start() {
        material = new Material(Shader.Find("MyShader/LinearDepthNormalsTexture"));
        material.hideFlags = HideFlags.DontSave;
    }

    private void OnEnable() {
        GetComponent<Camera>().depthTextureMode |= DepthTextureMode.Depth;
        GetComponent<Camera>().depthTextureMode |= DepthTextureMode.DepthNormals;
    }

    private void OnRenderImage(RenderTexture src, RenderTexture dest) {
        if (material != null) {
            Graphics.Blit(null, dest, material);
        } else {
            Graphics.Blit(src, dest);
        }
    }
}

2)基于 _CameraDepthTexture 的深度纹理

​ LinearDepthNormalsTexture.shader

cpp 复制代码
Shader "MyShader/LinearDepthNormalsTexture" { // 线性深度和法线纹理

    SubShader{
        Pass {
            // 深度测试始终通过, 关闭深度写入
            ZTest Always ZWrite Off

            CGPROGRAM

            #include "UnityCG.cginc"

            #pragma vertex vert_img // 使用内置的vert_img顶点着色器
            #pragma fragment frag

            sampler2D _CameraDepthTexture; // 深度纹理

            fixed4 frag(v2f_img i) : SV_Target{ // v2f_img为内置结构体, 里面只包含pos和uv
                float depth = SAMPLE_DEPTH_TEXTURE(_CameraDepthTexture, i.uv); // 非线性的深度(即计算的深度值与实际深度值不是线性关系)
                float linear01Depth = Linear01Depth(depth); // 观察空间中的线性且归一化的深度
                return fixed4(linear01Depth, 0, 0, 1);
            }

            ENDCG
        }
    }

    FallBack off
}

​ 运行后效果如下:

3)基于 _CameraDepthNormalsTexture 的深度纹理

​ LinearDepthNormalsTexture.shader

cpp 复制代码
Shader "MyShader/LinearDepthNormalsTexture" { // 线性深度和法线纹理

    SubShader{
        Pass {
            // 深度测试始终通过, 关闭深度写入
            ZTest Always ZWrite Off

            CGPROGRAM

            #include "UnityCG.cginc"

            #pragma vertex vert_img // 使用内置的vert_img顶点着色器
            #pragma fragment frag

            sampler2D _CameraDepthNormalsTexture; // 深度&法线纹理

            fixed4 frag(v2f_img i) : SV_Target{ // v2f_img为内置结构体, 里面只包含pos和uv
                fixed4 tex = tex2D(_CameraDepthNormalsTexture, i.uv);
                float depth = DecodeFloatRG(tex.zw); // 观察空间中的线性且归一化的深度
                return fixed4(depth, 0, 0, 1);
            }

            ENDCG
        }
    }

    FallBack off
}

​ 运行后效果同第 2)节。

4)基于 _CameraDepthNormalsTexture 的法线纹理

​ LinearDepthNormalsTexture.shader

cpp 复制代码
Shader "MyShader/LinearDepthNormalsTexture" { // 线性深度和法线纹理

    SubShader{
        Pass {
            // 深度测试始终通过, 关闭深度写入
            ZTest Always ZWrite Off

            CGPROGRAM

            #include "UnityCG.cginc"

            #pragma vertex vert_img // 使用内置的vert_img顶点着色器
            #pragma fragment frag

            sampler2D _CameraDepthNormalsTexture; // 深度&法线纹理

            fixed4 frag(v2f_img i) : SV_Target{ // v2f_img为内置结构体, 里面只包含pos和uv
                fixed4 tex = tex2D(_CameraDepthNormalsTexture, i.uv);
                float3 normal = DecodeViewNormalStereo(tex); // 观察空间中的法线向量
                return fixed4(normal * 0.5 + 0.5, 1);
            }

            ENDCG
        }
    }

    FallBack off
}

​ 运行效果如下:

5)基于 _CameraDepthNormalsTexture 的深度&法线纹理

​ LinearDepthNormalsTexture.shader

cpp 复制代码
Shader "MyShader/LinearDepthNormalsTexture" { // 线性深度和法线纹理

    SubShader{
        Pass {
            // 深度测试始终通过, 关闭深度写入
            ZTest Always ZWrite Off

            CGPROGRAM

            #include "UnityCG.cginc"

            #pragma vertex vert_img // 使用内置的vert_img顶点着色器
            #pragma fragment frag

            sampler2D _CameraDepthNormalsTexture; // 深度&法线纹理

            fixed4 frag(v2f_img i) : SV_Target{ // v2f_img为内置结构体, 里面只包含pos和uv
                return tex2D(_CameraDepthNormalsTexture, i.uv);
            }

            ENDCG
        }
    }

    FallBack off
}

​ 运行效果如下:

​ 声明:本文转自【Unity3D】屏幕深度和法线纹理简介

相关推荐
lin zaixi()8 天前
手把手教你写Unity3D飞机大战(2)天空盒布置
unity3d
Thomas_YXQ13 天前
Unity3D中管理Shader效果详解
开发语言·游戏·unity·unity3d·游戏开发
羊羊203514 天前
线性代数:Matrix2x2和Matrix3x3
线性代数·数学建模·unity3d
天人合一peng19 天前
Unity hub登录时一直无法进入license
unity3d
天涯学馆20 天前
Three.js灯光阴影与动画交互
前端·unity3d·three.js
Cool-浩22 天前
Unity3D 开发技巧
开发语言·前端·unity·c#·unity3d·实用技巧·unity开发教程
Cool-浩24 天前
Unity Vision Pro 保姆级开发教程-PolySpatial VisionOS Samples 示例场景
unity·游戏引擎·unity3d·案例·polyspatial·applevision pro·vision pro教程
Thomas_YXQ1 个月前
Unity3D中Excel表格的数据处理模块详解
linux·windows·算法·excel·unity3d·游戏开发
Thomas_YXQ1 个月前
Unity3D ScrollView 滚动视图组件详解及代码实现
开发语言·游戏·unity·架构·unity3d
指尖上的生活1 个月前
Unity使用jslib构建失败
unity3d·webgl