using UnityEngine;
using UnityEngine.Rendering.Universal;
using UnityEngine.Rendering;
using System.Diagnostics.CodeAnalysis;
namespace UnityEditor.Rendering.Universal
{
///
/// Class used for Scriptable Shader keyword stripping in URP.
///
internal class ShaderScriptableStripper : IShaderVariantStripper, IShaderVariantStripperScope
{
public bool active => UniversalRenderPipeline.asset != null;
// Interfaces / Structs
// Interface for info gathered when making builds. Used for
// Scriptable Stripping and when testing the scriptable stripper.
internal interface IShaderScriptableStrippingData
{
public ShaderFeatures shaderFeatures { get; set; }
public VolumeFeatures volumeFeatures { get; set; }
public bool isGLDevice { get; set; }
public bool strip2DPasses { get; set; }
public bool stripSoftShadowQualityLevels { get; set; }
public bool stripDebugDisplayShaders { get; set; }
public bool stripScreenCoordOverrideVariants { get; set; }
public bool stripUnusedVariants { get; set; }
public bool stripUnusedPostProcessingVariants { get; set; }
public Shader shader { get; set; }
public ShaderType shaderType { get; set; }
public ShaderCompilerPlatform shaderCompilerPlatform { get; set; }
public string passName { get; set; }
public PassType passType { get; set; }
public PassIdentifier passIdentifier { get; set; }
public bool IsHDRShaderVariantValid { get; set; }
public bool IsShaderFeatureEnabled(ShaderFeatures feature);
public bool IsVolumeFeatureEnabled(VolumeFeatures feature);
public bool IsKeywordEnabled(LocalKeyword keyword);
public bool PassHasKeyword(LocalKeyword keyword);
}
// Data containing all the info needed to compare
// against the features gathered in ShaderBuildPreprocessor.cs
internal struct StrippingData : IShaderScriptableStrippingData
{
public ShaderFeatures shaderFeatures { get; set; }
public VolumeFeatures volumeFeatures { get; set; }
public bool isGLDevice {
get => variantData.shaderCompilerPlatform == ShaderCompilerPlatform.GLES20
|| variantData.shaderCompilerPlatform == ShaderCompilerPlatform.GLES3x
|| variantData.shaderCompilerPlatform == ShaderCompilerPlatform.OpenGLCore;
set{}
}
public bool stripSoftShadowQualityLevels { get; set; }
public bool strip2DPasses { get; set; }
public bool stripDebugDisplayShaders { get; set; }
public bool stripScreenCoordOverrideVariants { get; set; }
public bool stripUnusedVariants { get; set; }
public bool stripUnusedPostProcessingVariants { get; set; }
public Shader shader { get; set; }
public ShaderType shaderType { get => passData.shaderType; set{} }
public ShaderCompilerPlatform shaderCompilerPlatform { get => variantData.shaderCompilerPlatform; set {} }
public string passName { get => passData.passName; set {} }
public PassType passType { get => passData.passType; set {} }
public PassIdentifier passIdentifier { get => passData.pass; set {} }
public bool IsHDRShaderVariantValid { get => HDROutputUtils.IsShaderVariantValid(variantData.shaderKeywordSet, PlayerSettings.allowHDRDisplaySupport); set { } }
public bool IsKeywordEnabled(LocalKeyword keyword)
{
return variantData.shaderKeywordSet.IsEnabled(keyword);
}
public bool IsShaderFeatureEnabled(ShaderFeatures feature)
{
return (shaderFeatures & feature) != 0;
}
public bool IsVolumeFeatureEnabled(VolumeFeatures feature)
{
return (volumeFeatures & feature) != 0;
}
public bool PassHasKeyword(LocalKeyword keyword)
{
return ShaderUtil.PassHasKeyword(shader, passData.pass, keyword, passData.shaderType, shaderCompilerPlatform);
}
public ShaderSnippetData passData { get; set; }
public ShaderCompilerData variantData { get; set; }
}
// Shaders
Shader m_BokehDepthOfField = Shader.Find("Hidden/Universal Render Pipeline/BokehDepthOfField");
Shader m_GaussianDepthOfField = Shader.Find("Hidden/Universal Render Pipeline/GaussianDepthOfField");
Shader m_CameraMotionBlur = Shader.Find("Hidden/Universal Render Pipeline/CameraMotionBlur");
Shader m_PaniniProjection = Shader.Find("Hidden/Universal Render Pipeline/PaniniProjection");
Shader m_Bloom = Shader.Find("Hidden/Universal Render Pipeline/Bloom");
Shader m_TerrainLit = Shader.Find("Universal Render Pipeline/Terrain/Lit");
Shader m_StencilDeferred = Shader.Find("Hidden/Universal Render Pipeline/StencilDeferred");
Shader m_UberPostShader = Shader.Find("Hidden/Universal Render Pipeline/UberPost");
Shader m_HDROutputBlitShader = Shader.Find("Hidden/Universal/BlitHDROverlay");
Shader m_DataDrivenLensFlareShader = Shader.Find("Hidden/Universal Render Pipeline/LensFlareDataDriven");
// Pass names
public static readonly string kPassNameUniversal2D = "Universal2D";
public static readonly string kPassNameGBuffer = "GBuffer";
public static readonly string kPassNameForwardLit = "ForwardLit";
public static readonly string kPassNameDepthNormals = "DepthNormals";
// Keywords
LocalKeyword m_MainLightShadows;
LocalKeyword m_MainLightShadowsCascades;
LocalKeyword m_MainLightShadowsScreen;
LocalKeyword m_AdditionalLightsVertex;
LocalKeyword m_AdditionalLightsPixel;
LocalKeyword m_AdditionalLightShadows;
LocalKeyword m_ReflectionProbeBlending;
LocalKeyword m_ReflectionProbeBoxProjection;
LocalKeyword m_CastingPunctualLightShadow;
LocalKeyword m_SoftShadows;
LocalKeyword m_SoftShadowsLow;
LocalKeyword m_SoftShadowsMedium;
LocalKeyword m_SoftShadowsHigh;
LocalKeyword m_MixedLightingSubtractive;
LocalKeyword m_LightmapShadowMixing;
LocalKeyword m_ShadowsShadowMask;
LocalKeyword m_Lightmap;
LocalKeyword m_DynamicLightmap;
LocalKeyword m_DirectionalLightmap;
LocalKeyword m_AlphaTestOn;
LocalKeyword m_GbufferNormalsOct;
LocalKeyword m_UseDrawProcedural;
LocalKeyword m_ScreenSpaceOcclusion;
LocalKeyword m_UseFastSRGBLinearConversion;
LocalKeyword m_LightLayers;
LocalKeyword m_DecalLayers;
LocalKeyword m_WriteRenderingLayers;
LocalKeyword m_RenderPassEnabled;
LocalKeyword m_DebugDisplay;
LocalKeyword m_DBufferMRT1;
LocalKeyword m_DBufferMRT2;
LocalKeyword m_DBufferMRT3;
LocalKeyword m_DecalNormalBlendLow;
LocalKeyword m_DecalNormalBlendMedium;
LocalKeyword m_DecalNormalBlendHigh;
LocalKeyword m_ForwardPlus;
LocalKeyword m_FoveatedRenderingNonUniformRaster;
LocalKeyword m_EditorVisualization;
LocalKeyword m_LODFadeCrossFade;
LocalKeyword m_LightCookies;
LocalKeyword m_LocalDetailMulx2;
LocalKeyword m_LocalDetailScaled;
LocalKeyword m_LocalClearCoat;
LocalKeyword m_LocalClearCoatMap;
LocalKeyword m_LensDistortion;
LocalKeyword m_ChromaticAberration;
LocalKeyword m_BloomLQ;
LocalKeyword m_BloomHQ;
LocalKeyword m_BloomLQDirt;
LocalKeyword m_BloomHQDirt;
LocalKeyword m_HdrGrading;
LocalKeyword m_ToneMapACES;
LocalKeyword m_ToneMapNeutral;
LocalKeyword m_FilmGrain;
LocalKeyword m_ScreenCoordOverride;
LocalKeyword m_EasuRcasAndHDRInput;
LocalKeyword m_SHPerVertex;
LocalKeyword m_SHMixed;
private LocalKeyword TryGetLocalKeyword(Shader shader, string name)
{
return shader.keywordSpace.FindKeyword(name);
}
private void InitializeLocalShaderKeywords([DisallowNull] Shader shader)
{
m_MainLightShadows = TryGetLocalKeyword(shader, ShaderKeywordStrings.MainLightShadows);
m_MainLightShadowsCascades = TryGetLocalKeyword(shader, ShaderKeywordStrings.MainLightShadowCascades);
m_MainLightShadowsScreen = TryGetLocalKeyword(shader, ShaderKeywordStrings.MainLightShadowScreen);
m_AdditionalLightsVertex = TryGetLocalKeyword(shader, ShaderKeywordStrings.AdditionalLightsVertex);
m_AdditionalLightsPixel = TryGetLocalKeyword(shader, ShaderKeywordStrings.AdditionalLightsPixel);
m_AdditionalLightShadows = TryGetLocalKeyword(shader, ShaderKeywordStrings.AdditionalLightShadows);
m_ReflectionProbeBlending = TryGetLocalKeyword(shader, ShaderKeywordStrings.ReflectionProbeBlending);
m_ReflectionProbeBoxProjection = TryGetLocalKeyword(shader, ShaderKeywordStrings.ReflectionProbeBoxProjection);
m_CastingPunctualLightShadow = TryGetLocalKeyword(shader, ShaderKeywordStrings.CastingPunctualLightShadow);
m_SoftShadows = TryGetLocalKeyword(shader, ShaderKeywordStrings.SoftShadows);
m_SoftShadowsLow = TryGetLocalKeyword(shader, ShaderKeywordStrings.SoftShadowsLow);
m_SoftShadowsMedium = TryGetLocalKeyword(shader, ShaderKeywordStrings.SoftShadowsMedium);
m_SoftShadowsHigh = TryGetLocalKeyword(shader, ShaderKeywordStrings.SoftShadowsHigh);
m_MixedLightingSubtractive = TryGetLocalKeyword(shader, ShaderKeywordStrings.MixedLightingSubtractive);
m_LightmapShadowMixing = TryGetLocalKeyword(shader, ShaderKeywordStrings.LightmapShadowMixing);
m_ShadowsShadowMask = TryGetLocalKeyword(shader, ShaderKeywordStrings.ShadowsShadowMask);
m_Lightmap = TryGetLocalKeyword(shader, ShaderKeywordStrings.LIGHTMAP_ON);
m_DynamicLightmap = TryGetLocalKeyword(shader, ShaderKeywordStrings.DYNAMICLIGHTMAP_ON);
m_DirectionalLightmap = TryGetLocalKeyword(shader, ShaderKeywordStrings.DIRLIGHTMAP_COMBINED);
m_AlphaTestOn = TryGetLocalKeyword(shader, ShaderKeywordStrings._ALPHATEST_ON);
m_GbufferNormalsOct = TryGetLocalKeyword(shader, ShaderKeywordStrings._GBUFFER_NORMALS_OCT);
m_UseDrawProcedural = TryGetLocalKeyword(shader, ShaderKeywordStrings.UseDrawProcedural);
m_ScreenSpaceOcclusion = TryGetLocalKeyword(shader, ShaderKeywordStrings.ScreenSpaceOcclusion);
m_UseFastSRGBLinearConversion = TryGetLocalKeyword(shader, ShaderKeywordStrings.UseFastSRGBLinearConversion);
m_LightLayers = TryGetLocalKeyword(shader, ShaderKeywordStrings.LightLayers);
m_DecalLayers = TryGetLocalKeyword(shader, ShaderKeywordStrings.DecalLayers);
m_WriteRenderingLayers = TryGetLocalKeyword(shader, ShaderKeywordStrings.WriteRenderingLayers);
m_RenderPassEnabled = TryGetLocalKeyword(shader, ShaderKeywordStrings.RenderPassEnabled);
m_DebugDisplay = TryGetLocalKeyword(shader, ShaderKeywordStrings.DEBUG_DISPLAY);
m_DBufferMRT1 = TryGetLocalKeyword(shader, ShaderKeywordStrings.DBufferMRT1);
m_DBufferMRT2 = TryGetLocalKeyword(shader, ShaderKeywordStrings.DBufferMRT2);
m_DBufferMRT3 = TryGetLocalKeyword(shader, ShaderKeywordStrings.DBufferMRT3);
m_DecalNormalBlendLow = TryGetLocalKeyword(shader, ShaderKeywordStrings.DecalNormalBlendLow);
m_DecalNormalBlendMedium = TryGetLocalKeyword(shader, ShaderKeywordStrings.DecalNormalBlendMedium);
m_DecalNormalBlendHigh = TryGetLocalKeyword(shader, ShaderKeywordStrings.DecalNormalBlendHigh);
m_ForwardPlus = TryGetLocalKeyword(shader, ShaderKeywordStrings.ForwardPlus);
m_FoveatedRenderingNonUniformRaster = TryGetLocalKeyword(shader, ShaderKeywordStrings.FoveatedRenderingNonUniformRaster);
m_EditorVisualization = TryGetLocalKeyword(shader, ShaderKeywordStrings.EDITOR_VISUALIZATION);
m_LODFadeCrossFade = TryGetLocalKeyword(shader, ShaderKeywordStrings.LOD_FADE_CROSSFADE);
m_LightCookies = TryGetLocalKeyword(shader, ShaderKeywordStrings.LightCookies);
m_ScreenCoordOverride = TryGetLocalKeyword(shader, ShaderKeywordStrings.SCREEN_COORD_OVERRIDE);
m_LocalDetailMulx2 = TryGetLocalKeyword(shader, ShaderKeywordStrings._DETAIL_MULX2);
m_LocalDetailScaled = TryGetLocalKeyword(shader, ShaderKeywordStrings._DETAIL_SCALED);
m_LocalClearCoat = TryGetLocalKeyword(shader, ShaderKeywordStrings._CLEARCOAT);
m_LocalClearCoatMap = TryGetLocalKeyword(shader, ShaderKeywordStrings._CLEARCOATMAP);
m_EasuRcasAndHDRInput = TryGetLocalKeyword(shader, ShaderKeywordStrings.EasuRcasAndHDRInput);
// Post processing
m_LensDistortion = TryGetLocalKeyword(shader, ShaderKeywordStrings.Distortion);
m_ChromaticAberration = TryGetLocalKeyword(shader, ShaderKeywordStrings.ChromaticAberration);
m_BloomLQ = TryGetLocalKeyword(shader, ShaderKeywordStrings.BloomLQ);
m_BloomHQ = TryGetLocalKeyword(shader, ShaderKeywordStrings.BloomHQ);
m_BloomLQDirt = TryGetLocalKeyword(shader, ShaderKeywordStrings.BloomLQDirt);
m_BloomHQDirt = TryGetLocalKeyword(shader, ShaderKeywordStrings.BloomHQDirt);
m_HdrGrading = TryGetLocalKeyword(shader, ShaderKeywordStrings.HDRGrading);
m_ToneMapACES = TryGetLocalKeyword(shader, ShaderKeywordStrings.TonemapACES);
m_ToneMapNeutral = TryGetLocalKeyword(shader, ShaderKeywordStrings.TonemapNeutral);
m_FilmGrain = TryGetLocalKeyword(shader, ShaderKeywordStrings.FilmGrain);
m_SHPerVertex = TryGetLocalKeyword(shader, ShaderKeywordStrings.EVALUATE_SH_VERTEX);
m_SHMixed = TryGetLocalKeyword(shader, ShaderKeywordStrings.EVALUATE_SH_MIXED);
}
/*********************************************************
Volume Features
*********************************************************/
internal bool StripVolumeFeatures_UberPostShader(ref IShaderScriptableStrippingData strippingData)
{
if (strippingData.shader != m_UberPostShader)
return false;
ShaderStripTool stripTool = new ShaderStripTool(strippingData.volumeFeatures, ref strippingData);
if (stripTool.StripMultiCompileKeepOffVariant(m_LensDistortion, VolumeFeatures.LensDistortion))
return true;
if (stripTool.StripMultiCompileKeepOffVariant(m_ChromaticAberration, VolumeFeatures.ChromaticAberration))
return true;
if (stripTool.StripMultiCompileKeepOffVariant(m_BloomLQ, VolumeFeatures.Bloom))
return true;
if (stripTool.StripMultiCompileKeepOffVariant(m_BloomHQ, VolumeFeatures.Bloom))
return true;
if (stripTool.StripMultiCompileKeepOffVariant(m_BloomLQDirt, VolumeFeatures.Bloom))
return true;
if (stripTool.StripMultiCompileKeepOffVariant(m_BloomHQDirt, VolumeFeatures.Bloom))
return true;
if (stripTool.StripMultiCompileKeepOffVariant(m_ToneMapACES, VolumeFeatures.ToneMapping))
return true;
if (stripTool.StripMultiCompileKeepOffVariant(m_ToneMapNeutral, VolumeFeatures.ToneMapping))
return true;
if (stripTool.StripMultiCompileKeepOffVariant(m_FilmGrain, VolumeFeatures.FilmGrain))
return true;
return false;
}
internal bool StripVolumeFeatures_BokehDepthOfFieldShader(ref IShaderScriptableStrippingData strippingData)
{
if (strippingData.shader != m_BokehDepthOfField)
return false;
return !strippingData.IsVolumeFeatureEnabled(VolumeFeatures.DepthOfField);
}
internal bool StripVolumeFeatures_GaussianDepthOfFieldShader(ref IShaderScriptableStrippingData strippingData)
{
if (strippingData.shader != m_GaussianDepthOfField)
return false;
return !strippingData.IsVolumeFeatureEnabled(VolumeFeatures.DepthOfField);
}
internal bool StripVolumeFeatures_CameraMotionBlurShader(ref IShaderScriptableStrippingData strippingData)
{
if (strippingData.shader != m_CameraMotionBlur)
return false;
return !strippingData.IsVolumeFeatureEnabled(VolumeFeatures.CameraMotionBlur);
}
internal bool StripVolumeFeatures_PaniniProjectionShader(ref IShaderScriptableStrippingData strippingData)
{
if (strippingData.shader != m_PaniniProjection)
return false;
return !strippingData.IsVolumeFeatureEnabled(VolumeFeatures.PaniniProjection);
}
internal bool StripVolumeFeatures_BloomShader(ref IShaderScriptableStrippingData strippingData)
{
if (strippingData.shader != m_Bloom)
return false;
return !strippingData.IsVolumeFeatureEnabled(VolumeFeatures.Bloom);
}
internal bool StripVolumeFeatures(VolumeFeatures features, ref IShaderScriptableStrippingData strippingData)
{
if (StripVolumeFeatures_UberPostShader(ref strippingData))
return true;
if (StripVolumeFeatures_BokehDepthOfFieldShader(ref strippingData))
return true;
if (StripVolumeFeatures_GaussianDepthOfFieldShader(ref strippingData))
return true;
if (StripVolumeFeatures_CameraMotionBlurShader(ref strippingData))
return true;
if (StripVolumeFeatures_PaniniProjectionShader(ref strippingData))
return true;
if (StripVolumeFeatures_BloomShader(ref strippingData))
return true;
return false;
}
/*********************************************************
Unused Variants
*********************************************************/
internal bool StripUnusedFeatures_DebugDisplay(ref IShaderScriptableStrippingData strippingData)
{
return strippingData.stripDebugDisplayShaders && strippingData.IsKeywordEnabled(m_DebugDisplay);
}
internal bool StripUnusedFeatures_ScreenCoordOverride(ref IShaderScriptableStrippingData strippingData)
{
return strippingData.stripScreenCoordOverrideVariants && strippingData.IsKeywordEnabled(m_ScreenCoordOverride);
}
internal bool StripUnusedFeatures_PunctualLightShadows(ref IShaderScriptableStrippingData strippingData)
{
// Shadow caster punctual light strip
if (strippingData.passType == PassType.ShadowCaster && strippingData.PassHasKeyword(m_CastingPunctualLightShadow))
{
bool mainLightShadowsDisabled =
!strippingData.IsShaderFeatureEnabled(ShaderFeatures.MainLightShadows) &&
!strippingData.IsShaderFeatureEnabled(ShaderFeatures.MainLightShadowsCascade) &&
!strippingData.IsShaderFeatureEnabled(ShaderFeatures.ScreenSpaceShadows);
if (mainLightShadowsDisabled && !strippingData.IsKeywordEnabled(m_CastingPunctualLightShadow))
return true;
if (!strippingData.IsShaderFeatureEnabled(ShaderFeatures.AdditionalLightShadows) && strippingData.IsKeywordEnabled(m_CastingPunctualLightShadow))
return true;
}
return false;
}
internal bool StripUnusedFeatures_FoveatedRendering(ref IShaderScriptableStrippingData strippingData)
{
// Strip Foveated Rendering variants on all platforms (except PS5 and Metal)
// TODO: add a way to communicate this requirement from the xr plugin directly
#if ENABLE_VR && ENABLE_XR_MODULE
if (strippingData.shaderCompilerPlatform != ShaderCompilerPlatform.PS5NGGC && strippingData.shaderCompilerPlatform != ShaderCompilerPlatform.Metal)
#endif
{
if (strippingData.IsKeywordEnabled(m_FoveatedRenderingNonUniformRaster))
return true;
}
return false;
}
internal bool StripUnusedFeatures_DeferredRendering(ref IShaderScriptableStrippingData strippingData)
{
// TODO: Test against lightMode tag instead.
if (strippingData.passName == kPassNameGBuffer)
{
if (!strippingData.IsShaderFeatureEnabled(ShaderFeatures.DeferredShading))
return true;
}
return false;
}
internal bool StripUnusedFeatures_MainLightShadows(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
// strip main light shadows, cascade and screen variants
if (strippingData.IsShaderFeatureEnabled(ShaderFeatures.ShadowsKeepOffVariants))
{
if (stripTool.StripMultiCompileKeepOffVariant(
m_MainLightShadows, ShaderFeatures.MainLightShadows,
m_MainLightShadowsCascades, ShaderFeatures.MainLightShadowsCascade,
m_MainLightShadowsScreen, ShaderFeatures.ScreenSpaceShadows))
return true;
}
else
{
if (stripTool.StripMultiCompile(
m_MainLightShadows, ShaderFeatures.MainLightShadows,
m_MainLightShadowsCascades, ShaderFeatures.MainLightShadowsCascade,
m_MainLightShadowsScreen, ShaderFeatures.ScreenSpaceShadows))
return true;
}
return false;
}
internal bool StripUnusedFeatures_AdditionalLightShadows(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
// No additional light shadows
if (strippingData.IsShaderFeatureEnabled(ShaderFeatures.ShadowsKeepOffVariants))
{
if (stripTool.StripMultiCompileKeepOffVariant(m_AdditionalLightShadows, ShaderFeatures.AdditionalLightShadows))
return true;
}
else if (stripTool.StripMultiCompile(m_AdditionalLightShadows, ShaderFeatures.AdditionalLightShadows))
return true;
return false;
}
internal bool StripUnusedFeatures_MixedLighting(ref IShaderScriptableStrippingData strippingData)
{
// Strip here only if mixed lighting is disabled
if (!strippingData.IsShaderFeatureEnabled(ShaderFeatures.MixedLighting))
{
if (strippingData.IsKeywordEnabled(m_MixedLightingSubtractive))
return true;
// No need to check here if actually used by scenes as this taken care by builtin stripper
if (strippingData.IsKeywordEnabled(m_LightmapShadowMixing) || strippingData.IsKeywordEnabled(m_ShadowsShadowMask))
return true;
}
return false;
}
internal bool StripUnusedFeatures_SoftShadows(ref ShaderStripTool stripTool)
{
// TODO: Strip off variants once we have global soft shadows option for forcing instead as support
return stripTool.StripMultiCompileKeepOffVariant(m_SoftShadows, ShaderFeatures.SoftShadows);
}
internal bool StripUnusedFeatures_SoftShadowsQualityLevels(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
var forcedStrip = strippingData.stripSoftShadowQualityLevels &&
(strippingData.IsShaderFeatureEnabled(ShaderFeatures.SoftShadowsLow) ||
strippingData.IsShaderFeatureEnabled(ShaderFeatures.SoftShadowsMedium) ||
strippingData.IsShaderFeatureEnabled(ShaderFeatures.SoftShadowsHigh));
return forcedStrip || (stripTool.StripMultiCompileKeepOffVariant(
m_SoftShadowsLow, ShaderFeatures.SoftShadowsLow,
m_SoftShadowsMedium, ShaderFeatures.SoftShadowsMedium,
m_SoftShadowsHigh, ShaderFeatures.SoftShadowsHigh));
}
internal bool StripUnusedFeatures_HDRGrading(ref ShaderStripTool stripTool)
{
return stripTool.StripMultiCompileKeepOffVariant(m_HdrGrading, ShaderFeatures.HdrGrading);
}
internal bool StripUnusedFeatures_UseFastSRGBLinearConversion(ref ShaderStripTool stripTool)
{
return stripTool.StripMultiCompile(m_UseFastSRGBLinearConversion, ShaderFeatures.UseFastSRGBLinearConversion);
}
internal bool StripUnusedFeatures_LightLayers(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
if (strippingData.shaderCompilerPlatform == ShaderCompilerPlatform.GLES20)
{
// GLES2 does not support bitwise operations.
if (strippingData.IsKeywordEnabled(m_LightLayers))
return true;
}
else
{
if (stripTool.StripMultiCompile(m_LightLayers, ShaderFeatures.LightLayers))
return true;
}
return false;
}
internal bool StripUnusedFeatures_RenderPassEnabled(ref ShaderStripTool stripTool)
{
return stripTool.StripMultiCompileKeepOffVariant(m_RenderPassEnabled, ShaderFeatures.RenderPassEnabled);
}
internal bool StripUnusedFeatures_ReflectionProbes(ref ShaderStripTool stripTool)
{
// Reflection probes
if (stripTool.StripMultiCompile(m_ReflectionProbeBlending, ShaderFeatures.ReflectionProbeBlending))
return true;
if (stripTool.StripMultiCompile(m_ReflectionProbeBoxProjection, ShaderFeatures.ReflectionProbeBoxProjection))
return true;
return false;
}
internal bool StripUnusedFeatures_ForwardPlus(ref ShaderStripTool stripTool)
{
return stripTool.StripMultiCompile(m_ForwardPlus, ShaderFeatures.ForwardPlus);
}
internal bool StripUnusedFeatures_SHAuto(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
// SH auto mode is per-vertex or per-pixel. Strip unused variants
if (strippingData.IsShaderFeatureEnabled(ShaderFeatures.AutoSHMode))
{
if (strippingData.IsShaderFeatureEnabled(ShaderFeatures.AutoSHModePerVertex))
{
// Strip Mixed variant and Off(perPixel) variant
if (stripTool.StripMultiCompile(m_SHMixed, ShaderFeatures.ExplicitSHMode, m_SHPerVertex, ShaderFeatures.AutoSHModePerVertex))
return true;
}
else
{
// Strip Mixed variant and PerVertex variant
if (stripTool.StripMultiCompileKeepOffVariant(m_SHPerVertex, ShaderFeatures.AutoSHModePerVertex, m_SHMixed, ShaderFeatures.ExplicitSHMode))
return true;
}
}
return false;
}
internal bool StripUnusedFeatures_AdditionalLights(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
// Forward Plus doesn't use Vertex or the Pixel Light variants.
// It enables the Pixel keyword through a define.
if (strippingData.IsShaderFeatureEnabled(ShaderFeatures.ForwardPlus))
{
if (strippingData.IsShaderFeatureEnabled(ShaderFeatures.AdditionalLightsVertex))
return true;
if (strippingData.IsShaderFeatureEnabled(ShaderFeatures.AdditionalLightsPixel))
return true;
}
// Additional light are shaded per-vertex or per-pixel.
if (strippingData.IsShaderFeatureEnabled(ShaderFeatures.AdditionalLightsKeepOffVariants))
{
if (stripTool.StripMultiCompileKeepOffVariant(m_AdditionalLightsVertex, ShaderFeatures.AdditionalLightsVertex, m_AdditionalLightsPixel, ShaderFeatures.AdditionalLightsPixel))
return true;
}
else
{
if (stripTool.StripMultiCompile(m_AdditionalLightsVertex, ShaderFeatures.AdditionalLightsVertex, m_AdditionalLightsPixel, ShaderFeatures.AdditionalLightsPixel))
return true;
}
return false;
}
internal bool StripUnusedFeatures_ScreenSpaceOcclusion(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
// Screen Space Occlusion
if (strippingData.IsShaderFeatureEnabled(ShaderFeatures.ScreenSpaceOcclusionAfterOpaque))
{
// SSAO after opaque setting requires off variants
if (stripTool.StripMultiCompileKeepOffVariant(m_ScreenSpaceOcclusion, ShaderFeatures.ScreenSpaceOcclusion))
return true;
}
else
{
if (stripTool.StripMultiCompile(m_ScreenSpaceOcclusion, ShaderFeatures.ScreenSpaceOcclusion))
return true;
}
return false;
}
internal bool StripUnusedFeatures_DecalsDbuffer(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
// DBuffer
if (strippingData.isGLDevice)
{
// Decal DBuffer is not supported on gl
if (strippingData.IsKeywordEnabled(m_DBufferMRT1) ||
strippingData.IsKeywordEnabled(m_DBufferMRT2) ||
strippingData.IsKeywordEnabled(m_DBufferMRT3))
return true;
}
else
{
if (stripTool.StripMultiCompile(
m_DBufferMRT1, ShaderFeatures.DBufferMRT1,
m_DBufferMRT2, ShaderFeatures.DBufferMRT2,
m_DBufferMRT3, ShaderFeatures.DBufferMRT3))
return true;
}
return false;
}
internal bool StripUnusedFeatures_DecalsNormalBlend(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
// Decal Normal Blend
if (stripTool.StripMultiCompile(
m_DecalNormalBlendLow, ShaderFeatures.DecalNormalBlendLow,
m_DecalNormalBlendMedium, ShaderFeatures.DecalNormalBlendMedium,
m_DecalNormalBlendHigh, ShaderFeatures.DecalNormalBlendHigh))
return true;
return false;
}
internal bool StripUnusedFeatures_DecalLayers(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
// Rendering layers are not supported on gl
if (strippingData.isGLDevice)
{
if (strippingData.IsKeywordEnabled(m_DecalLayers))
return true;
}
else if (stripTool.StripMultiCompile(m_DecalLayers, ShaderFeatures.DecalLayers))
return true;
return false;
}
internal bool StripUnusedFeatures_WriteRenderingLayers(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
// Rendering layers are not supported on gl
if (strippingData.isGLDevice)
{
if (strippingData.IsKeywordEnabled(m_WriteRenderingLayers))
return true;
}
else
{
if (strippingData.passName == kPassNameDepthNormals)
{
if (stripTool.StripMultiCompile(m_WriteRenderingLayers, ShaderFeatures.DepthNormalPassRenderingLayers))
return true;
}
if (strippingData.passName == kPassNameForwardLit)
{
if (stripTool.StripMultiCompile(m_WriteRenderingLayers, ShaderFeatures.OpaqueWriteRenderingLayers))
return true;
}
if (strippingData.passName == kPassNameGBuffer)
{
if (stripTool.StripMultiCompile(m_WriteRenderingLayers, ShaderFeatures.GBufferWriteRenderingLayers))
return true;
}
}
return false;
}
internal bool StripUnusedFeatures_AccurateGbufferNormals(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
return stripTool.StripMultiCompile(m_GbufferNormalsOct, ShaderFeatures.AccurateGbufferNormals);
}
internal bool StripUnusedFeatures_LightCookies(ref IShaderScriptableStrippingData strippingData, ref ShaderStripTool stripTool)
{
return stripTool.StripMultiCompileKeepOffVariant(m_LightCookies, ShaderFeatures.LightCookies);
}
internal bool StripUnusedFeatures_DataDrivenLensFlare(ref IShaderScriptableStrippingData strippingData)
{
// If this is not the right shader, then skip
if (strippingData.shader != m_DataDrivenLensFlareShader)
return false;
return !strippingData.IsShaderFeatureEnabled(ShaderFeatures.DataDrivenLensFlare);
}
internal bool StripUnusedFeatures(ref IShaderScriptableStrippingData strippingData)
{
if (StripUnusedFeatures_DebugDisplay(ref strippingData))
return true;
if (StripUnusedFeatures_ScreenCoordOverride(ref strippingData))
return true;
if (StripUnusedFeatures_MixedLighting(ref strippingData))
return true;
if (StripUnusedFeatures_PunctualLightShadows(ref strippingData))
return true;
if (StripUnusedFeatures_FoveatedRendering(ref strippingData))
return true;
if (StripUnusedFeatures_DeferredRendering(ref strippingData))
return true;
if (StripUnusedFeatures_DataDrivenLensFlare(ref strippingData))
return true;
ShaderStripTool stripTool = new ShaderStripTool(strippingData.shaderFeatures, ref strippingData);
if (StripUnusedFeatures_MainLightShadows(ref strippingData, ref stripTool))
return true;
if (StripUnusedFeatures_AdditionalLightShadows(ref strippingData, ref stripTool))
return true;
if (StripUnusedFeatures_SoftShadows(ref stripTool))
return true;
if (StripUnusedFeatures_SoftShadowsQualityLevels(ref strippingData, ref stripTool))
return true;
if (StripUnusedFeatures_HDRGrading(ref stripTool))
return true;
if (StripUnusedFeatures_UseFastSRGBLinearConversion(ref stripTool))
return true;
if (StripUnusedFeatures_LightLayers(ref strippingData, ref stripTool))
return true;
if (StripUnusedFeatures_RenderPassEnabled(ref stripTool))
return true;
if (StripUnusedFeatures_ReflectionProbes(ref stripTool))
return true;
if (StripUnusedFeatures_ForwardPlus(ref stripTool))
return true;
if (StripUnusedFeatures_AdditionalLights(ref strippingData, ref stripTool))
return true;
if (StripUnusedFeatures_SHAuto(ref strippingData, ref stripTool))
return true;
if (StripUnusedFeatures_ScreenSpaceOcclusion(ref strippingData, ref stripTool))
return true;
if (StripUnusedFeatures_DecalsDbuffer(ref strippingData, ref stripTool))
return true;
if (StripUnusedFeatures_DecalsNormalBlend(ref strippingData, ref stripTool))
return true;
if (StripUnusedFeatures_DecalLayers(ref strippingData, ref stripTool))
return true;
if (StripUnusedFeatures_WriteRenderingLayers(ref strippingData, ref stripTool))
return true;
if (StripUnusedFeatures_AccurateGbufferNormals(ref strippingData, ref stripTool))
return true;
if (stripTool.StripMultiCompileKeepOffVariant(m_LODFadeCrossFade, ShaderFeatures.LODCrossFade))
return true;
if (StripUnusedFeatures_LightCookies(ref strippingData, ref stripTool))
return true;
return false;
}
/*********************************************************
Unsupported Variants
*********************************************************/
internal bool StripUnsupportedVariants_DirectionalLightmap(ref IShaderScriptableStrippingData strippingData)
{
// We can strip variants that have directional lightmap enabled but not static nor dynamic lightmap.
if (strippingData.IsKeywordEnabled(m_DirectionalLightmap)
&& !(strippingData.IsKeywordEnabled(m_Lightmap) || strippingData.IsKeywordEnabled(m_DynamicLightmap)))
return true;
return false;
}
internal bool StripUnsupportedVariants_EditorVisualization(ref IShaderScriptableStrippingData strippingData)
{
// Editor visualization is only used in scene view debug modes.
if (strippingData.IsKeywordEnabled(m_EditorVisualization))
return true;
return false;
}
internal bool StripUnsupportedVariants_GLES2(ref IShaderScriptableStrippingData strippingData)
{
// As GLES2 has low amount of registers, we strip:
if (strippingData.shaderCompilerPlatform == ShaderCompilerPlatform.GLES20)
{
// VertexID - as GLES2 does not support VertexID that is required for full screen draw procedural pass;
if (strippingData.IsKeywordEnabled(m_UseDrawProcedural))
return true;
// Cascade shadows
if (strippingData.IsKeywordEnabled(m_MainLightShadowsCascades))
return true;
// Screen space shadows
if (strippingData.IsKeywordEnabled(m_MainLightShadowsScreen))
return true;
// Detail
if (strippingData.IsKeywordEnabled(m_LocalDetailMulx2) || strippingData.IsKeywordEnabled(m_LocalDetailScaled))
return true;
// Clear Coat
if (strippingData.IsKeywordEnabled(m_LocalClearCoat) || strippingData.IsKeywordEnabled(m_LocalClearCoatMap))
return true;
}
return false;
}
internal bool StripUnsupportedVariants(ref IShaderScriptableStrippingData strippingData)
{
// We can strip variants that have directional lightmap enabled but not static nor dynamic lightmap.
if (StripUnsupportedVariants_DirectionalLightmap(ref strippingData))
return true;
if (StripUnsupportedVariants_EditorVisualization(ref strippingData))
return true;
if (StripUnsupportedVariants_GLES2(ref strippingData))
return true;
return false;
}
/*********************************************************
Invalid Variants
*********************************************************/
internal bool StripInvalidVariants_HDR(ref IShaderScriptableStrippingData strippingData)
{
// We do not need to strip out HDR output variants if HDR display is enabled.
if (PlayerSettings.allowHDRDisplaySupport)
return false;
// Shared keywords between URP and HDRP.
if (!strippingData.IsHDRShaderVariantValid)
return true;
// HDR output shader variants specific to URP.
if (strippingData.IsKeywordEnabled(m_EasuRcasAndHDRInput))
return true;
return false;
}
internal bool StripInvalidVariants_TerrainHoles(ref IShaderScriptableStrippingData strippingData)
{
if (strippingData.shader == m_TerrainLit)
if (!strippingData.IsShaderFeatureEnabled(ShaderFeatures.TerrainHoles) && strippingData.IsKeywordEnabled(m_AlphaTestOn))
return true;
return false;
}
internal bool StripInvalidVariants_Shadows(ref IShaderScriptableStrippingData strippingData)
{
// Strip Additional Shadow variants if it's not set to PerPixel and not F+/Deferred
bool areAdditionalShadowsEnabled = strippingData.IsKeywordEnabled(m_AdditionalLightShadows);
bool hasShadowsOff = strippingData.IsShaderFeatureEnabled(ShaderFeatures.ShadowsKeepOffVariants);
if (hasShadowsOff && areAdditionalShadowsEnabled)
{
bool isPerPixel = strippingData.IsKeywordEnabled(m_AdditionalLightsPixel);
bool isForwardPlus = strippingData.IsKeywordEnabled(m_ForwardPlus);
bool isDeferred = strippingData.IsShaderFeatureEnabled(ShaderFeatures.DeferredShading);
if (!isPerPixel && !isForwardPlus && !isDeferred)
return true;
}
// Strip Soft Shadows if shadows are disabled for both Main and Additional Lights...
bool isMainShadowNoCascades = strippingData.IsKeywordEnabled(m_MainLightShadows);
bool isMainShadowCascades = strippingData.IsKeywordEnabled(m_MainLightShadowsCascades);
bool isMainShadowScreen = strippingData.IsKeywordEnabled(m_MainLightShadowsScreen);
bool isMainShadow = isMainShadowNoCascades || isMainShadowCascades || isMainShadowScreen;
bool isShadowVariant = isMainShadow || areAdditionalShadowsEnabled;
if (!isShadowVariant && (strippingData.IsKeywordEnabled(m_SoftShadows) ||
strippingData.IsKeywordEnabled(m_SoftShadowsLow) ||
strippingData.IsKeywordEnabled(m_SoftShadowsMedium)
|| strippingData.IsKeywordEnabled(m_SoftShadowsHigh)))
return true;
return false;
}
internal bool StripInvalidVariants(ref IShaderScriptableStrippingData strippingData)
{
if (StripInvalidVariants_HDR(ref strippingData))
return true;
if (StripInvalidVariants_TerrainHoles(ref strippingData))
return true;
if (StripInvalidVariants_Shadows(ref strippingData))
return true;
return false;
}
/*********************************************************
Unused Passes
*********************************************************/
internal bool StripUnusedPass_2D(ref IShaderScriptableStrippingData strippingData)
{
// Strip 2D Passes if there are no 2D renderers...
if (strippingData.passName == kPassNameUniversal2D && strippingData.strip2DPasses)
return true;
return false;
}
internal bool StripUnusedPass_Meta(ref IShaderScriptableStrippingData strippingData)
{
// Meta pass is needed in the player for Enlighten Precomputed Realtime GI albedo and emission.
if (strippingData.passType == PassType.Meta)
{
if (SupportedRenderingFeatures.active.enlighten == false
|| ((int)SupportedRenderingFeatures.active.lightmapBakeTypes | (int)LightmapBakeType.Realtime) == 0)
return true;
}
return false;
}
internal bool StripUnusedPass_ShadowCaster(ref IShaderScriptableStrippingData strippingData)
{
if (strippingData.passType == PassType.ShadowCaster)
{
if ( !strippingData.IsShaderFeatureEnabled(ShaderFeatures.MainLightShadows)
&& !strippingData.IsShaderFeatureEnabled(ShaderFeatures.AdditionalLightShadows))
return true;
}
return false;
}
internal bool StripUnusedPass_Decals(ref IShaderScriptableStrippingData strippingData)
{
// Do not strip GL passes as there are only screen space forward
if (strippingData.isGLDevice)
return false;
// DBuffer
if (strippingData.passName == DecalShaderPassNames.DBufferMesh
|| strippingData.passName == DecalShaderPassNames.DBufferProjector
|| strippingData.passName == DecalShaderPassNames.DecalMeshForwardEmissive
|| strippingData.passName == DecalShaderPassNames.DecalProjectorForwardEmissive)
if (!strippingData.IsShaderFeatureEnabled(ShaderFeatures.DBufferMRT1) && !strippingData.IsShaderFeatureEnabled(ShaderFeatures.DBufferMRT2) && !strippingData.IsShaderFeatureEnabled(ShaderFeatures.DBufferMRT3))
return true;
// Decal Screen Space
if (strippingData.passName == DecalShaderPassNames.DecalScreenSpaceMesh || strippingData.passName == DecalShaderPassNames.DecalScreenSpaceProjector)
if (!strippingData.IsShaderFeatureEnabled(ShaderFeatures.DecalScreenSpace))
return true;
// Decal GBuffer
if (strippingData.passName == DecalShaderPassNames.DecalGBufferMesh || strippingData.passName == DecalShaderPassNames.DecalGBufferProjector)
if (!strippingData.IsShaderFeatureEnabled(ShaderFeatures.DecalGBuffer))
return true;
return false;
}
internal bool StripUnusedPass(ref IShaderScriptableStrippingData strippingData)
{
if (StripUnusedPass_2D(ref strippingData))
return true;
if (StripUnusedPass_Meta(ref strippingData))
return true;
if (StripUnusedPass_ShadowCaster(ref strippingData))
return true;
if (StripUnusedPass_Decals(ref strippingData))
return true;
return false;
}
/*********************************************************
Unused Shaders
*********************************************************/
internal bool StripUnusedShaders_Deferred(ref IShaderScriptableStrippingData strippingData)
{
if (!strippingData.stripUnusedVariants)
return false;
// Remove DeferredStencil if Deferred Rendering is not used
if (strippingData.shader == m_StencilDeferred)
if (!strippingData.IsShaderFeatureEnabled(ShaderFeatures.DeferredShading))
return true;
return false;
}
internal bool StripUnusedShaders_HDROutput(ref IShaderScriptableStrippingData strippingData)
{
if (!strippingData.stripUnusedVariants)
return false;
// Remove BlitHDROverlay if HDR output is not used
if (strippingData.shader == m_HDROutputBlitShader)
if (!PlayerSettings.allowHDRDisplaySupport)
return true;
return false;
}
internal bool StripUnusedShaders(ref IShaderScriptableStrippingData strippingData)
{
if (!strippingData.stripUnusedVariants)
return false;
// Remove DeferredStencil if Deferred Rendering is not used
if (StripUnusedShaders_Deferred(ref strippingData))
return true;
// Remove BlitHDROverlay if HDR output is not used
if (StripUnusedShaders_HDROutput(ref strippingData))
return true;
return false;
}
/*********************************************************
Main Callbacks
*********************************************************/
public bool CanRemoveVariant([DisallowNull] Shader shader, ShaderSnippetData passData, ShaderCompilerData variantData)
{
IShaderScriptableStrippingData strippingData = new StrippingData()
{
volumeFeatures = ShaderBuildPreprocessor.volumeFeatures,
stripSoftShadowQualityLevels = !ShaderBuildPreprocessor.s_UseSoftShadowQualityLevelKeywords,
strip2DPasses = ShaderBuildPreprocessor.s_Strip2DPasses,
stripDebugDisplayShaders = ShaderBuildPreprocessor.s_StripDebugDisplayShaders,
stripScreenCoordOverrideVariants = ShaderBuildPreprocessor.s_StripScreenCoordOverrideVariants,
stripUnusedVariants = ShaderBuildPreprocessor.s_StripUnusedVariants,
stripUnusedPostProcessingVariants = ShaderBuildPreprocessor.s_StripUnusedPostProcessingVariants,
shader = shader,
passData = passData,
variantData = variantData
};
// All feature sets need to have this variant unused to be stripped out.
bool removeInput = true;
for (var index = 0; index < ShaderBuildPreprocessor.supportedFeaturesList.Count; index++)
{
strippingData.shaderFeatures = ShaderBuildPreprocessor.supportedFeaturesList[index];
if (StripUnusedShaders(ref strippingData))
continue;
if (StripUnusedPass(ref strippingData))
continue;
if (StripInvalidVariants(ref strippingData))
continue;
if (StripUnsupportedVariants(ref strippingData))
continue;
if (StripUnusedFeatures(ref strippingData))
continue;
removeInput = false;
break;
}
// Check PostProcessing variants...
if (!removeInput && strippingData.stripUnusedPostProcessingVariants)
if (StripVolumeFeatures(ShaderBuildPreprocessor.volumeFeatures, ref strippingData))
removeInput = true;
return removeInput;
}
public void BeforeShaderStripping(Shader shader)
{
if (shader != null)
InitializeLocalShaderKeywords(shader);
}
public void AfterShaderStripping(Shader shader)
{
}
}
}