【Unity Shader入门精要学习】Surface Shader(一)

Surface Shader的组成

一、编译指令

编译指令

所谓编译指令就是使用命令来告诉Unity我将使用那些函数以及光照模型来处理相应的数据,并输出我想要的颜色,同时不要生成哪些渲染路径下的代码(如延迟渲染等)

        // surf - 要使用的表面函数.
        // CustomLambert - 要使用的自定义光照函数.
        // vertex:myvert - 自定义模型定点修改函数.
        // finalcolor:mycolor - 最终要使用的处理颜色的函数.
        // addshadow - 生成正确的阴影,因为在修改模型定点后会产生错误的阴影.
        // exclude_path:延时渲染路径的代码不要生成
        // nometa - do not generate a “meta” pass (that’s used by lightmapping & dynamic global illumination to extract surface information).
        #pragma surface surf CustomLambert vertex:myvert finalcolor:mycolor addshadow exclude_path:deferred exclude_path:prepass nometa

二、默认自定义函数

1、表面函数(surFunction)
        void surf (Input IN, inout SurfaceOutput o) 
        {
            fixed4 tex = tex2D(_MainTex, IN.uv_MainTex);
            o.Albedo = tex.rgb;
            o.Alpha = tex.a;
            o.Normal = UnpackNormal(tex2D(_BumpMap, IN.uv_BumpMap));
        }

因为在计算光照引起的颜色的时候我们经常需要根据输入的一些信息如纹理颜色,透明度,反射率,法线等来计算最终的颜色,所以SurFunction就相当于把这部分抽象出来,根据输入改变对应的值。

2、自定义改变顶点函数
        void myvert (inout appdata_full v) 
        {
            v.vertex.xyz += v.normal * _Amount;
        }

这个函数的作用是用来改变输入给顶点着色器的数据,理论上凡是输入给顶点着色器的数据都可以改变:

v2f_surf vert_surf (appdata_full v)
 {
    v2f_surf o;
    ......
    myvert (v);
    o.pos = UnityObjectToClipPos(v.vertex);
    o.pack0.xy = TRANSFORM_TEX(v.texcoord, _MainTex);
    o.pack0.zw = TRANSFORM_TEX(v.texcoord, _BumpMap);
    .....
    return o;
}
3、自定义光照模型函数
        half4 LightingCustomLambert (SurfaceOutput s, half3 lightDir, half atten)
         {
            half NdotL = dot(s.Normal, lightDir);
            half4 c;
            c.rgb = s.Albedo * _LightColor0.rgb * (NdotL * atten);
            c.a = s.Alpha;
            return c;
        }

如果不想使用提供好的光照模型,我们可以自定义关照模型,但是函数的参数类型和个数是固定的。

4、自定义最终改变颜色函数
        void mycolor (Input IN, SurfaceOutput o, inout fixed4 color) 
        {
            color *= _ColorTint;
        }

在最终输出颜色之前会调用这个函数,经过这个函数后才知我们想要的颜色。

三、代码的生成以及调用过程

1、代码生成

以下代码作为示例:

Shader "Custom/NormalExtrusion"
{
    Properties {
        _ColorTint ("Color Tint", Color) = (1,1,1,1)
        _MainTex ("Base (RGB)", 2D) = "white" {}
        _BumpMap ("Normalmap", 2D) = "bump" {}
        _Amount ("Extrusion Amount", Range(-0.5, 0.5)) = 0.1
    }
    SubShader {
        Tags { "RenderType"="Opaque" }
        LOD 300
        
        CGPROGRAM
    
        #pragma surface surf CustomLambert vertex:myvert finalcolor:mycolor addshadow exclude_path:deferred exclude_path:prepass nometa
        #pragma target 3.0
        
        fixed4 _ColorTint;
        sampler2D _MainTex;
        sampler2D _BumpMap;
        half _Amount;
        
        struct Input {
            float2 uv_MainTex;
            float2 uv_BumpMap;
        };
        
        void myvert (inout appdata_full v) 
        {
            v.vertex.xyz += v.normal * _Amount;
        }
        
        void surf (Input IN, inout SurfaceOutput o) 
        {
            fixed4 tex = tex2D(_MainTex, IN.uv_MainTex);
            o.Albedo = tex.rgb;
            o.Alpha = tex.a;
            o.Normal = UnpackNormal(tex2D(_BumpMap, IN.uv_BumpMap));
        }
        
        half4 LightingCustomLambert (SurfaceOutput s, half3 lightDir, half atten)
         {
            half NdotL = dot(s.Normal, lightDir);
            half4 c;
            c.rgb = s.Albedo * _LightColor0.rgb * (NdotL * atten);
            c.a = s.Alpha;
            return c;
        }
        
        void mycolor (Input IN, SurfaceOutput o, inout fixed4 color) 
        {
            color *= _ColorTint;
        }
        
        ENDCG
    }
    FallBack "Legacy Shaders/Diffuse"
}

以上代码会形成以法线方向上的膨胀效果,因为改变了模型的顶点位置。


生成代码

生成代码后打开:


代码生成

虽然很多,但发现基本上只有三个Pass,一个Base Pass用来处理逐像素平行光,一个Add Pass用来处理其他逐像素的光,一个Shadow Caster用来处理阴影投射。
无论几个Pass,里面都包含了我们所写的几个函数:
image.png
2、调用过程

自定义的顶点修改函数会在每个Pass的顶点着色器中最开始的地方执行:

myvert

SurFunction用于输出改变后的法线,反射率,透明度等信息,会在每个Pass的片元着色器中,根据我们给的INPUT结构调用并输出:
surf

自定义光照模型函数在SurFunction后执行,使用的正是SurFunction的输出
LightingCustomLambert

自定义最终改变颜色函数一般只会在Base Pass的片元着色器中执行一次,用来做最后输出:
mycolor

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