Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02

这篇具有很好参考价值的文章主要介绍了Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。


前言

上一篇:Unity Shader 学习(一):初识ShaderLab – 以“Unlit Shader”模板为例 01

在Unlit Shader模板中引用了UnityCG.cginc中的很多宏和函数方法,这篇继续学习模板中的几个函数以及UnityCG.cginc中一些可能会用到的结构;最后再了解一些自己写Shader时大概率会用到的Cg/HLSL函数方法。


一、Unlit Shader 模板中的函数

sampler2D _MainTex;
float4 _MainTex_ST;

v2f vert (appdata v)
{
    v2f o;
    o.vertex = UnityObjectToClipPos(v.vertex);
    o.uv = TRANSFORM_TEX(v.uv, _MainTex);
    // UNITY_TRANSFER_FOG(o,o.vertex);
    return o;
}

fixed4 frag (v2f i) : SV_Target
{
    // sample the texture
    fixed4 col = tex2D(_MainTex, i.uv);
    // apply fog
    // UNITY_APPLY_FOG(i.fogCoord, col);
    return col;
}

有两个方法:UnityObjectToClipPostex2D;一个宏:TRANSFORM_TEX

1. UnityObjectToClipPos

这个方法的定义UnityShaderUtilities.cginc

Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎
注释:(inlinein这些关键字,是Cg的语法,参考:Cg_language)

// Tranforms position from object to homogenous space

方法的含义是把顶点坐标从对象空间转换到齐次裁剪空间,是一个坐标变换方法;这正是渲染流程中顶点着色器阶段的主要工作之一

2. TRANSFORM_TEX

这个宏的定义UnityCG.cginc

Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎
模板中的顶点着色器函数等价于:

sampler2D _MainTex;
float4 _MainTex_ST;

v2f vert (appdata v)
{
    ...
    o.uv = v.uv.xy * _MainTex_ST.xy + _MainTex_ST.zw;
    ...
}

这下知道了为什么模板中要定义一个看起来没有被调用过的变量_MainTex_ST,其实是藏在里了。

_MainTex_ST的含义,就是纹理的一个额外属性:缩放和偏移值。

注释:

// Transforms 2D UV by scale/bias property

方法的含义是:顶点uvTilingOffset两个属性值进行运算,计算出实际显示时的顶点uv

在Material上能看到:

  • Tiling:对应scale property,表示纹理放置比例XY通过_MainTex_ST.xy读取;
  • Offset:对应bias property,表示纹理位置偏移量XY通过_MainTex_ST.zw读取。

Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎

更好的理解,可参考:Unity中Material的“Tiling" 和 "Offset"参数介绍

3. tex2D

这个方法是Cg/HLSL语言内置的

官方文档:

  • tex2D - Nvidia
  • tex2D (HLSL 参考)

Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎

方法的含义是查找并返回采样纹理(_MainTex)的(i.uv)坐标处的颜色值;即纹理采样

sampler2D _MainTex;

fixed4 frag (v2f i) : SV_Target
{
    // sample the texture
    fixed4 col = tex2D(_MainTex, i.uv);
    ...
}

更好的理解,可参考:

  • 【unity Shader 学习笔记】3-3纹理采样(了解 Wrap Mode)
  • UnityShader关键字:tex2D(sampler2D,float2) 和 _MainTex_ST(了解 Filter Mode)

拓展:使用采样器状态 - Unity Manual

以上,是 Unlit Shader 模板中使用到的方法,也是在顶点着色器和片元着色器阶段非常核心的方法。

二、CGInclude中的数据结构

这里暂时只了解一些输入输出结构,更多请参考:

  • (二)unity自带的着色器源码剖析之—UnityCG.cginc文件(上篇)
  • (三)unity自带的着色器源码剖析之—UnityCG.cginc文件(下篇)

1. 顶点着色器输入结构

顶点着色器输入结构appdata,是模板中自定义的,里面只有两个属性:

struct appdata
{
    float4 vertex : POSITION;
    float2 uv : TEXCOORD0;
};

v2f vert (appdata v)
{
    v2f o;
    o.vertex = UnityObjectToClipPos(v.vertex);
    o.uv = TRANSFORM_TEX(v.uv, _MainTex);
    // UNITY_TRANSFER_FOG(o,o.vertex);
    return o;
}

