目录

最近在做一个数字人渲染DEMO,本博客将用于记录搜集到的 Skin Rendering 技术,持续更新中。

现实皮肤模型

在渲染皮肤时主要关注以下光照路径:

  • 镜面反射:6%入射皮肤的光被皮肤最上层油脂菲涅尔反射

    • 实现例子:Dual Specular BRDF

  • 次表面散射(Subsurface Scattering,SSS):94% 的光进入皮肤次表层,在围绕进入点的 3D 邻域中返回和离开表面。现实的皮肤介质是多层的,而较物理的渲染一般建模成至少两个不同的层。散射路径还可进一步细分为单次散射和多次散射的路径:

    • 单散射:在介质内部只散射了一次的光照贡献,与光线角度相关性大
    • 多散射:在介质内部散射了2次或更多次的光照贡献(也是diffuse现象的主要来源,而皮肤主要贡献便是多散射)
    • 实现例子:BSSRDF

  • 透射 [可选]:对于较薄的物体(如耳朵边缘、鼻子边缘),光射入的平面和射出的平面很可能是不一样的。实际上如果次表面散射的路径实现的足够物理,自然而然也会包含透射现象,因此该光路是可选的

    • 实现例子:BTDF with Thickness Map & Shadow Thickness

BSSRDF 渲染模型 [2001]

光线进入 shading point 周围的像素后经过在散射介质内部的随机游走(Random Walk),可能会从 shading point 射出。但 SSS 的光线在散射介质中的游走行为难以模拟,为了达成实时渲染的性能要求,便有了以下思路:

  • 提供一个类似于 BRDF 的函数(也就是 BSSRDF),但输入参数从4D(入射方向、出射方向)变成了8D(入射点位置、入射方向、出射点位置、出射方向)
  • 太远的像素对 shading point 的散射光贡献极小,因此可以忽略它们,只对一定范围内的表面 A 进行积分

图左是 BRDF 的行为,图右是 BSSRDF 的行为

于是便有了基于 BSSRDF 的渲染方程,相当于在 BRDF 渲染方程的基础上增加了对表面的积分:

\(L_{o}\left(p_{o}, \omega_{o}\right)=\int_{A} \int_{2\pi} S\left(p_{o}, \omega_{o}, p_{i}, \omega_{i}\right) L_{i}\left(p_{i}, \omega_{i}\right)\left|\cos \theta_{i}\right| d \omega_{i} d A\)

然后又基于下列假设:

  • 次表面散射的物体是一个曲率为零的平面
  • 这个平面的厚度,大小都是无限
  • 平面内部的介质参数是均匀的
  • 光线永远是从垂直的方向入射表面

得出 BSSRDF 函数的形式:

\(S\left(p_{o}, \omega_{o}, p_{i}, \omega_{i}\right)=\frac{1}{\pi} F_{t}\left(\eta_{o}, \omega_{o}\right) R_{d}\left(\left\|p_{i}-p_{o}\right\|\right) F_{t}\left(\eta_{i}, \omega_{i}\right)\)

Diffusion Profile(扩散剖面)[2001]

Diffusion Profile 描述了光线如何在半透明物体中进行扩散和分布,其函数形式如下:

\(R_{d}\left(\left\|p_{i}-p_{o}\right\|\right)\)

其实就是一个记录了 BSSRDF 答案的二维 LUT ,输入参数是单散射的albedo和距离。

实际上 Diffusion Profile 就是预计算散射光线在介质内部的 Random Walk 行为,然后做成 LUT 以供 BSSRDF 快速查询结果,从而达成实时渲染的性能需求。

常见有计算 Diffusion Profile 的方法:

  • 偶极子(Dipole)[2002]
  • 多级子(Multipole)[2005]
  • 高斯和(Sum-of-Gaussians)拟合 [2007]
  • Burley Normalized Diffusion 拟合 [2015]

偶极子 [2002]

将 BSSRDF 分成单散射部分和漫反射部分去拟合:

\(S = S_{ss}+S_{ms}\)

单散射近似:为了简化折射路径,假设光源离表面距离很远,将使用直射的ray来近似折射,因此可用 BRDF 来近似,结合蒙特卡洛方法计算出来。单散射的 radiance,受折射路径距离、折射吸收率扩散率影响:

