Shaders are small programs that execute on the GPU, and loading them can take some time. Each individual GPU program typically does not take much time to load, but shaders often have a lot of “variants” internally.
着色器是在GPU上执行的小程序,加载它们需要一些时间。每个独立的GPU程序一般不需要多少时间加载,但是着色器常常有很多内在的“变体”。
For example, the Standard shader, if fully compiled, ends up being many thousands of slightly different GPU programs. This creates two potential problems:
比如,标准着色器如果完全编译,会生成几千个不同的小GPU程序。这会造成两个潜在的问题:
Large numbers of these shader variants increase game build time, and game data size.
Loading large numbers of shader variants during game is slow and takes up memory.
这些着色器的大量变体会增加游戏打包时间,使游戏包变大。
在游戏中加载大量的着色器变体很慢,并且消耗内存。
Shader build time stripping 着色器生成时去除
While building the game, Unity can detect that some of the internal shader variants are not used by the game, and skip them from build data. Build-time stripping is done for:
生成游戏包时,Unity可以检测到某些内在着色器变体没有被使用,然后忽略它们。生成时去除可用于:
#pragma shader_feature
- . If none of the used materials use a particular variant, then it is not included into the build. Seeinternal shader variants documentation. Out of built-in shaders, the Standard shader uses this.
- Shader variants to handle Fog and Lightmapping modes not used by any of the scenes are not included into the game data. See Graphics Settings if you want to override this behavior.
- 个别着色器特性,使用#pragma着色器特性的着色器。如果一个变体没有被任何用到的材质使用,那么生成时就不把它打进去。参考内在着色器文档,内置着色器中的标准着色器使用这种方式。
- 没被任何场景使用的处理雾和光照贴图模式的着色器变体,也不会打包到游戏数据中。如果你想改变这个行为,请参考文档图形设置。
Combination of the above often substantially cuts down on shader data size. For example, a fully compiled Standard shader would take several hundred megabytes, but in typical projects it often ends up taking just a couple megabytes (and is often compressed further by the application packaging process).
以上方式的结合可以大大减少着色器数据的大小。比如,完全编译的标准着色器有几百兆,但是在一般的工程中常常只有几兆(而且app打包时会被进一步压缩)。
Default Unity shader loading behavior 默认Unity着色器加载行为
Under all default settings, Unity loads the shaderlab Shader object into memory, but does not create the internal shader variants until they are actually needed.
默认设置下,Unity加载shaderlab着色器对象到内存中,但是内部着色器变体直到被真正用到的时候才创建。
This means that shader variants that are included into the game build can still potentially be used, but there’s no memory or load time cost paid until they are needed. For example, shaders always include a variant to handle point lights with shadows, but if you never end up using a point light with shadows in your game, then there’s no point in loading this particular variant.
这意味着打进游戏包中的着色器变体潜在的可能被用到,但是在用到之前没有占用内存或者消耗时间加载。比如,着色器常常包含一个处理点光源阴影的变体,但是如果你的游戏里从来没有用到点光源阴影,那么加载这个变体就没必要。
ShaderVariantCollection
然而,这种默认行为的一个弊端是一些着色器变体第一次被用到的时候可能会出现卡顿(hiccup)——因为需要加载一个新的GPU程序代码到图形驱动。在游戏中这是不能接受的,所以Unity有一个叫ShaderVariantCollection的资源来帮助解决这个问题。
Shader Variant Collections 着色器变体群
ShaderVariantCollection is an asset that is basically a list of Shaders, and for each of them, a list of Pass types and shader keyword combinations to load.
ShaderVariantCollection基本上是一个着色器列表资源,每次加载一个由通道类型和着色器关键字组合的列表。
着色器变体群检视器
To help with creating these assets based on actually used shaders and their variants, the editor can track which shaders and their variants are actually used. In Graphics Settings, there is a button to create a new ShaderVariantCollection out of currently tracked shaders, or to clear the currently tracked shader list.
为了帮助创建这些基于被真正用到的着色器和它们变体的资源,编辑器可以追踪哪些着色器和它们的变体被真正用到了。当前追踪着色器的图形设置检视器上,有一个按钮可以创建一个新的ShaderVariantCollection,也可以清除当前追踪的着色器列表。
从编辑器使用的着色器创建ShaderVariantCollection
Once you have some ShaderVariantCollection assets, you can set for these variants to be automatically preloaded while loading the game (under Preloaded Shaders list in Graphics Settings), or you can preload an individual shader variant collection from a script. See ShaderVariantCollection scripting class.
一旦有了一些ShaderVariantCollection资源,你可以设置这些着色器变体在加载游戏时自动预加载(在图形设置的预加载着色器列表下),或者你也可以用脚本预加载一个独立的着色器变体。参考ShaderVariantCollection脚本类。
See Also 另外可参考
- Optimizing Graphics Performance. 中文翻译:Unity3D - 图形性能优化。
- Graphics Settings. 图形设置。
- Shaders reference. 着色器参考。