实际上,顶点着色器输入的顶点数据远不止这些,只不过是有些输入的属性暂时用不到,被无视了;在UnityCG.cginc中,还提供了其它几种输入结构,这里只列出最全的那个:

struct appdata_full 
{
    float4 vertex : POSITION; // 顶点坐标
    float4 tangent : TANGENT; // 切线向量
    float3 normal : NORMAL; // 法线向量
    float4 texcoord : TEXCOORD0; // 第一套纹理坐标
    float4 texcoord1 : TEXCOORD1; // 第二套纹理坐标
    float4 texcoord2 : TEXCOORD2; // 第三套纹理坐标
    float4 texcoord3 : TEXCOORD3; // 第四套纹理坐标
    fixed4 color : COLOR; // 顶点颜色
    UNITY_VERTEX_INPUT_INSTANCE_ID // 即:uint instanceID : SV_InstanceID;
};

Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎
可以看到appdata_img类似于模板自定义的appdata,只不过多了一个UNITY_VERTEX_INPUT_INSTANCE_ID

Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎

把上面模板的顶点着色器函数改写成:

v2f vert (appdata_img v)
{
    v2f o;
    o.vertex = UnityObjectToClipPos(v.vertex);
    o.uv = TRANSFORM_TEX(v.texcoord, _MainTex);
    // UNITY_TRANSFER_FOG(o,o.vertex);
    return o;
}

至于这个UNITY_VERTEX_INPUT_INSTANCE_ID到底是什么,再深入学习就是另外一个话题了,仅做拓展了解:

  • GPU 实例化
  • SV_InstanceID

2. 片元着色器输入结构

与上面顶点着色器输入同理,模板自定义了v2f结构:

struct v2f
{
    float2 uv : TEXCOORD0;
    // UNITY_FOG_COORDS(1)
    float4 vertex : SV_POSITION;
};

v2f vert (appdata v) // 顶点着色器
{
    v2f o;
    o.vertex = UnityObjectToClipPos(v.vertex);
    o.uv = TRANSFORM_TEX(v.uv, _MainTex);
    // UNITY_TRANSFER_FOG(o,o.vertex);
    return o;
}

fixed4 frag (v2f i) : SV_Target // 片元着色器
{
    // sample the texture
    fixed4 col = tex2D(_MainTex, i.uv);
    // apply fog
    // UNITY_APPLY_FOG(i.fogCoord, col);
    return col;
}

同样,片元着色器输入的数据也不是只有两个,比如UnityCG.cginc中的v2f_vertex_lit,就定义了两个颜色值表示顶点漫反射颜色(diff)和镜面反射颜色(spec);

Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎

顶点着色器函数输入变量appdata v赋值过程不可见的,只能直接调用;与之不同的是,输出变量v2f o赋值过程是开发者自己写的:

v2f o;
o.vertex = UnityObjectToClipPos(v.vertex);
o.uv = TRANSFORM_TEX(v.uv, _MainTex);

因此,顶点着色器输出/片元着色器输入的结构v2f定义自由度比较高没有什么统一的模板结构;结构中属性的数量、每个属性的含义可由开发者灵活定义,想输出什么就定义什么,当然,想输出的值由自己在顶点着色器函数中计算求得。

这可能是Unity内置CGInclude文件中顶点着色器输出/片元着色器输入结构不多的原因:通用性不强;即使是已有的两个结构:v2f_vertex_litv2f_img也是Unity为了实现某些方法自己定义的:

Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎
Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎

如果一定要将自己定义的v2f改为v2f_img,除了不再支持原来的雾效相关代码之外,没什么区别,因为赋值都是一样的:

v2f_img vert (appdata v)
{
    v2f_img o;
    o.pos = UnityObjectToClipPos(v.vertex);
    o.uv = TRANSFORM_TEX(v.uv, _MainTex);
    return o;
}

fixed4 frag (v2f_img i) : SV_Target
{
    // sample the texture
    fixed4 col = tex2D(_MainTex, i.uv);
    // v2f_img中没有UNITY_FOG_COORDS(1),i.fogCoord也就不存在了
    return col;
}

如果不是要用到某些CGInclude文件中特定的方法,函数的输入输出结构自己定义或者不定义结构直接写多参数函数(参考:自定义着色器基础 - Unity Manual)就可以。

三、Cg/HLSL中用到的函数方法

1. 构造函数(向量类型)

