Unity中Shader表面着色器基础

写与灯光交互的Shader是很复杂的,因为有不同的灯光类型(light Type),不同的阴影选项(Shadow Options),不同的渲染路径(renderingPath:forward,deferred rendering),所以Shader应该以某种方式来处理所有的复杂情况。

Unity中的==Surface Shaders==是一种比使用底层的顶点/像素着色器程序(vertex/pixel shader programs)写起来更为简单通用的方式。 surfaceShaders 也是用Cg/hlsl来写。

How It Works

首先你需要定义一个"Surface"函数来获取任何uv信息或者你需要的数据作为输入,最后会填充在输出结构体"SurfaceOutput"里,SurfaceOutput是对表面属性的基本描述(例如Albedo,Normal,Emission,Specular等等),你可以将这些代码写在cg/hlsl里。

然后Surface Shaders计算编译需要哪些输入数据,以及哪些输出被填充等等,然后生成原生的vertex/pixed shaders,同时渲染通道会去处理forward和deferred渲染。

标准的表面着色器输出结构:

struct SurfaceOutput
{
    fixed3 Albedo;  // diffuse color /反射颜色
    fixed3 Normal;  // tangent space normal, if written(法向量)
    fixed3 Emission;//自发光颜色
    half Specular;  // specular power in 0..1 range//镜面反射度
    fixed Gloss;    // specular intensity(光泽度)
    fixed Alpha;    // alpha for transparencies(透明度)
};

在Unity5中,SurfaceShader也可以使用光照模型的基本物理属性。内置的"Standard"和"StandardSpecular"光照模型分别使用下面的输出结构体:

(Standard和StandardSpecular光照模型对应普通没有物理的的Lambert何BlinnPhong光照模型)

struct SurfaceOutputStandard
{
    fixed3 Albedo;      // base (diffuse or specular) color
    fixed3 Normal;      // tangent space normal, if written
    half3 Emission;
    half Metallic;      // 0=non-metal, 1=metal,金属度
    half Smoothness;    // 0=rough, 1=smooth,光滑度
    half Occlusion;     // occlusion (default 1),遮挡率
    fixed Alpha;        // alpha for transparencies
};
struct SurfaceOutputStandardSpecular
{
    fixed3 Albedo;      // diffuse color
    fixed3 Specular;    // specular color
    fixed3 Normal;      // tangent space normal, if written
    half3 Emission;
    half Smoothness;    // 0=rough, 1=smooth
    half Occlusion;     // occlusion (default 1)
    fixed Alpha;        // alpha for transparencies
};

对比SurfaceOutput和SurfaceOutputStandard,会发现前者的结构中Specular和Gloss被替换成在后者的结构中的Metallic,Smoothness,Occlusion三个属性。这三个属性就是Unity5里新加的物理属性。同样比较SurfaceOutputStandard和SurfaceOutputStanaedSpecular结构体,发现Metallic属性被Specular属性替换掉。

所以Unity5.x基于光照模型添加的基本物理属性应该是在Specular镜面发射上做了一些处理,添加了遮挡和粗糙度的属性以及混合Specular而形成的Metallic属性,使物体表面具有更加接近真实的物理光泽效果,从而减轻开发者自己书写相似效果的工作量。

##Surface Shader compile directives****(表面着色器编译指令)
表面着色器和其它Shader一样被放在CGPROGRAM……ENDCG块中执行,不同的是:

  • 代码必须写在子着色器块(SubShader)而不是通道(Pass)中,表面着色器会自己把它编译到多个通道里。
  • 代码必须使用#pragma surface ... 指令去指示它的表面着色器。

“#Pragma surface” 指令是:

#pragma surface surfaceFunction lightModel [optionalparams]

