Unity3d shader之次表面散射(Subsurface Scattering)
次表面散射是一种非常常用的效果,可以用在很多材质上
如皮肤,牛奶,奶油奶酪,番茄酱,土豆等等
初衷是想做一个牛奶shader的,但后来就干脆研究了sss
这是在vray上的次表面散射效果
这是本文在unity中实现了的次表面散射效果:
左侧为BRDF(双向反射分布),右侧为BSSRDF(双向次表面散射反射分布)
各参数如上为:
S:BSSRDF结果
Rd: BSSRDF的漫反射
Fr:: 菲涅尔反射
Ft: 菲涅尔透明度,透射比
Fdr: 菲涅尔漫反射的反射率
E:辐照方向
Wiki
中给出:radiant fluence is the radiant energy received by a surface per unit
area, or equivalently the irradiance of a surface integrated over time
of irradiation
Phi:每单位表面受到的辐照能
Sigma A: 吸收率
Sigma S: 散射率
Sigma T: 消散率
Sigma T’ : 减少消散率
Sigma TR : 有效消散率
D:漫反射常量
Alpha: 反射率
P:相函数
Eta:反射的相关指数
g:散射角的平均cos值
Q:源分布值
Q0:第0个源分布
Q1:第1个源分布
漫反射近似
漫反射近似是基于光线高分散媒介倾向于各向异性的观察,光源的分布与相函数是各向异性的。每次散射都模糊了光线的分布,随着散射的次数增多导致光线的分布更加均匀。
这种辐照类似于一个二项式涉及单位表面受到的辐照能和辐照方向
使用了Henyey-Greenstein的相函数:
常量决定于单位表面受到的辐照能和辐照方向。
对于一个无穷小的光线进入了一个媒介,入射能量将随着进入深度s呈指数性减小
减小强度:
<span style="font-size:14px;">float Lri(float3 w_P, float phi_x, float p_L_Dist, float D)
{
float _Sigma_t = _Sigma_A + _Sigma_S;
float L = 1 / (4 * PIE) * phi_x + 3 / (4 * PIE) * dot(w_P, -D*_Nabla * phi_x); float Lri = L * pow(E, -_Sigma_t* p_L_Dist);
return Lri;
}</span>
第一次散射减小强度,被作为体积来源处理
for (int i = 0; i < 30; i++)
{
w_P = normalize(float3(N.x + rand(fixed2(i*0.05, i*0.05)), N.y + rand(fixed2(-i*0.05, i*0.05)), N.z + rand(fixed2(i*0.05, -i*0.05))));
// float3 w_P = normalize(float3(lightDir.x + rand(i.uv_MainTex + fixed2(i*0.01, i*0.01)), lightDir.y + rand(i.uv_MainTex + fixed2(-i*0.01, i*0.01)), lightDir.z + rand(i.uv_MainTex + fixed2(i*0.01, -i*0.01)))); Q += phase(dot(lightDir, w_P))*Lri(w_P, phi_x, p_L_Dist, D);
Q *= _Sigma_S;
Q1 += Q*w_P; }
30次随机光线散射方向
观察光在体积内部传播行为,这个方程式很有用
这个方程式与辐照度标量或通量相关
第0个与第一个源分布公式
Sigma参数之间的互相推倒,
光线变成各向异性的,后向散射关系改变了净通量,前向散射与无散射是没有区别的。
此处D = 1/(3* sigma_T’);是漫反射常量
最终我们得到了漫反射公式
漫反射部分的推导公式,得到如下结果
在做定积分时进行叠加了30次随机光线散射方向,效果还算不错。
漫反射的反射部分
然后就是求漫反射的反射部分
菲涅尔反射公式,在可传导介质的菲涅耳漫反射的反射:
媒介本身的性质不同反射器情况也不同,Eta为这种性质的相关指数
这是经过精确测量的反射率,我们可以用这个公式来免去计算消耗
通量公式:
Dr = ||x - xr||为当前点与光源的距离
Dv = ||x-xv||为当前点与眼睛(相机)的距离
if (_WorldSpaceLightPos0.w != 0)
{
p_L_Dist = distance(_WorldSpaceLightPos0, i.worldPos);
}
float v_C_Dist = distance(_WorldSpaceCameraPos, i.worldPos)*0.3;
Φ为光源强度
最终,我们的反射公式为
<span style="font-size:14px;">float3 ref = -D * (dot(N, _Nabla*phi_x_S)) / (diff*_LumPow_D);</span>
在最后加入Physically-Based Rendering的specular,大功告成
参数调节
关于参数调节,参数非常不好调,pdf上和自己弄得参数不搭,只能自己调了
最终效果:
参考:
1. A Practical Model for Subsurface Light Transport
2. A Measurement-Based Skin Reflectance Model for Face Rendering and Editing
Unity3d shader之次表面散射(Subsurface Scattering)的更多相关文章
- 次表面散射(SubSurface Scattering) Shader 【转】
原文 http://www.azure.com.cn/article.asp?id=231 用深度值近似模拟物体的厚度,厚度越小处透光越多. varying vec4 position;varying ...
- unity3d shader之实时室外光线散射(大气散射)渲染
散射需要:吸收,内散射,外散射分为瑞利散射Rayleigh Scattering和米氏散射 Mie Scattering 后面会详细讲解 大气中散射由多种原因产生,微粒,尘埃,水蒸气等等 阳光由于散 ...
- Unity3d 屏幕空间人体皮肤知觉渲染&次表面散射Screen-Space Perceptual Rendering & Subsurface Scattering of Human Skin
之前的人皮渲染相关 前篇1:unity3d Human skin real time rendering 真实模拟人皮实时渲染 前篇2:unity3d Human skin real time ren ...
- (转)GEM -次表面散射的实时近似
次表面散射(Subsurface Scattering),简称SSS,或3S,是光射入非金属材质后在内部发生散射, 最后射出物体并进入视野中产生的现象, 即光从表面进入物体经过内部散射,然后又通过物体 ...
- 【译】Unity3D Shader 新手教程(1/6)
本文为翻译,附上原文链接. 转载请注明出处--polobymulberry-博客园. 刚开始接触Unity3D Shader编程时,你会发现有关shader的文档相当散,这也造成初学者对Unity3D ...
- Unity3D shader简介
Unity3D shader简介 可以肯定的说Unity3D使得很多开发者开发游戏更容易.毫无疑问,shader(着色器)编码,仍有很长的路要走.shader是一个专门运行在GPU的程序,经常被神秘包 ...
- 【浅墨Unity3D Shader编程】之一 夏威夷篇:游戏场景的创建 & 第一个Shader的书写
本系列文章由@浅墨_毛星云 出品,转载请注明出处. 文章链接:http://blog.csdn.net/poem_qianmo/article/details/40723789 作者:毛星云(浅墨) ...
- 转 猫都能学会的Unity3D Shader入门指南(二)
猫都能学会的Unity3D Shader入门指南(二) 关于本系列 这是Unity3D Shader入门指南系列的第二篇,本系列面向的对象是新接触Shader开发的Unity3D使用者,因为我本身自己 ...
- Unity3D Shader入门指南(二)
关于本系列 这是Unity3D Shader入门指南系列的第二篇,本系列面向的对象是新接触Shader开发的Unity3D使用者,因为我本身自己也是Shader初学者,因此可能会存在错误或者疏漏,如果 ...
随机推荐
- (转)php 获取今日、昨日、上周、本月的起始时间戳和结束时间戳的方法
php 获取今日.昨日.上周.本月的起始时间戳和结束时间戳的方法,主要使用到了 php 的时间函数 mktime. 下面首先还是直奔主题以示例说明如何使用 mktime 获取今日.昨日.上周.本月的起 ...
- CSS3高性能动画
CSS动画属性会触发整个页面的重排relayout.重绘repaint.重组recomposite Paint通常是其中最花费性能的,尽可能避免使用触发paint的CSS动画属性,在CSS动画中使用w ...
- MyEclipse激活失败,解决办法
文章参考:http://www.cnblogs.com/dingyuanxin/p/4046356.html 失败可能是:systemid和exe破解出来的那个对应不上: 1.启动MyEclipse, ...
- delphi 功能函数大全-备份用
function CheckTask(ExeFileName: string): Boolean;constPROCESS_TERMINATE=$0001;varContinueLoop: BOOL; ...
- My97 DatePicker 选择时间后弹出选择的时间
项目中用到这个时间插件,注册用户时可以选中永久和选择时间,二者是互斥关系, 所以在选择时间插件时,需要绑定一个事件,所以看到了这个插件: <input id="yydate" ...
- CSS 列表
CSS列表属性作用如下: 设置不同的列表项标记为有序列表 设置不同的列表项标记为无序列表 设置列表项标记为图像 列表 在HTML中,有两种类型的列表: 无序列表 - 列表项标记用特殊图形(如小黑点.小 ...
- 打印出不同顺序的字符串&单引号和双引号的差异
发现一个很好玩的打印顺序 package com.liaojianya.chapter1; /** * This program demonstrates the string. * @author ...
- PC110304/UVA850
这题目WA了好几次,主要是我没有理解清楚No solution.这个情况. 如果在match原文做好了,基本map一下就能过了. 与原句match的条件就是: 1.出现了26个字母 2.该空格的地方要 ...
- hibernate一些方法
session.flush() 同步缓存与数据库数据 session.evict(obj) 关闭指定对象缓存 session.clear() 关闭所有缓存 iterator(会把数据放入缓存) 下次 ...
- C/C++之Exercise
一.C/C++之初学Demo---C++调用C.h文件使用实例: 工程结构: exercise.h code: #ifndef _EXERCISE_H_ #define _EXERCISE_H_ #i ...