Multiple cameras, image effects and the deferred rendering path

I’m working with the Deferred rendering path and I’m having issues with image effects and my multi-camera setup.

Here’s a simple scene setup :

So, to sum up, Camera_0 renders the Default layer and Camera_1 renders the Test layer on top of the first camera. So far so good.

But if I add an image effect to Camera_0 (let say the default “Sepia” effect), it simply does nothing. At all. If I put it on Camera_1 it works but that’s not what I want, I only want to apply it to Camera_0.

It works as soon as I switch to the Forward rendering path. Unity 4.x had the same issue with the legacy Deferred path but there was a workaround. Of course, this doesn’t work in Unity 5 anymore.

Has anyone found a solution to this issue ? Is it a bug or “by design” ? I can post the test scene if anyone’s interested but it’s easy to reproduce.

1 Like

Yes I have had this issue as well even in 4.x cycle

Same issue here. Seems like a bug to me.

Hi @Chman , thx for your awesome Colorful Asset, did you have find a solution this no brainer must fix bug?

I had some luck with script on this page using the standard unity effects, maybe with some kind work it could work with Colorful?

It’s essential for me to try and keep a full stack of deferred cameras with separate FX otherwise my game goes back to the 90’s vibe. :slight_smile:

Think I found a fix. Please read the last post on that thread.

The only “complete fix” is to render each camera manually to a global render texture and blit it to screen. It works fine, it’s not really hard to setup but adds another layer of messiness to the scene…

Like this?

using System.Collections;
using UnityEngine;

// fixes the deferred lighting missing final copy&resolve, so the next camera gets the correctly final processed image in the temp screen RT as input
// NOTE: The script must be the last in the image effect chain, so order it in the inspector!
[ExecuteInEditMode]
public class CopyToScreenRT : MonoBehaviour
{
    private RenderTexture activeRT; // hold the org. screen RT

    private void OnPreRender()
    {
        if (GetComponent<Camera>().actualRenderingPath == RenderingPath.DeferredLighting) {
            activeRT = RenderTexture.active;
        }
        else {
            activeRT = null;
        }
    }

    private void OnRenderImage(RenderTexture src, RenderTexture dest)
    {
        if (GetComponent<Camera>().actualRenderingPath == RenderingPath.DeferredLighting && activeRT) {
            if (src.format == activeRT.format) {
                Graphics.Blit(src, activeRT);
            }
            else {
                Debug.LogWarning("Cant resolve texture, because of different formats!");
            }
        }

        // script must be last anyway, so we don't need a final copy?
        Graphics.Blit(src, dest); // just in case we are not last!
    }
}

This only works with a Standard Effect like fisheye included in the chain. When disabled it has no effect, do you know why?

This is the old fix for the old deferred path in Unity 4.x. I’m not sure why it would work with a standard effect and not one of Colorful FX. To be honest I haven’t really investigated it.

What I meant previously was to manually render the cameras one by one in a render texture. Something along these lines (typed on the fly so it’s untested but just so you get the idea) :

using UnityEngine;

[RequireComponent(typeof(Camera))]
[ExecuteInEditMode]
public class CameraGlobalRT : MonoBehaviour
{
    public Camera Camera0;
    public Camera Camera1;
    // Add more cameras or use a list instead

    protected Camera m_Camera;
    protected RenderTexture m_RenderTexture;

    void OnEnable()
    {
        m_Camera = GetComponent<Camera>();
    }

    void Start()
    {
        if (Camera0 == null || Camera1 == null)
            return;

        Camera0.enabled = false;
        Camera1.enabled = false;
    }

    void OnPreRender()
    {
        if (Camera0 == null || Camera1 == null)
            return;

        m_RenderTexture = RenderTexture.GetTemporary(m_Camera.pixelWidth, m_Camera.pixelHeight);

        Camera0.targetTexture = m_RenderTexture;
        Camera0.Render();
        Camera0.targetTexture = null;

        Camera1.targetTexture = m_RenderTexture;
        Camera1.Render();
        Camera1.targetTexture = null;
    }

