using System; using UnityEngine.Rendering; #if HDRP_PRESENT using UnityEngine.Rendering.HighDefinition; #elif URP_PRESENT using UnityEngine.Rendering.Universal; #endif #if HDRP_PRESENT || URP_PRESENT namespace UnityEngine.Perception.GroundTruth { /// /// Custom Pass which will apply a lens distortion (per the respective volume override in URP or HDRP, or /// through a custom override directly through the pass) to an incoming mask / texture. The purpose of this /// is to allow the same lens distortion being applied to the RGB image in the perception camera to be applied /// to the respective ground truths generated. /// class LensDistortionCrossPipelinePass : GroundTruthCrossPipelinePass { const string k_ShaderName = "Perception/LensDistortion"; // NOTICE: Serialize the shader so that the shader asset is included in player builds when the SemanticSegmentationPass is used. // Currently commented out and shaders moved to Resources folder due to serialization crashes when it is enabled. // See https://fogbugz.unity3d.com/f/cases/1187378/ // [SerializeField] Shader m_LensDistortionShader; Material m_LensDistortionMaterial; bool m_Initialized; static readonly int k_DistortionParams1Id = Shader.PropertyToID("_Distortion_Params1"); static readonly int k_DistortionParams2Id = Shader.PropertyToID("_Distortion_Params2"); LensDistortion m_LensDistortion; internal float? lensDistortionOverride = null; // Largely for testing, but could be useful otherwise RenderTexture m_TargetTexture; RenderTexture m_DistortedTexture; public LensDistortionCrossPipelinePass(Camera targetCamera, RenderTexture targetTexture) : base(targetCamera) { m_TargetTexture = targetTexture; } public override void Setup() { base.Setup(); m_LensDistortionShader = Shader.Find(k_ShaderName); var shaderVariantCollection = new ShaderVariantCollection(); if (shaderVariantCollection != null) shaderVariantCollection.Add(new ShaderVariantCollection.ShaderVariant(m_LensDistortionShader, PassType.ScriptableRenderPipeline)); m_LensDistortionMaterial = new Material(m_LensDistortionShader); if (shaderVariantCollection != null) shaderVariantCollection.WarmUp(); // Set up a new texture if (m_DistortedTexture == null || m_DistortedTexture.width != Screen.width || m_DistortedTexture.height != Screen.height) { if (m_DistortedTexture != null) m_DistortedTexture.Release(); m_DistortedTexture = new RenderTexture(Screen.width, Screen.height, 0, RenderTextureFormat.ARGBFloat, RenderTextureReadWrite.Linear); m_DistortedTexture.enableRandomWrite = true; m_DistortedTexture.filterMode = FilterMode.Point; m_DistortedTexture.Create(); } // Grab the lens distortion #if HDRP_PRESENT // Grab the Lens Distortion from Perception Camera stack var hdCamera = HDCamera.GetOrCreate(targetCamera); var stack = hdCamera.volumeStack; m_LensDistortion = stack.GetComponent(); #elif URP_PRESENT var stack = VolumeManager.instance.stack; m_LensDistortion = stack.GetComponent(); #endif m_Initialized = true; } protected override void ExecutePass(ScriptableRenderContext renderContext, CommandBuffer cmd, Camera camera, CullingResults cullingResult) { if (m_Initialized == false) return; if (SetLensDistortionShaderParameters() == false) return; // Blit mayhem cmd.Blit(m_TargetTexture, m_DistortedTexture, m_LensDistortionMaterial); cmd.Blit(m_DistortedTexture, m_TargetTexture); renderContext.ExecuteCommandBuffer(cmd); cmd.Clear(); } public bool SetLensDistortionShaderParameters() { if (m_Initialized == false) return false; // This code is lifted from the SetupLensDistortion() function in // https://github.com/Unity-Technologies/Graphics/blob/257b08bba6c11de0f894e42e811124247a522d3c/com.unity.render-pipelines.universal/Runtime/Passes/PostProcessPass.cs // This is in UnityEngine.Rendering.Universal.Internal.PostProcessPass::SetupLensDistortion so it's // unclear how to re-use this code var intensity = 0.5f; var scale = 1.0f; var center = new Vector2(0.0f, 0.0f); var mult = new Vector2(1.0f, 1.0f); #if HDRP_PRESENT if(m_LensDistortion == null) return false; #elif URP_PRESENT if (targetCamera == null) return false; var additionalCameraData = targetCamera.GetUniversalAdditionalCameraData(); if (additionalCameraData.renderPostProcessing == false && lensDistortionOverride.HasValue == false) return false; if (m_LensDistortion.active == false) return false; #else return false; #endif if (lensDistortionOverride.HasValue) { intensity = lensDistortionOverride.Value; } else if (m_LensDistortion != null) { // This is a bit finicky for URP - since Lens Distortion comes off the VolumeManager stack as active // even if post processing is not enabled. An intensity of 0.0f is untenable, so the below checks // ensures post processing hasn't been enabled but Lens Distortion actually overriden if (m_LensDistortion.intensity.value != 0.0f) { intensity = m_LensDistortion.intensity.value; center = m_LensDistortion.center.value * 2f - Vector2.one; mult.x = Mathf.Max(m_LensDistortion.xMultiplier.value, 1e-4f); mult.y = Mathf.Max(m_LensDistortion.yMultiplier.value, 1e-4f); scale = 1.0f / m_LensDistortion.scale.value; } else { return false; } } var amount = 1.6f * Mathf.Max(Mathf.Abs(intensity * 100.0f), 1.0f); var theta = Mathf.Deg2Rad * Mathf.Min(160f, amount); var sigma = 2.0f * Mathf.Tan(theta * 0.5f); var p1 = new Vector4( center.x, center.y, mult.x, mult.y ); var p2 = new Vector4( intensity >= 0f ? theta : 1f / theta, sigma, scale, intensity * 100.0f ); // Set Shader Constants m_LensDistortionMaterial.SetVector(k_DistortionParams1Id, p1); m_LensDistortionMaterial.SetVector(k_DistortionParams2Id, p2); return true; } public override void SetupMaterialProperties(MaterialPropertyBlock mpb, Renderer renderer, Labeling labeling, uint instanceId) { SetLensDistortionShaderParameters(); } } } #endif // ! HDRP_PRESENT || URP_PRESENT