[Unity Shader] 小谈深度法线纹理

在实现一些特定的效果,我们需要获取场景中的深度纹理和法线纹理。虽然,在ShaderLab中,获取它们很简单,但在使用他们之前,应该了解他们根源。

1 概念

法线信息是通过顶点输入阶段而得的,这节我们只讨论深度纹理,因为深度信息比较特殊,需要通过计算得到的。

1.1 深度纹理

深度纹理实际上来自深度缓冲,它包含了一个介于 0.0 和 1.0 之间的深度值,通常用于做深度测试(depth test)。
实际上这个深度值并不是均匀分布的,这个深度值是依据下列公式计算:

depth = \frac{\frac{1}{z}-\frac{1}{near}}{\frac{1}{far}-\frac{1}{near}}

这个 z 值是观察空间(view space)z 分量的相反数(由于观察空间为右手坐标系)。深度值和 \frac{1}{z} 成正比,函数图如下所示:

深度值计算函数

可以看到 z 值在 [1,2] 范围中精度占了50%,现实中我们对近距离的深度精度需求要高于远距离的精度,这样可以缓解深度冲突(Z-fighting) 问题。

那么,问题来了,这个深度值计算公式又是怎么得到的。

image

首先,这个深度值来自于顶点变化后的 NDC 的 z 分量, 由于经过了投影变换 (通常是透视投影),这个深度值则是非线性。还有, NDC z 分量的范围为 [-1,1] 我们需要对其进行处理:

depth = 0.5 * z_{ndc} + 0.5

特别的,在 DirectX 接口中 z 分量的范围为 [0,1]

接着,我们根据视锥矩阵,来得到裁剪 z_{clip} 分量和观察 z_v 之间的关系:

M_{frustum} = \begin{bmatrix} \frac{ \cot{ \frac{FOV}{2} } }{Aspect} & 0 & 0 & 0 \\ 0 & \cot{ \frac{FOV}{2} } & 0 & 0 \\ 0 & 0 & -\frac{Far + Near}{Far - Near} & -\frac{2 * Near * Far}{Far - Near} \\ 0 & 0 & -1 & 0 \end{bmatrix}

摄像机

由此可得:

z_{clip} = - z_{view}\frac{(Far + Near) }{Far - Near} -\frac{2 * Near * Far}{Far - Near}

w_{clip} = -z_{view}

同时,我们可以得到 NDC 和 z_{view} 的关系:

z_{ndc} = \frac{z_{clip}}{w_{clip}} = -\frac{(Far + Near) }{Far - Near} -\frac{2 * Near * Far}{(Far - Near) * z_{view}}

由于我们知道了 NDC 的 z 分量和深度值之间的关系,我们就能推出深度值最开始计算深度值的函数关系(我们需要对 z_{view} 取反,因为他是观察坐标系,总是为负数)

2 深度法线可视化

在Unity中,深度纹理可以直接来自于真正的深度缓存,也可以由一个单独的Pass进行渲染,取决于使用的渲染路径和硬件。在延迟渲染中,深度和纹理信息会直接渲染到G-buffer中,所以可以直接访问。而前向渲染中,是不会创建法线纹理的,因此,Unity底层使用了一个单独的 Pass (在 buildin_shaders-xxx/DefaultResources/Camera-DepthNormalTexture.shader 中)把整个场景都渲染了一遍
我们可以让摄像机生成一张深度纹理(精度为24位或16位,取决于深度缓存的精度)或者深度+法线纹理(共32位,深度、法线纹理各占16位)。

2.1 Unity API

我们需要在脚本中设置摄像机纹理的渲染模式:

// 渲染深度信息 在shader中通过 _CameraDepthTexture访问
camera.depthTextureMode = DepthTextureMode.Depth; 

// 渲染深度和法线信息,通过_CameraDepthNormalsTexture
camera.depthTextureMode = DepthTextureMode.DepthNormals;
// 可以同时产生深度和深度+法线纹理
camera.depthTextureMode |= DepthTextureMode.Depth;
camera.depthTextureMode |= DepthTextureMode.DepthNormals; 

大多情况,我们可以通过 tex2D 函数直接采样,由于某些平台(如 PS3和PS2)上,我们需要一些特殊处理。因此,需要使用宏 SAMPLE_DEPTH_TEXTURE 进行采样:

float d = SAMPLE_DEPTH_TEXTURE(_CameraDepthTexture, i.uv);

还有类似的宏如:SAMPLE_DEPTH_TEXTURE_PROJSAMPLE_DEPTH_TEXTURE_LOD

通过纹理采样得到的深度值是非线性的,即为ndc坐标下的深度值,我们可以通过LinearEyeDepth将采样结果转化为观察空间下的深度值。还有Linear01Depth则会返回一个范围在 [0,1] 的线性深度值。这两个函数使用了内置的_ZBufferParams变量来得到远近裁剪平面的距离。

获取深度+法线纹理,可以直接使用tex2D函数对_CameraDepthNormalsTexture进行采样。Unity提供了辅助函数DecodeDepthNormal来为这个采样结果进行解码,从而获得深度值和法线信息,它在UnityCG.cginc被定义:

inline void DecodeDepthNormal(float4 enc, out float depth, out float3 normal){
    // 范围为[0,1]的线性深度值
    depth = DecodeFloatRG(enc.zw);
    
    // 观察空间下的法线信息
    normal = DecodeViewNormalStereo(enc);
}

当我们开启摄像机渲染深度+法线纹理时,我们可以在 Frame Debugger 观察深度信息和法线信息。

渲染结果
深度信息
发信信息

当然,可以通过编写 Shader 的方式来观察深度和法线信息。

2.2 深度信息可视化
fixed4 frag(v2f i): SV_TARGET{
    float d = SAMPLE_DEPTH_TEXTURE(_CameraDepthTexture, i.uv_depth);
    float depth = 1-d;
    return fixed4(depth,depth,depth,1.0);
}

原图

运行效果

运行效果
2.3 法线信息可视化
fixed4 frag(v2f i): SV_TARGET{
    half4 depthNormal = tex2D(_CameraDepthNormalsTexture, i.uv);
    half3 normal = DecodeViewNormalStereo(depthNormal);   
    return fixed4(normal,1.0);
}
原图

运行效果

运行效果
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,616评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,020评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,078评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,040评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,154评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,265评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,298评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,072评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,491评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,795评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,970评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,654评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,272评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,985评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,815评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,852评论 2 351