\(L_{o}^{(1)}\left(x_{o}, \vec{\omega}_{o}\right)=\frac{\sigma_{s}\left(x_{o}\right) F p\left(\vec{\omega}_{i} \cdot \vec{\omega}_{o}\right)}{\sigma_{t c}} e^{-s_{i}^{\prime} \sigma_{t}\left(x_{i}\right)} e^{-s_{o}^{\prime} \sigma_{t}\left(x_{o}\right)} L_{i}\left(x_{i}, \vec{\omega}_{i}\right)\)

虽然BRDF是基于直射路径的,但是关于折射路径的距离仍然可以通过以下斯涅耳定律公式估计出:

\(s_{i}^{\prime}=s_{i} \frac{\left|\vec{\omega}_{i} \cdot \vec{n}_{i}\right|}{\sqrt{1-\left(\frac{1}{\eta}\right)^{2}\left(1-\left|\vec{\omega}_{i} \cdot \vec{n}\left(x_{i}\right)\right|^{2}\right)}}\)

漫反射近似:假设两次及以上的散射事件导致了光分布的均匀,因此这部分散射视为漫反射。使用偶极子来表示入射光源分布,上面为正的真实光源,下面为负的虚拟光源。

仍然采用蒙特卡洛方法,取N个sample分布于不同的半径 \(z_r\)(与负光源的距离),然后推算出一系列变量(如\(Z_v\)),其中 1 个sample 受偶极子源影响的漫反射:

\(\begin{aligned} R_{d}(r) &=\frac{\alpha^{\prime}}{4 \pi}\left[\left(\sigma_{t r} d_{r}+1\right) \frac{e^{-\sigma_{t r} d_{r}}}{\sigma_{t}^{\prime} d_{r}^{3}}+z_{v}\left(\sigma_{t r} d_{v}+1\right) \frac{e^{-\sigma_{t r} d_{v}}}{\sigma_{t}^{\prime} d_{v}^{3}}\right] \end{aligned}\)

偶极子方法实在太老而且极子部分涉及的物理知识完全没学过,没什么参考价值还是跳过把。

高斯和 [2007]

用若干个高斯函数去拟合 Diffusion Profile:

  • 1个高斯函数可以很好拟合多散射效果,但无法拟合单散射+多散射
  • 2个高斯函数可以勉强拟合单散射+多散射
  • 6个高斯函数可以得到相当高精度的拟合效果


Burley Normalized Diffusion [2015]

放弃物理,直接拟合。

  • 开销低,精确度却相当逼近基于蒙特卡洛暴力积分的无偏解
  • CDF 有解析解,可进行重要性采样优化

\(R_{d}(r)=\frac{e^{-r / d}+e^{-r / 3 d}}{8 \pi d r}\)

基于模糊的 SSS 方法

纹理空间模糊(Texture Space Blur) [2003]

  1. 渲染任何阴影贴图

  2. 渲染拉伸校正贴图(可以是预先计算的)

  3. 渲染辐照度纹理

  4. 纹理模糊:

    • 卷积核的权重:由高斯和拟合的 Diffusion Profile 确定
    • 卷积核的半径:由拉伸矫正贴图提供半径拉伸系数
    • 对每个高斯核进行可分离的模糊 Pass,并保存对应高斯卷积纹理
  5. 在 3D 中渲染网格:

    • 访问每个高斯卷积纹理并线性组合
    • 为每个光源添加镜面反射

屏幕空间模糊(Screen Space Blur) [2009]

  • 只需要对屏幕中 Stencil 标记过的 Skin像素进行若干卷积操作,极大地降低了 Blur 的像素数目
  • 卷积核的大小:根据当前像素的深度 z(x,y) 及其深度两个方向的导数来确定

Pre-Integrated Skin(预积分的皮肤着色)[2010]

Pre-Integrated Skin 是一个远远脱离物理基础的算法,其特点主要是高效、简单、视觉可接受

核心思路:

  • 预先对纹理进行卷积,而不用运行时卷积。
  • 查找表的参数分别是dot(N,L)和曲率,这两者结合就可以反映出光照随着曲率的变化。