在学习各种内置函数之前,先学会最基础、最常用的函数:向量类型的构造函数

根据Cg的语法,构造一个向量有两种方式:

Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎

  • 对每个分量逐一赋值构造

    float2 v2 = float2(0, 1);
    half3 v3 = half3(-1, 1.1, 0);
    fixed4 v4 = fixed4(1, 0, -1.0, 0);
    

    注意:每个分量都要给一个确定的值,与C#不同,内部的构造函数不会自动补零;

    在C#中也许可以这样写:

    Vector4 vector = new Vector4(1, 1); // 构造函数内部默认把z、w分量设置为0了
    

    但是在Cg/HLSL中,这样写会编译错误的:

    fixed4 v4 = fixed4(1, 1); // 错误写法,编译不通过
    

    Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎

  • 基于其它向量构造:(最常用)

    float2 v2 = float2(0, 1);
    float3 v3 = float3(v2,1); // float3 v3 = float3(1,v2);
    float4 v4 = float4(v3,1); // float4 v4 = float4(1,v3); // float4 v4 = float4(v2,v2); // float4 v4 = float4(1,v2,1);
    

    低维向量按任意顺序组合成高维向量,只需要保证:3=1+2,4=1+3、2+2或1+1+2;

    即使精度不同也可以构造,而且低精度可以构造高精度高精度也可以构造低精度,以上面为例:

    fixed4 v4 = fixed4(1, 0, -1, 0);
    half3 v3 = half3(v4.xy, 1);
    float2 v2 = float2(v3.z, v4.w);
    
    float2 v2 = float2(0, 1);
    half3 v3 = half3(v2, 1);
    fixed4 v4 = fixed4(1, v3);
    

    当然,同一维度不同精度类型向量之间也可以互相转换:

    float2 v2 = fixed2(1, 1);
    half3 v3 = float3(v2, 0);
    float4 v4 = half4(v3, 0);
    

这里有一点要说明的是,不同维度向量类型之间的转换(同样不用担心精度问题):

  • 高维向量转换为低维向量,显式或隐式转换

    fixed4 v4 = fixed4(1, 0, 1, 0);
    half3 v3 = v4; // v4.xyz	// 这样写也可以:half3 v3 = (half3)v4
    float2 v2 = v4; // v4.xy	// 这样写也可以:float2 v2 = (float2)v4
    
  • 低维向量转换为高维向量,构造

    下面写法是错误的:

    float2 v2 = float2(1, 1);
    float3 v3 = v2;	// 错误写法,编译不通过
    

    如果用显式转换呢?

    float2 v2 = float2(1, 1);
    float3 v3 = (float3)v2; // 错误写法,编译不通过
    

    注意:在Cg/HLSL中,显式转换写法也会有同样的编译错误!

    Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎

    Cg官方手册有说明:
    Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎
    所以,低维向量转换为高维向量需要重新构造

    float2 v2 = float2(1, 1);
    half3 v3 = half3(v2, 0);
    fixed4 v4 = fixed4(v3, 0);
    

可以看出,Cg/HLSL对向量不同精度之间转换的限制是比较宽松的。

这里还有一点要说明的是,在Cg语法中,可以从一个向量中取出任意几个分量(可以重复取)按任意顺序组合形成新的向量(最多不超过四维),即 Swizzle操作符
Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02,Shader,unity,学习,游戏引擎

如图所示,有三套字符,最常用的两套是:xyzwrgba;使用哪套字符没有什么限制,二者完全等价。只不过为了代码的可读性,一般习惯于:xyzw(普通向量)、rgba(表示颜色的向量)

fixed4 v4 = fixed4(1, 0, -1, 0);
fixed3 col = v4.rgb;
fixed2 v2 = v4.www;
fixed4 v1 = v2.yyxx;

本篇完,下一篇,专题学习一下 “Cg/HLSL的内置函数” ,要开始改写“Unlit Shader”模板了…文章来源地址https://www.toymoban.com/news/detail-773979.html

