For a couple days now I have a problem and I can’t find a clue. I want to create custom post-processing effect that is going to work with LWRP and I want to avoid using Post Processing Stack V2 as a base for my effect. I’m looking for something like OnRenderImage in the legacy rendering pipeline. I already tried adding command buffer to the camera but this doesn’t work on LWRP too… So my question is there any other way to apply fullscreen image effect using LWRP?
How can you do Custom Post-Processing effects in LWRP/URP? Can someone give clear instructions on the steps involved. eg. I want world space normals and motion vectors, as I want to implement a special motion blur.
Idk how you can get texture with space normals and motion vectors but it should be available somewhere I guess. If you want to use post-processing stack as a foundation for your custom image effect here is instruction how to write custom effect → Writing Custom Effects · Unity-Technologies/PostProcessing Wiki · GitHub
Thanks for this! I think I’m somewhere near but I still can’t push my image effect to screen buffer. I’ve written custom render pass which applies image effect to screen texture but I can’t push the final image to the screen buffer and I don’t have clue why. Frame Debugger shows that my effect is in fact rendered.
Here is code of custom render pass:
public class CustomRenderPassFeature : ScriptableRendererFeature
{
class CustomRenderPass : ScriptableRenderPass
{
private int screenCopyID;
// This method is called before executing the render pass.
// It can be used to configure render targets and their clear state. Also to create temporary render target textures.
// When empty this render pass will render to the active camera render target.
// You should never call CommandBuffer.SetRenderTarget. Instead call <c>ConfigureTarget</c> and <c>ConfigureClear</c>.
// The render pipeline will ensure target setup and clearing happens in an performance manner.
public override void Configure(CommandBuffer cmd, RenderTextureDescriptor cameraTextureDescriptor)
{
screenCopyID = Shader.PropertyToID("_ScreenCopyTexture");
cmd.GetTemporaryRT(screenCopyID, cameraTextureDescriptor);
}
// Here you can implement the rendering logic.
// Use <c>ScriptableRenderContext</c> to issue drawing commands or execute command buffers
// https://docs.unity3d.com/ScriptReference/Rendering.ScriptableRenderContext.html
// You don't have to call ScriptableRenderContext.submit, the render pipeline will call it at specific points in the pipeline.
public override void Execute(ScriptableRenderContext context, ref RenderingData renderingData)
{
CommandBuffer buffer = CommandBufferPool.Get("MOPP");
buffer.name = "VIgnette";
buffer.Blit(BuiltinRenderTextureType.CurrentActive, screenCopyID);
Material VignetteMaterial = GetMaterial(VignetteShader);
buffer.Blit(screenCopyID, BuiltinRenderTextureType.CameraTarget, VignetteMaterial, 0);
context.ExecuteCommandBuffer(buffer);
CommandBufferPool.Release(buffer);
}
/// Cleanup any allocated resources that were created during the execution of this render pass.
public override void FrameCleanup(CommandBuffer cmd)
{
cmd.ReleaseTemporaryRT(screenCopyID);
}
private const string VignetteShader = "Hidden/VignetteShader";
private Dictionary<string, Material> Materials = new Dictionary<string, Material>();
public Material GetMaterial(string shaderName)
{
Material material;
if (Materials.TryGetValue(shaderName, out material))
{
return material; //a
}
else
{
Shader shader = Shader.Find(shaderName);
if (shader == null)
{
Debug.LogError("Shader not found (" + shaderName + "), check if missed shader is in Shaders folder if not reimport this package. If this problem occurs only in build try to add all shaders in Shaders folder to Always Included Shaders (Project Settings -> Graphics -> Always Included Shaders)");
}
Material NewMaterial = new Material(shader);
NewMaterial.hideFlags = HideFlags.HideAndDontSave;
Materials.Add(shaderName, NewMaterial);
return NewMaterial;
}
}
}
CustomRenderPass m_ScriptablePass;
public override void Create()
{
m_ScriptablePass = new CustomRenderPass();
// Configures where the render pass should be injected.
m_ScriptablePass.renderPassEvent = RenderPassEvent.AfterRendering;
}
// Here you can inject one or multiple render passes in the renderer.
// This method is called when setting up the renderer once per-camera.
public override void AddRenderPasses(ScriptableRenderer renderer, ref RenderingData renderingData)
{
renderer.EnqueuePass(m_ScriptablePass);
Debug.Log("Render pass added!");
}
}
Also here is evidence from frame debug that my effect is rendered but not to screen buffer (I need to use something different than BuiltinRenderTextureType.CameraTarget as my image effect target but I don’t know what)
Blitting to renderer.cameraColorTarget does make the whole thing work quite nicely. I found that the two renderPassEvent flags that work are RenderPassEvent.AfterRendering (only if postproc AA is disabled) and RenderPassEvent.BeforeRenderingPostProcessing.
It’s obviously not the most efficient solution due to the extra blit, but it’s definitely nice to be able to at least test some custom effects.
Interesting in my case it works even with flags like RenderPassEvent.AfterRenderingOpaques without problem maybe because for now I’am just testing this with simple vignette shader so I’m not using depth buffers etc. Anyway, I found another problem with this rendering method… performance on my quite old now phone (xiaomi redmi note 4) simple vignette effect take like 30ms! This is waaaay too much but I don’t have a clue why it happens.
I tried different things to optimize this:
-creating commandbuffer and filling it with commands only once at startup (not worth it, almost 0 impact on performance)
-using temporary render textures with different parameters than cameraTextureDescriptor have, this actually gave a significant boost in performance (especially changing depthBits to 0 because I don’t use depth anyway)
After this optimizations I’m now achieving rendering times like 30ms for just vignette :((for comparsion vignette effect from post processing stack build into lwrp take 2-3 ms). So it may be some bug connected with my device or I’m just missing something important in my code.
any progress on this? I would like to start using Universal Renderpipeline but all the stuff i read makes it feel very limited. Especially with custom post processing in mind.
Hey, as far as I know, render feature is still the only option for custom post-processing. About limits of urp/lwrp around a month ago I wanted to port my volumetric lights to support urp but I had to stop because I didn’t find any way to get shadow map texture which is necessary for this effect, oh and build into urp post processing stack lacks SSAO. Tbh I don’t know why urp/lwrp in the current state is called production-ready and not alpha or something xD.
yeah that’s weird to me too but maybe you just need more people using it to boost the development… so they trying to get more people to use it?
Or maybe just a contract with oculus to get it done as soon as quest is out and now they have to officially call it “ready” to keep the agreement idk. ^^
… whatever i’m still happy that they try to get the best performance out of the engine and don’t hesitate too long with introducing larger changes to reach that goal.
With “Render Feature” you mean the steps you described above? Which gave you 30ms overhead? Or did i miss something?
Yep render feature but after couple times trying it out I am not sure about why back then I was getting such big overhead now I still do the same thing copy screen to texture → apply some shaders etc → copy this to destination texture and everything works with reasonable rendering time so maybe it was problem with using some specific lwrp version or something.