必要参数

  • surfaceFunction - 含有表面着色器代码的cg函数。这个函数必须是void surf(Input IN,inout SurfaceOutput o)的形式,Input是一个自己定义的结构体,Input结构体里应该包含一些纹理坐标和其它的surface函数需要的其它的可以被自动识别的变量(autonmatic variables)。
  • lightmodel - 使用的光照模型。内置一些基于物理的StandardStandardSpecular光照模型,以及一些没有基于物理的Lambert(Diffuse)和BlinnPhong(specular)光照模型,当然也可以自己写光照模型(参见:自定义光照模型
  • Standard光照模型使用SurfaceOutputStandard作为输出结构体,对应Unity中的Standard(Metallic workflow)Shdaer。
  • StandardSpecular光照模型使用SurfaceOutputStandardSpecular作为输出结构体,对应Unity中的Standard(Specular workflow)Shader
  • Lambert和BlinnPhong光照模型是不基于物理的(大部分在Unity4.x中),但是在低端设备上使用会更加快,性能更好。

****可选参数****

Transparency和Alpha testing是由alpha和alphatest指令控制。不透明度主要分为两种:传统的alpha混合(一般用于淡出物体)或更加看似物理的“premultiplied blending”(允许半透明表面保留适当的镜面反射效果)。使半透明能够使用生成的表面着色器代码包含blending指令,而能够让alpha基于已有的变量剔除被生成的pixed shader丢弃的片段。

  • ****alpha**** 或者 ****alpha:auto****: 将会为简单的光照函数挑出淡出透明度(就像alpha:fade),并且为基于物理的光照函数预乘透明度(就像 alpha:premul)。
  • ****alpha:fade****:使用传统的淡出透明。
  • ****alpha:premul****:使用预乘alpha透明度。
  • ****alphatest : variableName****:使用alpha裁剪透明度。剔除值是一个浮点型变量variableName,你可能也会想使用addshadow指令去生成适当的阴影接受通道(shadow caster pass)。
  • ****keepalpha****:默认不透明(Opaque)选项,不管输出结构的Alpha或者光照函数的返回值都将表面着色器将Alpha通道值设置为1.0。使用这个选项允许保持光照函数的返回alpha值,即使使用的是不透明表面着色器。
  • ****decal:add****: 额外的贴花着色器(例如,terrian AddPass),这意味着物体在其它表面之上使用additive blending。
  • ****decal:blend****:半透明贴花着色器,这意味物体可以在其它物体表面使用alpha blending。

自定义函数可以被用来改变或计算来自顶点的数据,或者改变最后计算得出的片段颜色。

  • ****vertex:VertexFunction****,用户自定义的顶点修改函数。这个函数会在生成的顶点着色器开始被调用,并且可以修改或计算每个顶点的数据。
  • ****finalcolor:ColorFunction****,自定义的最终颜色修改函数。

Shadows and Tessellation(阴影和镶嵌花纹),可以控制如何处理阴影和镶嵌花纹的额外指令。

  • ****addshadow****,生成一个阴影投射通道,通常被用在自定义顶点修改函数中,因为阴影投射也获得程序顶点动画。通常shaders不需要任何特殊的shadows处理,因为可以使用它们FallBack里的阴影投射通道。
  • ****fullforwardshadows****,支持正向渲染路径(forward render path)的所有的灯光阴影。默认的着色器仅支持来自directional light的正向渲染(为了节省内部着色器变量数量),如果需要point light和spotlight在正向渲染中的阴影,可以直接使用这个。
  • ****tessellate:TessFunction****,使用DX11 GPU Tessellation,函数计算镶嵌纹理因子。

Code Generation(代码生成) 选项,通过默认生成的表面着色器代码试图去处理各种可能的lighting/shadowing/lightmap情况,然而有些情况下你不需要他们中的一些内容,这里你就可以调整代码生成去跳过它们。这样的结果会使你的shader更小并且被更快的加载。

  • ****exclude_path:deferred,forward,prepass****.不要为已经给的渲染路径生成pass,
  • ****noshadow****,不支持阴影接收。
  • ****noambient****,不适用于环境光或者light probes。
  • ****novertexlights****,不适用于正向渲染(forward rendering)的任何light probes或者逐顶点光照。
  • ****nolightmap****,不支持光照贴图。
  • ****nodynlightmap****,不支持动态的全局光照。
  • ****nodirlightmap**** , shader不支持定向光照贴图。
  • ****nofog****,不支持所有的内置雾效。
  • ****nometa****,不生产”meta“通道,常被用于lightmapping和动态全局照明去提取表面信息。
  • ****noforwardadd****,不适用正向渲染额外的通道。这个使shaders支持一个全局光,而使用其它灯光逐顶点计算??梢允箂hader更小更有效率。

其它

  • ****softvegetation****,使表面着色器仅仅在软植被上的时候被渲染。
  • ****interpolateview****,计算在顶点着色器上视觉方向并且进行差值来代替在像素着色器中计算??梢允瓜袼刈派鞲?,但是会使用多张贴图进行差值。
  • ****halfasview****,通过half-direction向量进入光照函数来代替视角方向,half-direction会被逐顶点计算并归一化。这会更快,但并不总是正确。
  • ****approxview****,Unity5中移除了,使用interpolateview代替。
  • ****dualforward****,在正向渲染路径中使用贴花光照贴图。

表面着色器输入结构

在输入结构体Input中通?;嵊幸恍﹕hader使用需要的纹理坐标。纹理坐标必须以uv命名开头,后面街上纹理名称(或者以uv2开头,使用第二个纹理坐标)。

其它可以被放到Input结构中的值有:

  • ****float3 viewDir****,将含有视角方向,用于计算视差效果,边缘光照等。
  • ****float4 ****加上****COLOR****语意绑定********,将会包含差值的逐顶点颜色。
  • float4 screenPos,将包含屏幕空间位置,用于反射后屏幕空间效果。
  • ****float3 worldPos****,包含世界空间位置信息。
  • ****float3 worldRef1****,如果表面着色器没有写给o.Normal,将包含世界反射向量。
  • ****float3 worldNormal****,如果表面着色器没有写给o.Normal,将包含世界法向量。
  • ****float3 worldRef1;INTERNAL_DATA****,如果表面着色器没有写给o.Normal,将包含世界反射向量。使用WorldReflectionVector (IN, o.Normal)基于逐顶点法线贴图获取反射向量。
  • ****float3 worldNomal;INTERANT_DATA****,如果表面着色器没有写给o.Normal,使用WorldReflectionVector (IN, o.Normal)基于逐顶点法线贴图获取法线向量。

现在有部分表面着色器编译管线不支持DX11上特殊的HLSL语法,所以如果你使用HLSL一些特性,比如StructuredBuffers,RWTextures和其它的DX9没有的特性,可以使用宏包裹住。

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

推荐阅读更多精彩内容