Separable SSS(可分离的次表面散射)[2015]

BSSRDF 实质就是进行一次2D卷积运算,可分离的卷积核就是在寻找一个可分离的核来表示的 Diffusion Profile 的近似,然后转成两个1D 卷积 Pass ,让时间复杂度从O(n^2)=>O(n)

如图,先经过一次 X 方向的卷积 Pass ,再经过一次 Y 方向的卷积 Pass 就能得到相当于 2D 卷积的效果,最后再添加 Specular 就能得到很好的 SSS 材质效果。

注:不可以先添加 Specular 再进行卷积,不然会出现明显的 aritifact (例如高光处被卷积成十字形状)

  1. 高斯和函数的卷积核:传统拟合方法;1个2D高斯虽然可分离,6个2D高斯和却是不可分离的,除非使用6*2个1D卷积 Pass ,或者老老实实 2D 卷积,但都比较耗性能
  2. 基于SVD分解的可分一维卷积核:有局限
  3. 艺术家易用的卷积核:分解为近距离散射和远距离散射两个部分,两个高斯函数先相加后相乘,相当于四个高斯函数单独作用再相加

\(a_{m}(x)=w G\left(x, \sigma_{n}\right)+(1-w) G\left(x, \sigma_{f}\right)\)

\(A_{m}(x, y)=a_{m}(x) a_{m}(y)\)

  1. 增加引导函数的可分一维卷积核:给每个位置增加了一个重要性系数,用户可自定义,但没有直接的物理意义

  1. 预积分可分离核(Pre-integrated Separable Kernel):paper主要推荐的卷积核。

前提是假设 \(E\) 可分(相当于假设平面曲率为0),然后根据下列公式推导,2D核 \(R_d\) 就可被分离成两个1D核 \(a_p\) :

\(\begin{aligned} M_{e}(x, y)=& \iint E\left(x^{\prime}, y^{\prime}\right) R_{d}\left(x-x^{\prime}, y-y^{\prime}\right) d x^{\prime} d y^{\prime} \\=& \int E_{1}\left(x^{\prime}\right) \underbrace{\int R_{d}\left(x-x^{\prime}, y-y^{\prime}\right) d y^{\prime} d x^{\prime}}_{a_{p}\left(x-x^{\prime}\right)} \\ &+\int E_{2}\left(y^{\prime}\right) \underbrace{\int R_{d}\left(x-x^{\prime}, y-y^{\prime}\right) d x^{\prime}}_{a_{p}\left(y-y^{\prime}\right)} d y^{\prime} \\=& \iint E\left(x^{\prime}, y^{\prime}\right) \frac{1}{\left\|a_{p}\right\|_{1}} a_{p}\left(x-x^{\prime}\right) a_{p}\left(y-y^{\prime}\right) d x^{\prime} d y^{\prime} \end{aligned}\)

其中,\(\left\|a_{p}\right\|_{1}=\left\|R_{d}\right\|_{1} \quad\|x\|_{1} =\sum_{i}\left|x_{i}\right|\)

这样可以把2D卷积写为2个1D卷积:

\(\left(E * R_{d}\right)(x, y) = ((E * a_p)*a_p)(x, y)\)

具体实现

  1. 需提前有一份较精确的 diffusion profile(如6个高斯和拟合成的 \(R_d\) 函数),然后对 \(R_d\) 进行预先卷积(对距离的积分)得到 \(a_p(offset)\) 函数,换句话说就是一张一维表(输入一个 \(offset\) 值,输出一个 \(a_p\) 值)
  2. 在具体实现中,则使用若干个 sample 来取代这张一维表,规定好每个sample 的 offset 值 和计算出对应的 \(a_p\) 输出值
  3. 将这个 sample list 传给 shader,让 shader 根据这些配置好的 sample 进行两次 1D 卷积 Pass

注:即使E实际不可分,作者认为预计分可分离核的使用效果仍然足以接受,因此目前仍广泛适用于各类实时BSSRDF皮肤渲染上