到了这里,关于Unity Shader 学习(二):初识ShaderLab -- 以“Unlit Shader”模板为例 02的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 深入URP之Shader篇3: Unlit Shader分析[下]

    上篇中我们分析了Unlit shader的Properties在ShaderGUI中的处理,接下来看Sub Shader。 unlit shader以及其他URP shader包含两个SubShader,分别是针对ShaderModel4.5和2.0。由于unlit shader本身很简单,这两个SubShader几乎一样,唯一的差别是ShaderModel 4.5的SubShader会定义 #pragma multi_compile _ DOTS_INSTANCIN

    2024年02月10日
    浏览(41)
  • 深入URP之Shader篇2: 目录结构和Unlit Shader分析[上]

    我使用的Unity版本为2020.3.33f1,对应的URP和SRP Core版本为10.8.1。阅读URP源码建议把package从Library/PackageCache中拷贝到Packages目录,也就是自定义package的方式,然后推荐使用VS code打开工程,这样可以很方便的跳转代码阅读。 首先,我们看一下URP源码的目录结构,看一下Shader代码的

    2023年04月23日
    浏览(37)
  • Shader学习第七篇:几种Unity的Shader的例子

    下面是几种Shader的例子,从简单到复杂,一步一步了解 Shader 的编写机制。 顶点/片元着色器 Vertex/Fragment Shader,下面我们介绍的示例就是这个。 表面着色器 Surface Shader ,而这个底层Unity也是转成了顶点/片元着色器 固定函数着色器 Fixed Function Shader (已弃用) 在一些低端设备使

    2024年02月09日
    浏览(53)
  • shader学习摘要(九)unity阴影

    在前向渲染中,如果场景中最重要的平行光开启了阴影,unity就会为该光源计算它的阴影映射纹理(shadowmap)。这张阴影映射纹理本质上也是一张深度图,它记录了从该光源的位置出发、能看到的场景中距离它最近的表面位置(深度信息)。 我们通过修改光源的light控件下的

    2024年02月08日
    浏览(50)
  • Unity Shader学习3:透明效果

    Unity中的透明效果由透明通道控制(RGBA中的A),其值为0是完全透明,为1时完全不透明。有两种方法可以实现透明效果: 透明度测试(Alpha Test) 和 透明度混合(Alpha Blend) 。 透明度测试是指通过特定的条件(通常是Alpha通道的值是否超过某个阈值)来判断片元是否透明,只

    2024年01月19日
    浏览(62)
  • Unity Shader学习1:基础光照模型

    Phong光照模型 是描述物体的直接光照的简易模型,它认为从物体出发进入摄像机的光由四部分组成: 自发光(emissive),环境光(ambient),漫反射(diffuse),高光(specular)。 c = c e m i s s i v e + c a m b i e n t + c d i f f u s e + c s p e c u l a r c = c_{emissive} + c_{ambient} + c_{diffuse} + c_

    2024年02月03日
    浏览(49)
  • Unity Shader学习(九)物体边缘实现

    根据前面的学习,我们了解到除了可以对点的颜色进行处理,还可以对点本身进行操作,例如我们可以改变点的位置,这样就可以实现对模型渲染的操控。物体边缘效果是我们常用的一种效果,要实现物体边缘,原理也很简单。 首先我们要了解到,模型在渲染时,有正面和背

    2024年02月16日
    浏览(49)
  • Unity Shader入门精要学习——透明效果

    要么完全透明,要么完全不透明。 实现简单,实质上是一种剔除机制,通过将不满足条件(通常使用小于某个阈值来判定,一般使用clip方法)的片元舍弃的方法来达到完全透明效果。这些被舍弃的片元不会再进行任何的处理,也不会对颜色缓冲产生任何影响,其余满足条件

    2024年02月17日
    浏览(50)
  • UnityShader入门学习(三)——Unity的Shader

    Tags Tags可以写在SubShader的一开始(所有的Pass用),也可以写在Pass块的内部(该Pass用) 渲染设置 同Tags一样也可以写在Pass中或者写在Pass外面(SubShader一开始) Pass介绍 Pass里面可以定义Pass名称 Pass里面的Tags还可以有额外的设置 还有CG语言所写的代码,主要是顶点片元着色器,

    2024年02月16日
    浏览(43)
  • 学习100个Unity Shader (14) ---透明效果

    由”Queue“ 标签决定,索引号越小越早被渲染: 名称 队列索引号 Background 1000 Geometry 2000 AlphaTest 2450 Transparent 3000 Overlay 4000 某一片元的透明度小于某个阈值,即被舍弃,反之,按非透明物体处理,进行正常的深度测试和深度写入【不需要关闭深度写入】。 UnityObjectToClipPos 将顶

    2024年04月29日
    浏览(45)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包