前言
本书的定位与适合的读者
本书不是一本3D图形编程入门书,也不是一本Unity 3D着色器编程入门书。本书的定位是“Unity 3D着色器代码分析教程+Unity 3D着色器编程参考手册”。因此,本书的目标读者应具有一定的图形编程知识:有一定的Unity 3D开发经验;学习过基础的Direct 3D/OpenGL图形编程,如知道顶点、纹理和着色器的概念与作用;了解矩阵乘法、四元数和坐标系变换等数学概念。
本书是为下面3类读者撰写的。你在阅读本书之前,阅读过Unity 3D自带的或者第三方编写的着色器代码,依样画葫芦地修改过,对大部分代码知道其实现的功能,但又有很多细节不明白;或者你已经能写一些Unity 3D ShaderLab代码了,但还不清楚Unity 3D提供的内置着色器库可以提供多少现有的工具代码;抑或你因工作需要,要把Unity 3D提供的着色器进行精简改造以适应自己的项目,对千头万绪的细节感到困难。本书有助于解决这些读者的痛点。
本书所剖析的着色器代码版本
本书所剖析的Unity 3D内置着色器代码版本是2017.2.0f3,读者可以从Unity 3D官网下载这些着色器代码。这些代码以名为builtin_shaders-2017.2.0f3.zip的压缩包的形式提供,解压缩后,内有4个目录和1个license.txt文件,介绍如下。
目录CGIncludes存放了37个扩展名为cginc的文件,两个扩展名为glslinc的文件。这些文件就是Unity 3D提供的内置着色器的头文件。本书将重点剖析表0-1中头文件的实现。
表0-1 头文件
目录DefaultResources存放了Unity 3D引擎内置的简单着色器。
目录DefaultResourcesExtra提供了大量渲染效果的着色器实现,Mobile子目录下的shader文件就是移动平台下的漫反射效果、粒子系统、法线贴图和光照图效果的实现。本书将详细剖析该目录下的Standard.shader文件,即标准着色器的实现。
目录Editor中唯一的文件是StandardShaderUI.cs。该段代码是当材质文件使用了标准着色器时,材质对应的inspector界面的实现。
文件license.txt用于说明Unity 3D开发公司对这些着色器代码的版权。
本书内容和建议阅读方式
既然本书的定位是“Unity 3D着色器代码分析教程+Unity 3D着色器编程参考手册”,那么读者需要按照一定的阅读顺序才能达到最佳的阅读效果。对于初中级读者,可先从第1章开始精读。第1章对当前主流的渲染流水线进行阐述,讲述顶点处理阶段、光栅化阶段、片元处理与输出合并阶段这三大处理阶段的实现。这三大处理阶段是主流渲染流水线都必须实现的阶段。
1.1节概述了渲染流水线,讲述了主流渲染流水线的各个阶段,以及各个阶段的操作。1.2节介绍顶点处理阶段。首先详细讲述顶点的组织方式、坐标系的确定方式,然后对把顶点从模型空间变换至世界空间、从世界空间变换至观察空间、从观察空间变换到裁剪空间所用到的各个变换矩阵进行详细说明,最后分析Unity 3D中这些矩阵的封装代码,让读者在懂得使用这些矩阵的同时能知其然且知其所以然。1.3节介绍光栅化阶段,对其中的各个子阶段进行详细的数学推导说明。光栅化阶段是由硬件实现不可编程的,对它进行详细的数学说明也是为了让读者深入了解其原理,在开发工作中能够从底层去理解渲染流水线的机制。今后读者如果工作中需要用到其他的3D引擎,能融会贯通且更快地上手。1.4节介绍片元处理与输出合并阶段。其中,片元处理子阶段就是片元着色器的内容;输出合并子阶段也是由硬件实现不可编程的,但流水线提供了若干功能函数以对它进行控制。该节重点讲述输出合并中的深度值操作和Alpha值操作,以及Unity 3D为这两个操作所提供的控制函数。
精读完第1章后,可接着通读第2章。第2章主要从物理学的角度阐述图形渲染中本质的问题,即光的能量传递与分布问题。其中,2.1节和2.2节是学习基于物理渲染的前置知识,里面所阐述的各个物理量和它们的数学关系是阅读第10章与第11章的基础;2.3节讲述计算机如何对颜色进行数学建模,而该节则是理解2.4节中颜色空间的基础;2.4节重点讲述计算机图形学中关于“伽马校正”的内容,阅读完该节后,相信读者在工作中碰到“画面颜色总是不对且偏暗”的问题时,能理解它产生的缘由并能解决之。
第3章对Unity 3D特有的外观着色器进行分析,阐述外观着色器和传统的顶点/片元着色器之间的关系。Unity 3D的内置着色器代码中大量使用了着色器多样体,因此同一套着色器能够被编译到各个不同的硬件平台,了解着色器多样体的原理是剖析Unity 3D内置着色器代码所必需的。因此,3.4节详细分析这些着色器多样体的原理和使用方法。如果读者已经熟悉该章内容,可以跳过它。
因为Unity 3D是一个跨平台引擎,所以在Unity 3D内置着色器代码中要时刻考虑通用性问题。尤其是在开发手机游戏时,开发环境通常是Windows/Mac平台,而运行环境多是Android/iOS平台。因此,一套着色器代码起码要支持开发和运行两种不同的环境。不同的平台下使用的着色器语言也有所不同。虽然Unity 3D着色器推荐以Cg语言作为前端的开发语言,但是Unity 3D会在后台将Cg语言代码编译为目标平台的最佳运行语言的字节码。例如,在Windows平台上最佳运行语言是HLSL,而Android/iOS平台上则是OpenGL ES。Unity 3D着色器语言提供了一系列消除平台和开发语言差异性的机制。3.5节和3.6节会阐述这些机制。
Unity 3D提供了大量的通用工具函数和一些由引擎底层在运行期赋值的着色器变量。无论是Unity 3D内置着色器或者第三方编写的着色器,都大量使用了这些预定义的通用工具函数和着色器变量。这些通用工具函数集中在UnityCG.cginc文件中,着色器变量则定义在UnityShader Variables.cginc文件中。第4章重点剖析这两个文件的实现,尤其讲述UnityCG.cginc文件中工具函数的实现原理。开发者在编写自己的着色器时,如果碰到一些要实现的功能,不妨先查阅该章,看看Unity 3D引擎是否已经提供了已有的实现。同时,因为Unity 3D内置着色器自身也大量使用了这两个文件中的内容,所以本书其他章节中也大量交叉引用了该章内容,读者在阅读剖析着色器代码的章节时,也应经常查看该章内容。
Unity 3D内置着色器大量使用了GPU多例化技术。Unity 3D在UnityInstancing.cginc文件中提供了使用GPU多例化技术要用到的宏。第5章讲述GPU多例化技术的实现原理,并剖析UnityInstancing.cginc文件中Unity 3D引擎对它的封装实现。
Unity 3D有两种渲染方式:一种是前向渲染,另一种是延迟渲染。Unity 3D提供的标准着色器文件Standard.shader中有这两种渲染方式的实现。第6章讲述前向渲染和延迟渲染的基本原理,以及Unity 3D对延迟渲染的一些实现细节。
图形渲染的两大主题是光照和阴影的计算。Unity 3D引擎除了支持光源对物体的照明计算(即直接照明)之外,还支持物体之间的光照效果,即间接照明。两者统称为全局照明。第7~9章讲述Unity 3D的全局光照和阴影计算原理。其中,7.7节从数学原理出发,重点阐述球谐光照原理和Unity 3D对它的封装实现。Unity 3D提供了大量完成光照计算和阴影计算的工具函数与宏,无论是第三方着色器还是引擎内置着色器都会大量使用到它们。这些函数与宏分别在UnityShadow Library.cginc文件和AutoLight.cginc文件中定义。第8章和第9章详细剖析这两个文件的实现。如果开发者在自己编写的着色器中需要实现某功能,或者在阅读第三方着色器代码时碰到这些文件中定义的函数和宏,可以查阅这两章。
近年来,能够产生更为逼真效果的基于物理的光照模型开始广泛应用在各大3D引擎中。第10章分析若干简单的光照模型,并从数学和物理原理上分析基于Cook-Torrance模型的光照模型的实现。Standard.shader文件则是基于物理光照模型的着色器的实现。第11章详细分析Standard.shader文件的实现,以及实现Standard.shader时要用到的分布在UnityStandardInput.cginc、UnityStandard Utils.cginc等文件中的工具宏和函数。
第12章是着色器编程实战案例。该章将使用Unity 3D着色器,在不使用任何纹理贴图的方式下,利用带符号距离场技术,通过片元着色器绘制一个名为“星夜之海”的动态场景。
版式约定
代码段的格式如下。
//文本块最左侧有一条竖线,表明这是一个代码块 //代码中的行状注释用“//”符号开头 //块状注释用“/**/”符号包含。代码中的Cg语言关键字加粗显示 float3a = float3(1.0,1.0,1.0); float3b = float3(2.0,1.0,1.0); float3c = dot(a,b);
原始代码中,原本是在一行中定义的,但由于纸面篇幅所限,会做一些换行处理。为了保持代码的严谨性,原始代码中的一些宏定义原本是没有转行声明符“\”的,在本书中会加上,例如:
#if defined(UNITY_COMPILER_HLSLCC) && !defined(SHADER_API_GLCORE) #define UNITY_DECLARE_TEX3D_FLOAT(tex) Texture3D_floattex;\ SamplerStatesampler##tex
上面的代码中,第二行末尾的“\”符号在原始代码中是不存在的。为了排版需要,把原来在一行的语句段分写成两行。在Cg语言中定义一个宏时,换行时要加上“\”符号。
本书引用的原始代码中,出于排版和剖析说明的原因,在保证不改变代码逻辑的前提下,会对原始代码做版面上的调整。例如,在原始文件中原本是书写成一行的代码,可能会变成多行书写;原来代码没有注释的地方,可能会在书中加上注释;原本有英文注释的地方,可能会换上中文注释。但为了便于读者对照着原始文件阅读本书,本书中引用的原始代码段所在原始文件中的名字、所在目录,以及在原始文件中的起始行和结束行都会在前面加上注释说明,如下所示。
// 所在文件:UnityGlobalIllumination.cginc // 所在目录:CGIncludes // 从原文件第44行开始,至第49行结束 inline voidResetUnityLight(outUnityLight outLight) { outLight.color = half3(0, 0, 0); outLight.dir = half3(0, 1, 0); //任意设置一个光线输出方向,不为空即可 outLight.ndotl = 0; //数据项未使用 }
提交勘误
作者和编辑尽最大努力来确保书中内容的准确性,但难免会存在疏漏。欢迎您将发现的问题反馈给我们,帮助我们提升图书的质量。
当您发现错误时,请登录异步社区,按书名搜索,进入本书页面,单击“提交勘误”,输入勘误信息,单击“提交”按钮即可。本书的作者和编辑会对您提交的勘误进行审核,确认并接受后,您将获赠异步社区的100积分。积分可用于在异步社区兑换优惠券、样书或奖品。
扫码关注本书
扫描下方二维码,您将会在异步社区微信服务号中看到本书信息及相关的服务提示。
与我们联系
我们的联系邮箱是contact@epubit.com.cn。
如果您对本书有任何疑问或建议,请您发邮件给我们,并请在邮件标题中注明本书书名,以便我们更高效地做出反馈。
如果您有兴趣出版图书、录制教学视频,或者参与图书翻译、技术审校等工作,可以发邮件给我们,邮箱为zhangtao@ptpress.com.cn。
如果您是学校、培训机构或企业,想批量购买本书或异步社区出版的其他图书,也可以发邮件给我们。
如果您在网上发现有针对异步社区出品图书的各种形式的盗版行为,包括对图书全部或部分内容的非授权传播,请您将怀疑有侵权行为的链接发邮件给我们。您的这一举动是对作者权益的保护,也是我们持续为您提供有价值的内容的动力之源。
关于异步社区和异步图书
“异步社区”是人民邮电出版社旗下IT专业图书社区,致力于出版精品IT技术图书和相关学习产品,为作译者提供优质出版服务。异步社区创办于2015年8月,提供大量精品IT技术图书和电子书,以及高品质技术文章和视频课程。更多详情请访问异步社区官网https://www.epubit.com。
“异步图书”是由异步社区编辑团队策划出版的精品IT专业图书的品牌,依托于人民邮电出版社近30年的计算机图书出版积累和专业编辑团队,相关图书在封面上印有异步图书的LOGO。异步图书的出版领域包括软件开发、大数据、AI、测试、前端、网络技术等。
异步社区
微信服务号