额外处理工作:

  • 抖动 Jittering:paper 使用的一个可分核的扩散近似,在入射辐射度发生高频变化的地方会产生人工痕迹。这是因为信号的空间占用(尺寸)比核的宽度更小,从而产生不对称的星形图案。因为本文使用的近似核不是完全经向对称的

    • 对1D过滤器作用一个随机旋转(逐个像素)
    • 这种方式不需要对进行重新预积分,因为核的宽度并没有变化
    • 为了减少计算负荷,作者只对靠近像素的采样进行处理(10%),增加区域可以取得更好的效果,看下图区别:

  • 核的宽度:卷积核的尺寸受一个于深度相关的拉伸因子

\(s_{x}=\frac{a}{d(x, y)+\beta \cdot a b s\left(V_{x} d(x, y)\right)}\)

\(s_{y}=\frac{a}{d(x, y)+\beta \cdot a b s\left(V_{y} d(x, y)\right)}\)

  • 重要性采样:

    • 为了计算1D函数的卷积,需要对这个函数进行采样(离散化)
    • 为了计算效率,三个颜色通道使用一样的重要性采样,这个重要性采样由主导通道决定

    Bilateral Filtering [2010]

    绝大部分 BSSRDF 模型有一个很大的缺陷就是假设表面是平面,并且在平面上进行采样。这显然错误,现在通过 Bilateral Filtering 技术,可以把深度信息考虑进去做采样。

    \(I \approx \frac{2 \pi}{n} \sum_{i=1}^{n} \frac{R\left(r_{i}, d_{i}\right)}{p\left(r_{i}, d_{i}\right)} L=\frac{2 \pi}{n} \sum_{i=1}^{n} \frac{R(\sqrt{r_{i}^{2}+d_{i}^{2}})}{p\left(r_{i}, d_{i}\right)} L\)

    接下来就碰到困难了,\(p(r_i,d_i)\) 怎么算,之前的pdf是按平面计算的,它们的对应关系很难求。Unity 在报告中采用强制能量守恒,添加一个归一化常数。

    \(I=\frac{\sum_{i=1}^{n} \frac{R\left(\sqrt{r_{i}^{2}+d_{l}^{2}}\right)}{p\left(r_{i}, d_{i}\right)} L}{\sum_{i=1}^{n} \frac{R\left(\sqrt{\left.r_{i}^{2}+d_{l}^{2}\right)}\right.}{p\left(r_{i}, d_{i}\right)}}\)

Participating Media 渲染模型

Participating Media 是描述次表面散射最精确的模型,它描述了光线传播时因散射、吸收而导致的变化率方程:

\(\frac{\partial}{\partial t} L_{o}(p, \omega)=-\sigma_{t}(p, \omega) L_{i}(p,-\omega)+\sigma_{s}(p, \omega) \int_{\delta^{2}} phase \left(p,-\omega^{\prime}, \omega\right) L_{i}\left(p, \omega^{\prime}\right) d \omega^{\prime}\)

\(\sigma_a\) = absorption 系数,\(\sigma_s\) = in/out-scattering 系数,\(\sigma_t=\sigma_a+\sigma_s\) = extinction 系数

散射介质内部的 albedo 定义为:

\(\rho=\frac{\sigma_{s}}{\sigma_{s}+\sigma_{a}}=\frac{\sigma_{s}}{\sigma_{t}}\)

使用这个渲染模型就意味着要模拟光线在物体内部空间的随机游走(Random Walk),相比于只求了表面积分的 BSSRDF 渲染方程,它更加精确但也是计算量最大的方法,基于该方法的基本都是离线的体积渲染方法:

  • Volumetric Path Tracing
  • Volumetric Photon Mapping
  • 等等

图左是 BSSRDF 渲染模型做的,图右是Participating Media 渲染模型做的

Phase 函数

phase 函数表示了每对传入传出方向的能量分配概率。

各向同性 Phase

相当多的次表面散射的扩散模型假设各向同性,不捕获各向异性散射。这样简化的各向同性 phase 函数:

\(p=\frac{1}{4\pi}\)

各向异性 Phase

常用的各向 phase 函数便是 Henyey-Greenstein (HG) 函数,其中 n 为散射次数:

\(p_{n}\left(\omega \rightarrow \omega^{\prime}\right)=\frac{1-g^{2 n}}{4 \pi\left(1+g^{2 n}-2 g\left|g^{n-1}\right|\left(-\omega \cdot \omega^{\prime}\right)\right)^{3 / 2}}\)

皮肤的真实测量显示了各向异性值为 g = 0.8,随着n的增长,越来越呈现各向同性。

Schlick Phase 近似

获得与 HG Phase 函数相似结果的一种更快的方法是使用 Blasi 等人提出的近似方法,通常为第三位作者命名为 Schlick Phase 函数:

\(p(\theta, k)=\frac{1-k^{2}}{4 \pi(1+k \cos \theta)^{2}}\)

其中, \(k \approx 1.55 g-0.55 g^{3}\)

它不包括任何复杂的幂函数,而只是一个平方,这是更快的计算。 为了将该函数映射到原始HG相函数,需要从g中计算k参数。对于具有恒定g值的参与介质,只需执行一次。 在实际应用中,Schlick相位函数是一个很好的能量守恒近似。

Random Walk SSS(Volumetric Path Tracing)

Diffusion-based 近似现在仍然广泛使用,在假设介质是半无限长的平板(semi-infinte slab)的情况下,可以有很好的效果。而 MC 方法才能表现出物理正确,尤其在皮肤边缘处(例如鼻子边缘、耳朵边缘)更加符合现实。

图左是 normalized diffusion,图右是 path-traced SSS

Manifold Next Event Estimation,MNEE [2015]

  • Manifold Next Event Estimation [2015]

核心思路:已知摄像机到散射介质内表面的折射路径 \(x_a,x_1,x_b\),然后在面光源上采样一个点 \(x_c\),连接 \(x_b、x_c\) 为种子路径 \(Y\),然后根据采样半程向量 \(h\) 的结果,来不断迭代让路径 \(Y\) 折成可以满足折射规律的可接受的路径 \(X\)

Subdivision Next-Event Estimation,SNEE [2016]

  • Subdivision Next-Event Estimation for Path-Traced Subsurface Scattering [2016]

传统的游走路径最终的 BTDF 可能很难打在光源上(图左),而SNEE的主要想法是通过一条额外的子路径来满足折射规律的情况下还能直接打到光源上。

采样边界顶点 \(x_k\),两个采样子路径:

  • 一个从光源出发的自顶向下路径(NEE)
  • 一个根据 Phase 函数分布的自底向上路径(正常的 Phase 函数分布采样)

不同散射介质适用不同的nee技术:

  • 高密度介质,如皮肤, 仅需要在最后一个点做NEE
  • 对于低一些密度的介质,例如水或者液体,需要在内部每个游走点(单散射的)做NEE,并且使用 transparent shadows 来通过几何
  • 对于散射介质含内部表面,如眼球,可以使用MNEE产生焦散,随着在介质的透射距离而减弱

Path Sampling 的一些方法 [2016、2017]

  • Practical and Controllable Subsurface Scattering for Production Path Tracing [2016] [Disney]
  • Path Traced Subsurface Scattering using Anisotropic Phase Functions and Non-Exponential Free Flights [2017] [Pixar Animation Studios]

距离采样

  • 固定步长,无重要性

  • 根据单一波长的消散系数,适用于单色介质:\(p(s)=\sigma_{t} e^{-\sigma_{t} s}\)

  • 根据RGB三种波长的 transmittance (spectral MIS):

    • [2016] 累计吞吐量(throughput),权重取决于单散射 albedo。当某个波长吞吐量到达0时,就不需要做更多的采样了

    • [2017] 使用基于 RGB 三种波长的 MIS:

      \(P(s)=P(\lambda) * P_{\lambda}(s)\)

      \(P_{\lambda}(s)=\sigma_{\lambda} e^{\left(-s \sigma_{\lambda}\right)}\)

      MIS 权重将使用平衡启发式:\(W_{\lambda}(s)=P(\lambda) /\left(P\left(\lambda_{R}\right)+P\left(\lambda_{G}\right)+P\left(\lambda_{B}\right)\right)\)

      剩下的问题就是选取哪个 \(P(λ)\) ,也就是选择R、G、B哪个通道的波长来做距离采样:使用当前吞吐量(throughput)来指导选取某个 \(P(λ)\) 的概率,所谓吞吐量其实就是通道的贡献度,贡献度越高的通道更容易被选取

