返回介绍

特定平台的渲染差异

发布于 2021-06-20 00:15:07 字数 3563 浏览 914 评论 0 收藏 0

Unity 在不同的平台上运行,而且在某些情况下存在运行差异。大部分时间里 Unity 都向您隐藏差异,但有时您仍然可以撞见它们。

渲染纹理坐标

垂直纹理坐标的惯例与 Direct3D、OpenGL 和 OpenGL ES 不同:

  • 在 Direct3D 中,坐标顶部为零,向下增加。
  • 在 OpenGL 和 OpenGL ES 中,坐标底部为零,向上增加。

大多数时候这都无关紧要,除了渲染到渲染纹理时。在此情况下,Unity 渲染到 Direct3D 上的纹理时,在内部翻转渲染,以便平台之间的惯例匹配。

当使用图像效果和抗锯齿 (Anti-Aliasing) 时,不会出现这种情况。在此情况下,Unity 渲染到屏幕以获得抗锯齿选项,然后将渲染“解析”为 RenderTexture 以便用图像效果 (Image Effect) 进一步处理。产生的图像效果的源纹理在 Direct3D 上没有翻转(不同于其他所有的渲染纹理 (Render Texture))。

如果图像效果 (Image Effect) 简单(一次处理一个纹理),这无关紧要,因为 Graphics.Blit 会进行处理。

然而,如果在图像效果 (Image Effect) 中同时处理一个以上的 RenderTexture,它们很可能会在不同的垂直方向出现(仅在类似 Direct3D 的平台上,并且仅在使用抗锯齿选项时)。您需要在顶点着色器中手动“翻转”屏幕纹理,像这样:

//在 D3D 上使用 AA 时,主纹理与场景深度纹理
//将在不同垂直方向出现。
//因此在这种情况下翻转纹理采样(主纹理
//纹理元件大小将为 Y 轴负值)。
#if UNITY_UV_STARTS_AT_TOP
if (_MainTex_TexelSize.y < 0)
        uv.y = 1-uv.y;
#endif

请在着色器替换样例工程中查看边缘检测 (Edge Detection) 场景以了解这样的示例。此处的边缘检测同时使用屏幕纹理和相机的深度 + 法线纹理

Alpha 测试 (AlphaTest) 和可编程着色器

某些平台,尤其是手机 (OpenGL ES 2.0) 和 Direct3D 11,没有固定函数 alpha 测试功能。使用可编程着色器时,建议改为使用像素着色器的 Cg/HLSL clip() 函数。

Direct3D 11 着色器编译器对语法更挑剔

Direct3D 9 和 OpenGL 使用 NVIDIA 的 Cg 来编译着色器,但 Direct3D 11(和 Xbox 360)使用 Microsoft 的 HLSL 着色器编译器。HLSL 编译器对着色器各种细微错误更挑剔。例如,它不接受没有适当初始化的函数输出值。

您会遇到此类情况的最常见位置是:

  • 有一个“输出”参数的表面着色器顶点修改器。确保像这样初始化输出:
      void vert (inout appdata_full v, out Input o) 
      {
        UNITY_INITIALIZE_OUTPUT(Input,o);
        // ...
      }
  • 部分初始化的值,例如返回 float4 的函数,但代码仅设置它的 .xyz 值。确保设置所有的值,或如果只需要这些值,就将其改为 float3。
  • 在顶点着色器中使用 tex2D 由于顶点着色器中不存在 UV 导数,因此 tex2D 无效;您需要抽取一个详细的纹理层次,例如使用 tex2Dlod (tex, float4(uv,0,0))。您还需要添加 #pragma glsl,使其在 OpenGL 上运行。

DirectX 11 HLSL 语法和表面着色器 (Surface Shader)

当前表面着色器编译管线的有些部分无法理解特定于 DX11 的 HLSL 语法。如果是诸如 StructuredBuffers、RWTextures 和其他非 DX9 语法的 HLSL 功能,则您必须将其纳入只有 DX11 的预处理宏:

    #ifdef SHADER_API_D3D11
    // 特定于 DX11 的编码,如
    StructuredBuffer<float4> myColors;
    RWTexture2D<float4> myRandomWriteTexture;
    #endif

使用带 OpenGL ES 2.0 的 OpenGL 着色语言 (GLSL) 着色器

OpenGL ES 2.0 只为 OpenGL 着色语言 (GLSL) 提供 有限的本地支持,例如 OpenGL ES 2.0 层不为着色器提供 内置参数。

Unity 执行内置参数的方式与 OpenGL 完全相同,但缺失以下内置参数:

  • gl_ClipVertex
  • gl_SecondaryColor
  • gl_DepthRange
  • gl_LightSourceParameters 结构的 halfVector 属性
  • gl_FrontFacing
  • gl_FrontLightModelProduct
  • gl_BackLightModelProduct
  • gl_BackMaterial
  • gl_Point
  • gl_PointSize
  • gl_ClipPlane
  • gl_EyePlaneR, gl_EyePlaneS, gl_EyePlaneT, gl_EyePlaneQ
  • gl_ObjectPlaneR, gl_ObjectPlaneS, gl_ObjectPlaneT, gl_ObjectPlaneQ
  • gl_Fog

iPad2、MSAA 和 alpha 混合几何结构

当 MSAA 已激活并且用 RGBA colorMask 绘制 alpha 混合几何结构时,Apple 驱动程序有产生假象的错误。为避免出现假象,遇到此配置时我们强制使用 RGBA colorMask,尽管它会将内置 Glow FX 渲染成无法使用(因为它需要 DST_ALPHA 获取强度值)。同时,如果您自己编写程序,请更新着色器(请参阅Pass Docs)中的“渲染设置 (Render Setup) -> ColorMask”。)

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文