    void OnRenderImage(RenderTexture source, RenderTexture destination)
    {
        if (Camera0 == null || Camera1 == null)
            Graphics.Blit(source, destination);
        else
            Graphics.Blit(m_RenderTexture, destination);

        RenderTexture.ReleaseTemporary(m_RenderTexture);
    }
}

Once again, this is very rough code just to give you an idea of what I meant. Add HDR & depth support if needed when requesting m_RenderTexture. That should give you a good starting point. Put this on a new Camera object somewhere on your scene.

2 Likes

Ok I’ve finally got round putting some time aside to learn more about RenderTexture usage and try this out. Thanks for it. If not too much trouble I have a question:

So is this a 3 camera config, 1 cam on which the script is placed and another another 2 cameras which are passed in as camera object params? Does the script cam combine the other 2 passed in into one renderTexture and render it on to the script cam. I think that’s it? I best try it out I suppose… Just trying to get the logic right, the disabling of the 2 cams is confusing me. Thanks

Yep exactly, one “main” camera that doesn’t render anything (only used to blit the global render texture to screen) and two other cameras that get rendered into the global render texture one by one (I used two as an example, you can add more of course).

Wow, just tried it. Yep works a treat. Are there any gotchas with this method? UFPS has gone a little crazy but thats because I’ve dropped cam 2 in with it. Thank you so much, it’s like magic. Got a big smile :slight_smile:

The only gotcha I can think about is an increased VRam usage for the global render texture. If you need HDR, use ARGBHalf instead of ARGBFloat to save some memory. Otherwise ARGB32 is fine but the bigger the screen resolution is, the more VRam it will require (not that it matters much on desktop platforms nowadays anyway, most GPU come with 2GB/4GB).

Thanks, very helpful.

I’ve had a interesting day messing with this and noticed a couple things and wondered if I can pick your brains again please?

16 bit depth buffer gives a little performance boast over 24 for obvious reasons, is 16 ok to use, whats the default?

My second cam with Depth Only (Deferred) is casting no shadows for items included on the mask? Thanks again, making great headway. I’ll be buying Lutify soon :wink:

24 comes with stencil buffer, 16 should be fine in most cases. The default value when you call GetTemporary() is 0.

By ‘mask’ you mean the camera on top ? In this case I wouldn’t expect it to cast any shadow on objects rendered by other cameras (remember, the camera is clearing the depth).

Cheers, oh yeah durr. It’s trying to cast onto the other camera which has already rendered the shadows with that cameras depth buffers. What I’m aiming to fix is my scattering fog effects bleeding through walls so tried putting my buildings onto another camera with cull masks. I’ve solved it by putting the buildings onto both layers. The top layer blocks the fog bleed now, simple but not used to having objects on 2 cams in this manner.

You really helped out Chman, I owe you one. Once I solve my Z-Fighting flicker on terrain my game will really look the part. :wink:

Is Scion your work? It mentions Lutify works with it, can I use Lutify perfectly well with my own stack. Thank you.

Nah, Scion is from the talented @Aieth :wink: Lutify is compatible with Scion if you already own it but it can work on its own if you only need a color grading pass.

Cool I’m going to get Lutify first, from checking my performance the image effects don’t have much overhead anyway. I’ve already paid out for single effects. Should I just go for it in your opinion?

Honestly, it depend on your needs. Lutify is a LUT library first, so if you need a lot of pre-made LUTs then yeah, go for it :slight_smile: If you just want it for the color grading effect itself and don’t care about the pre-made LUTs, then I would recommend getting something else (part of a bigger package) or using the default LUT effect that comes with Unity.

I think we’re getting out of topic :wink: PM me if you want to talk more about Lutify.

Hi @Chman ,

Now that you are an insider, have you gained any more insights about this issue? Is this setup really not supported?

I’ve found multiple posts about this problem dating as far as 2010.