RGB三种波长的消散系数往往是不同的(transimittance是不同的),例如皮肤介质中的红光比其他波长传播得更远

方向性采样

  • 假设各向同性时,无重要性
  • 假设各向异性时,根据 Phase 函数分布做重要性采样(皮肤介质的 g 参数大约是0.8)

Dwivedi Sampling

  • 根据介质厚度“bias”一下采样来增加 random walk 返回到入射平面或者从另一个出射平面射出的机会,从而加速收敛
  • 需要先通过一个探测射线来估计物体厚度,而且需要将该物体假设为曲率为0的平板物体

将这些采样方法通过 MIS 组合起来使用。最后Pixar的论文实现中的 MIS 共使用了6个采样方法(for RGB renders)

对艺术家调参友好

  • [2016] 近似公式映射单散射 albedo \(\alpha\) 、表面 diffuse color、多重散射 albedo \(A\) ,减少控制参数数量:

    \(\begin{aligned} \alpha &=1-e^{-5.09406 A+2.61188 A^{2}-4.31805 A^{3}} \\ s &=1.9-A+3.5 *(A-0.8)^{2} \\ \sigma_{t} &=1 /(d * s) \end{aligned}\)

\(d\) 是散射距离。

  • [2017] Numerical Albedo Inversion:支持 diffuse color 来反向计算出 单散射 albedo,对艺术家调参友好 ;主要通过多项式来拟合

先算出输入单散射 albedo 到输出 diffuse color 的映射表(40 albedo \(\alpha\) * 40 各向异性系数 \(g\) = 1600 组合的 diffuse color)

  • \(A、C、D、F\) 将使用含 \(g\) 的多项式拟合:\(A(g)=a_{0}+a_{1} g+a_{2} g^{2}+a_{3} g^{3}+a_{4} g^{4}+a_{5} g^{5}+a_{6} g^{6}+a_{7} g^{7}\)
  • \(B、E\) 将使用含 \(g\) 的指数多项式拟合:\(B(g)=a_{0}+a_{1} \exp \left(a_{2} g+a_{3} g^{2}+a_{4} g^{3}+a_{5} g^{4}+a_{6} g^{5}+a_{7} g^{6}+a_{8} g^{7}\right)\)

只要记好了各种 \(a_i\) 系数,之后渲染就可以通过输入表面颜色 \(x\) 和各向异性系数 \(g\) 来得到单散射 albedo

\(\alpha(x, g)=\left(1-x^{0.25}\right) \cdot A(g) \tan ^{-1}(B(g) x)^{C(g)}+x^{0.25} \cdot D(g) \tan ^{-1}(E(g) x)^{F(g)}\)

  • [2017] Non-Exponential Free Flight:将传统物理的 Transmittance 公式(即散射传播距离与剩余光子率的关系)改成非指数的公式,参数更好控制,对艺术家友好

工业实现方案

Nvidia 方案 [Siggraph 2021]

Realistic Digital Human Rendering with Omniverse RTX Renderer,基于 Participating Media 渲染模型。

Nvidia 的实现中,主要基于以下几点:

  • 通过散射系数来决定距离采样
  • 没有使用 NEE
  • 仅单次采样的 spectral MIS(Single-sample spectral MIS)
  • Phase 函数使用各向异性 HG 函数(mean cosine is currently non-spectral)
  • 输入、射出介质表面时,使用 Lambertian BTDF
  • 对艺术家调参友好,支持 diffuse color 来反向计算出 单散射 albedo

实际效果:

UE4 方案 [Metahuman]

Skin Pipeline

  • 几何处理 Pass:根据各种贴图及某些混合计算方式来算出每个像素点的 BaseColor、Specular、Roughness、Normal 等属性(G-Buffer)
  • Lighting Pass:根据光源到表面的 diffuse transmission 和表面的 BaseColor 等属性计算出该像素点的 diffuse radiance
  • SSS Pass:使用 Separable SSS/Burley SSS/Both 实现基于屏幕空间的模糊
  • 其它 Pass:
    • 增加高光效果,即 Dual Specular
    • 改进的 SSAO
    • 其他后处理效果

