内容概要
以反射率方程为线索,理解 PBR 技术集合中的各个技术点,进而理解整个 PBR 的原理,以及虚幻4引擎中的实现方案。上半部分,主要是理解其原理。
【欢迎转载,请注明作者:房燕良,原文出处:游戏程序员的自我修养】
PBR 从离线渲染进入游戏引擎
基于物理的渲染,即 PBR,Physically Based Rendering,这个技术名词最早是有马特·法尔(Matt Pharr)在他的知名著作中提出,这本书就是《Physically Based Rendering:From Theory To Implementation》,此书首次出版是在2004年。在2014年,此书的三位作者马特·法尔(Matt Pharr)、格雷戈·汉弗莱斯(Greg Humphreys)和帕特·汉拉汉(Pat Hanrahan)获得了第19届奥斯卡金像奖科技成果奖(Scientific and Technical Academy Award)1。 目前此书的第三版已经发布了在线版本:http://www.pbr-book.org 。这本书中提出的技术方案最开始是应用于离线渲染领域,典型的就是迪斯尼的在《无敌破坏王》(Wreck-It Ralph, 2012年)金属工作流(Metallic Workflow)。
在2006年的 SIGGRAPH 大会上纳蒂·霍夫曼(Naty Hoffman)等人进行了一场名为:SIGGRAPH 2006 Course: Physically-Based Reflectance for Games 的专题演讲,当时他还在顽皮狗(Naughty Dog)工作。纳蒂·霍夫曼对于 PBR 进入实时渲染领域十分热心,在 SIGGRAPH 2010 大会上,他再次组织一个关于 PBR 的专场:SIGGRAPH 2010 Course: Physically-Based Shading Models in Film and Game Production。这次的演讲引起了业界强烈的反响,PBR 成为热门话题!在后面的几年中,来自Ubisoft、迪斯尼、皮克斯、Epic Games、EA、Unity 等公司的牛人们轮番登场,在 SIGGRAPH 大会介绍他们对在电影、动画片、游戏中应用 PBR 技术的进展。对游戏行业影响最大的应该算是布瑞恩·卡里斯(Brian Karis)在 SIGGRAPH 2013 所做的演讲:Real Shading in Unreal Engine 4。虚幻4(Unreal Engine 4)并不是游戏业界第一个使用 PBR 技术的引擎,但是凭借虚幻引擎的影响力,以及后来免费、开源的大力推广,在行业内产生了无可替代的影响力。另外,2014年 EA 的 DICE 工作室所做的关于寒霜引擎(Frostbite Engine)的演讲:Moving Frostbite to PBR 也十分精彩!
PBR 对于现代 3D 引擎十分重要,它使得实时渲染突破了被常年诟病的“塑料感”,我们有了金属、皮革、瓷器!以从业者的角度看,更重要的是:它使我们可以把材质与光照解耦。也就是说一个 PBR 材质,在不同的光照环境下都应该得到正确的渲染结果,这让 3D 美术工作者可以更方便的调节对象的材质,并使得材质更具备通用性。
那么,它是怎么做到的呢?下面我就带你领略这一突破是如何完成的。我们首先要从物理、数学的层面来理解其原理,然后我们再来学习虚幻4中的实现方案。
理解 PBR 的原理
我们必须先明确一个事情:PBR 并不是“一项”技术,它是由一系列技术的集合,并不断改进的结果。从原理到实现方案,整体来看是很复杂的,但是如果你先把每一项技术搞明白,再看清楚它们是怎么串起来的,就容易理解多了。把所有珍珠串起来的这根金线就是渲染方程(The Rendering Equation)。下面我就以渲染方程为线索,讲述 PBR 背后的物理和数学原理,其中的具体方案来自虚幻4。
渲染方程(The Rendering Equation)
James T. Kajiya 在 SIGGRAPH 1986 中发表的论文《The Rendering Equation》中提出了这个概念。文章首先高屋建瓴的从辐射度量学的角度,用一个方程描述光在场景中的转播,并给出了下面这个方程,并且提出了使用蒙特卡洛方法求它的数值解的思路,并命名为 Path Tracing! 另外,文章标题中的冠词用的是 “The” ,而不是 “A”,实际是确认也是如此,后面的发展方向基本也是以这个方程为出发点了。
下面我们就看一下渲染方程的具体内容:
\[L_o(p, v) = \int\limits_H L_i(p,l) f(l, v) n \cdot l dl\]这个公式中的符号含义:
- $L$:辐射率(Radiance),这是一个物理概念,下面我用一小节专门解释
- $p$:物体表面上的一个点
- $v$:观察方向
- $l$:入射光方向
- $f$:BRDF,非常重要的概念,后面单独一小节解释
- $\int\limits_H … dl$:在半球域(Hemisphere)内,对所有入射光线进行积分
- $n$:这个最简单了,就是$p$点所在平面的法向量啦
需要说明的是虚幻4文档中列出的这个方程已经经过了概念上的简化。原始的反射率方程是在半球域上对立体角(Solid Angle)进行积分:$\int\limits_{\Omega} … d\omega_i$,立体角是一个十分重要的概念,后面我用一小节专门解释。在上面这个公式中,把无限小的立体角近似认为就是一个入射光线$l$。
这个公式挺复杂,我们先概括的理解一下它。它的含义是这样的: $p$ 点在 $v$ 方向上的辐射率,或者说从 $v$ 方向观察 $p$ 点的辐射率,等于在以法线$n$为轴所环绕的半球域内所有入射光的辐射率,乘以入射角度衰减($n \cdot l $),并受到 BRDF 的约束,即乘以$f(l, v)$。
想要更好的理解上面这个反射率方程,我们需要对上面涉及到的几个基础概念进行进一步的解释。
能量守恒
渲染方程的物理基础是能量守恒定律。简单来说就是,光接触物体表面后,会发生三种现象:反射、折射、吸收。排除自发光的情况,反射光 + 折射光 ≤ 入射光。上述方程也要满足能量守恒定律。
辐射率(Radiance)
辐射率是所谓的辐射度量学中的一个概念,它定义的是单位立体角上的辐射密度。在这里我们不讲太多的物理光学的东西,网上很多资料都讲了很多,我这篇科普是以数学为核心。我们就粗略的理解成为物体表面一点向指定方向的光能辐射强度就可以了,在渲染的过程中,我们将会使用 RGB 分量来对它进行表示。
BRDF
BRDF 是 Bidirectional Reflective Distribution Function 的缩写,中文译为双向反射分布函数。这个概念由 Fred E. Nicodemus 于1965年提出。
这个函数计算给定方向上的反射光的亮度与入射光在物体表面产生的照度的比率,它的计算结果是一个$[0, 1]$之间的值。这样说有点抽象,举一个最简单的例子吧。对于一个完美的镜面反射来说,我们可以按照几何光学来计算任何一个入射光$\omega_i$的反射方向$\omega_o$,那么 BRDF 应该只在这个反射方向$\omega_o$上返回 1.0,其他任何方向上都应该得到 0.0 的结果。
立体角(Solid Angle)
参考对于一个平面空间内的角,弧度的定义:弧长等于半径的弧,其所对的圆心角为1弧度,即:弧度=弧长/半径。 立体角的概念可以类比,立体角=它对应的弧面面积/半径的平方;对于单位球来说,半径为1,则立体角也可看做是单位球面上的一个面积,它的值为:
\[d\omega=\sin\theta\;d\theta\;d\phi\;\]一个无限小的立体角可以理解成一条射线。
Cook-Torrance 反射模型
在虚幻4中 $f(l, v)$ 的计算,也就是 BRDF 使用 Cook-Torrance 反射模型。这个模型是由当时就职于卢卡斯影业(Lucasfilm Ltd.)的 Robert L. Cook(现为皮克斯高级技术副总裁)和康奈尔大学(Cornell University)的 Kenneth E. Torrance 合作提出的,论文名为《A Reflectance Model for Computer Graphics》2。这是一个较 Blinn-Phong 更完善的模型,更重要的是 Blinn-Phong 带入反射率方程的话不是能量守恒的。它本质上仍然是一个经验模型,并不像 Radiosity 那样本质是物理正确的(只是无法求解,只能用 Progressive 的方式求近似解)。
这个反射模型包含漫反射和镜面反射量部分:
\[R_{bd} = k_d R_d + k_s R_s\]其中$k_d+k_s=1.0$,$R_d$和$R_s$和物体的材质有关。对于漫反射的计算 $R_d$ ,在虚幻4引擎中,布瑞恩·卡里斯进行了很多尝试,最终认为 Lambertian 漫反射模型已经足够好了。
这里需要注意的是,漫反射是指光进入物体内部,经由物体内部的粒子反射之后,呈现随机方向离开物体的现象。对于金属来说所有的折射光都会被吸收,所以完美的金属漫反射 $R_d$ 为0,只有镜面反射 $R_s$。
\[R_d = \frac{c_{diff}}{\pi}\]其中 $c_{diff}$ 就是物体的固有色。
虚幻4中计算镜面反射$R_s$的公式如下:
\[R_s = \frac{D(h)F(l,h)G(l,v,h)}{4 (n \cdot l)(n \cdot v)}\]其中D、G、F分别是三个函数:
- D:微平面在平面上的分布函数
- G:计算微平面由于互相遮挡而产生的衰减
- F:菲涅尔项
这三个函数是重点内容。D、G两项都用到了一个参数:粗糙度,想要说明白这个参数,还需要先讲一下微表面理论。
微平面模型(The Microfacet Model)
微平面理论认为,从微观角度看,物体表面是由一些细小的平面组成的。微观到什么程度呢,大概可以理解为可见光波长大小的尺度。这些细小的平面,就叫做微平面(Microfacets)。越光滑的表面,这些微平面排列的越规则,越粗糙的表面,这些微平面排列的越不规则。在图形学中,我们可以从统计学的角度使用粗糙度(Roughness)这样一个估算的参数来描述这一现象。
镜面反射$R_s$中的$D$、$F$、$G$
D:微平面分布函数(Normal distribution function)
我们先来看一下$D$项,即微表面分布函数。假设表面的宏观法向量为 $n$ ,这个函数计算得出所有微平面中法向量与 $n$ 一致的微平面的比例。例如,如果有 20% 的微表面的法向量与 $n$ 一致,则这个函数返回 0.2 。(看了一些其他的资料,都是讲的 $h$ 向量,我觉得用 $n$ 来理解更直观。)虚幻4采用 Trowbridge-Reitz GGX 模型:
\[D(h) = \frac{\alpha^2}{\pi((n \cdot h)^2 (\alpha^2 - 1) + 1)^2}\]其中:
- $n$ 为表面的宏观法向量
- $h$ 入射光和观察方向的中间向量
- $\alpha$ 为表面的粗糙度参数
G:几何遮挡因子(Geometric Occlusion Factor)
微平面理论认为,由于微平面的不规则排列,反射光有一定比率会被物体自身的微平面遮挡而衰减,这就是 G 项,虚幻4使用 Schlick 模型结合 Smith 模型计算此项,具体公式为:
\[k = \frac{(\alpha + 1)^2}{8}\] \[G_1(v) = \frac{n \cdot v}{(n \cdot v)(1 - k) + k }\] \[G(l, v, h) = G_1(1) G_1(v)\]F:菲涅尔项(Fresnel Factor)
金属以外的物体表面都会具有菲涅尔效果,虚幻4中计算菲涅尔项的公式为:
\[F(v, h) = F_0 +(1-F_0)2^{(-5.55473(v \cdot h)-6.98316)(v \cdot h)}\]其中 $F_0$ 为基础反射率,是一个常数,可以通过光学测定。
总结
至此,我们已经讲了反射率方程,以及在虚幻4中是如何计算这个方程中的各个部分的。那么虚幻4中是如何在实时渲染中求解这个复杂的积分的呢?我将在下一篇文章中分享我对虚幻4引擎的 PBR 实现的理解。
本文中的公式来自Cook-Torrance的论文,以及布瑞恩·卡里斯(Brian Karis)在 SIGGRAPH 2013 所做的演讲:Real Shading in Unreal Engine 4