皮肤毛孔

皮肤毛孔及其它皮肤表面细节导致了皮肤的各向异性,对于不同的地方应该有不同的specular、roughness等参数,一般是混合额外的细节贴图,这样就可以增加镜头很近时的皮肤细节

  • Specular:微观高光度贴图,与全局specular混合
  • Roughness:微观粗糙度贴图,与全局roughness混合,此外还通过Fresnel节点加强了角色视角边缘的粗糙度
  • Normal:微观法线贴图,主法线与微观法线按一定比例混合规范化

接近物理真实的渲染:\(cavity \cdot Specular(gloss) \cdot Fresnel(reflectance)\)

近似渲染: \(Specular(gloss) \cdot Fresnel(cavity \cdot reflectance)\)

cavity是凹陷度,可从cavity map(下图)中采样获得;Specular(gloss) 表明高光项;Fresnel(reflectance) 是与视觉角度相关的反射

皮肤反射

  • 直接反射:采用 双镜叶高光(Dual Lobe Specular),即两个 Cook-Torrance 的 BRDF Lobe 按一定比例混合:\(Lobe1⋅0.85 + Lobe2⋅0.15\)

  • 间接反射:Specular IBL

BSSRDF 渲染

Separable SSS:

  • 通过6个高斯和预计算profile,再用预积分的可分离核去近似拟合profile,运行时便是2个pass
  • 后处理:基于屏幕的模糊,且仅用两个pass(横向、纵向)去实现模糊,时间复杂度 O(n^2)=>O(2*n)

Burley Normalized SSS:

  • 基于 Burley Normalize Diffusion

Both:

  • 尚未看完源码,还不清楚怎么将这两种技术结合的

改进的 SSAO

  • AO 贴图混合了Bleed Color(皮肤通常取红色),使得皮肤渲染更加贴切,皮肤暗处的亮度和颜色更真实美观

Unity 方案 [Siggraph 2018]

薄物体透射

  • Thickness Map:专门用于记录物体厚度的纹理
  • Shadow Thickness:两层shadow 分别记录最远、最近深度来表示物体厚度
  • 结合两种方案,使用 \(max(textureThickness,shadowThickness)\)

实时&离线皮肤渲染技术(Real-time&Offline Skin Rendering)的更多相关文章

  1. 剖析Unreal Engine超真实人类的渲染技术Part 1 - 概述和皮肤渲染

    一.概述 1.1 数字人类的概要 数字人类(Digital Human)是利用计算机模拟真实人类的一种综合性的渲染技术.也被称为虚拟人类.超真实人类.照片级人类. 它是一种技术和艺术相结合的综合性模拟 ...

  2. 基于Unity 5的次世代卡通渲染技术 -- Unite 2017 米哈游总监贺甲分享实录

    在5月12日Unite2017开发者大会上,米哈游技术总监兼美术指导贺甲进行了主题为次世代卡通渲染的演讲.一下为详细分享内容: 大家好,首先自我介绍一下,我叫贺甲,在米哈游担任技术总监和美术指导工作, ...

  3. 剖析Unreal Engine超真实人类的渲染技术Part 3 - 毛发渲染及其它

    目录 四.毛发渲染 4.1 毛发的构造及渲染技术 4.1.1 毛发的构造 4.1.2 Marschner毛发渲染模型 4.1.3 毛发的间接光照 4.2 毛发的底层实现 4.3 毛发的材质解析 4.3 ...

  4. 剖析Unreal Engine超真实人类的渲染技术Part 2 - 眼球渲染

    目录 三.眼球渲染 3.1 眼球的构造及理论 3.1.1 眼球的构造 3.1.2 眼球的渲染理论 3.2 眼球的渲染技术 3.2.1 角膜的半透和光泽反射 3.2.2 瞳孔的次表面散射 3.2.3 瞳 ...

  5. 【原】实时渲染中常用的几种Rendering Path

    [原]实时渲染中常用的几种Rendering Path 本文转载请注明出处 —— polobymulberry-博客园 本文为我的图形学大作业的论文部分,介绍了一些Rendering Path,比较简 ...

  6. Unity酱~ 卡通渲染技术分析(二)

    前面的话 上一篇Unity酱~ 卡通渲染技术分析(一) 写了CharaMain.cginc,服装的渲染是怎么实现的.这篇来分析一下头发跟皮肤的实现 头发 本来以为unitychan的头发会有各向异性的 ...

  7. 实时离线一体化在资产租赁saas服务中使用

    流水查询需求 需求第一期: 基于TB级的在线数据,支持缴费帐单明细在线查询.大家都知道,像银行帐单流水一样,查几年的流水是常有的事. 支持的维度查询:帐期.欠费状态.日期范围.费用科目类型.房屋分类. ...

  8. Unity3d 屏幕空间人体皮肤知觉渲染&次表面散射Screen-Space Perceptual Rendering & Subsurface Scattering of Human Skin

    之前的人皮渲染相关 前篇1:unity3d Human skin real time rendering 真实模拟人皮实时渲染 前篇2:unity3d Human skin real time ren ...

  9. 新书《Cocos2dx 3.x 3D图形学渲染技术讲解》问世

    笔者介绍:姜雪伟,IT公司技术合伙人,IT高级讲师,CSDN社区专家,特邀编辑,畅销书作者,已出版书籍:<手把手教你架构3D游戏引擎>电子工业出版社和<Unity3D实战核心技术详解 ...

随机推荐

  1. Maven pom常用plugins配置说明

    maven-compiler-plugin 编译Java源码,一般只需设置编译的jdk版本 <plugin> <groupId>org.apache.maven.plugins ...

  2. 关于Jmeter线程组的设置,看这一篇就够了

    一.事件背景 个人感觉自己做性能测试,可以说是轻车熟路了,而且工作多年一直都是这一套测试思路及体系,从未质疑过自己,也许是狮子座的迷之自信吧! 也就在上周让我对自己的测试方法及体系产生了质疑! 为什么 ...

  3. myeclipse激活教程

    1.安装,解压,下一步一下一步,一直到finish..结束 2.汉化破解激活:下载破解压缩包:解压

  4. Jenkins系列-权限管理

    在实际工作中,存在多个团队都需要Jenkins来实现持续交付,但是又希望不同团队之间进行隔离,每个项目有自己的view, 只能看到自己项目的jenkins job. 但是,jenkins默认的权限管理 ...

  5. Protocol Buffer序列化Java框架-Protostuff

    了解Protocol Buffer 首先要知道什么是Protocol Buffer,在编程过程中,当涉及数据交换时,我们往往需要将对象进行序列化然后再传输.常见的序列化的格式有JSON,XML等,这些 ...

  6. web渗透工程师学习

    职位描述: 对公司网站.业务系统进行安全评估测试(黑盒.白盒测试): 对公司各类系统进行安全加固: 对公司安全事件进行响应,清理后门,根据日志分析攻击途径: 安全技术研究,包括安全防范技术,黑客技术等 ...

  7. 【NOI 2002 银河英雄传说】【带权并查集】

    题面 公元五八○一年,地球居民迁移至金牛座α第二行星,在那里发表银河联邦创立宣言,同年改元为宇宙历元年,并开始向银河系深处拓展. 宇宙历七九九年,银河系的两大军事集*在巴米利恩星域爆发战争.泰山压顶集 ...

  8. UOJ #228 - 基础数据结构练习题(势能线段树+复杂度分析)

    题面传送门 神仙题. 乍一看和经典题 花神游历各国有一点像,只不过多了一个区间加操作.不过多了这个区间加操作就无法再像花神游历各国那样暴力开根直到最小值为 \(1\) 为止的做法了,稍微感性理解一下即 ...

  9. cvc-complex-type.2.3: Element 'servlet' cannot have character [children], because the type's content

    错误原因:粘贴代码 <servlet> <servlet-name>barServlet</servlet-name> <servlet-class>S ...

  10. SQLite is 35% Faster Than The Filesystem

    比方说你要在C++/PHP里实现一个函数Image get_image(string id),不同的图片有1万张(用户头像),你可以把它们存在一个目录/文件夹里,然后fopen()再fread. 你也 ...