XR Plug-In Management and MARS: manual initialization

Hi,

recent ARFoundation versions utilize the XR Plug-In Management. In my use-case I don’t want AR (ARKit in my case) to be initialized at start, but rather when the AR/MARS scene is loaded.

I disabled “Initialize XR on Startup” in Project Settings → XR Plug-In Management. With ARFoundation (only) this works after some manual init code.

How do you pull this off with MARS? My screen stays black :confused:

Thanks a lot in advance,

atti

Hey @tteneder ,
This may be tricky, couple of suggestions:

  • Easiest way is probably using separate scenes (but sound like you already tried this): firstly without AR, and then switch to the AR enabled scene.

  • Try having the ARSession in your scene already, so that MARS will use that instead of initializing it’s own. That may do it.

thanks for your suggestions.

I tried a lot for a whole day and eventually found a way that works. I used the MARS sample scene “Game_Simple” as a minimum use case:

Create an initial scene with nothing but the following starter script. It starts XR and loads the actual MARS scene afterwards:

using System.Collections;
using UnityEngine;
using UnityEngine.XR.ARFoundation;
using UnityEngine.XR.Management;
using UnityEngine.SceneManagement;

public class ARStarter : MonoBehaviour {
    void Start() {
#if UNITY_EDITOR
        LoadARScene();
#else
            StartCoroutine(StartXR());
#endif
    }

    public IEnumerator StartXR() {
        yield return XRGeneralSettings.Instance.Manager.InitializeLoader();

        if (XRGeneralSettings.Instance.Manager.activeLoader == null) {
            Debug.LogError("Initializing XR Failed. Check Editor or Player log for details.");
        } else {
            Debug.Log("Starting XR...");
            XRGeneralSettings.Instance.Manager.StartSubsystems();
            yield return null;
            LoadARScene();
        }
    }

    void LoadARScene() {
        SceneManager.LoadScene("Game_Simple", LoadSceneMode.Single);
    }

    void StopXR() {
        Debug.Log("Stopping XR...");
        XRGeneralSettings.Instance.Manager.StopSubsystems();
        XRGeneralSettings.Instance.Manager.DeinitializeLoader();
        Debug.Log("XR stopped completely.");
    }
}

Any other attempt ( loading scene additionally or single scene that enables MARS/AR components once XR started) failed.

I’m now trying to apply this to my real world project.

1 Like

Here’s another version for projects where AR is optional. It tries to auto-detect if AR is supported:

UPDATE: This does not work on non-ARKit compatible devices!!!

    public IEnumerator StartXR() {

        yield return XRGeneralSettings.Instance.Manager.InitializeLoader();

        if (XRGeneralSettings.Instance.Manager.activeLoader == null) {
            Debug.LogError("Initializing XR Failed. Check Editor or Player log for details.");
        } else {
            Debug.Log("Starting XR...");
            XRGeneralSettings.Instance.Manager.StartSubsystems();
         
           // Have to add an ARSession instance to avoid Null Pointer Dereference
            gameObject.AddComponent<ARSession>();
            yield return null;

            if ((ARSession.state == ARSessionState.None) ||
                (ARSession.state == ARSessionState.CheckingAvailability))
            {
                yield return ARSession.CheckAvailability();
            }

            if (ARSession.state == ARSessionState.Unsupported)
            {
                Debug.LogError("AR is not supported! Do some fallback");
                yield break;
            }

            LoadARScene();
        }
    }
3 Likes

Sadly I have to report, that the code in the last post (optional AR) is NOT working.

Testing this on an iPhone 6 the ARSession.state ends up being ArSessionState.Ready :confused:

Should I file an issue or am I doing something obviously wrong?

Hey @tteneder , have you tried with a clean project to get just the AR Foundation delayed initialization working? We can try pull in someone from the AR Foundation team if that helps. Also I would not be surprised if this is related to the version of iPhone you are using, you may want to consider only supporting later devices for the AR aspect.

Hi leweyg,

I didn’t properly describe the context. With the above method MARS is working on devices that support it (good so far).

What I try to handle is devices that don’t support ARKit (like the iPhone 6) and provide a fallback. I’d expect ARSession.state to be ArSessionState.Unsupported on those devices.

I worked around this by manually detecting support for now:

Assets/Plugins/iOS/MyCustomARKitCheck.mm:

#import <ARKit/ARKit.h>

extern "C" {
    bool MyCustomCheckARKitSupport() {
        return ARWorldTrackingConfiguration.isSupported;
    }
}

and in C#:

using System.Runtime.InteropServices;
…
if(!MyCustomCheckARKitSupport()) {
            Debug.LogError("ARKit is not supported!");
}
…

[DllImport ("__Internal")]
extern static bool MyCustomCheckARKitSupport();

Hey @tteneder , thanks for the clarification that this is about devices that DO NOT support AR Foundation (iPhone 6 etc.)… and it looks like you have code to detect this case. The question is now how to disable those parts of MARS which may interact with ARF? If that is the case, I believe the trick may be to use a custom “Default Island” or modify it at runtime (something like Assets/MARS/Settings/FunctionalityIslands/Default Island.asset ), so that you can configure which systems are spun up (and disable any that would cause AR Foundation to launch)? How does that sound? Or what issue are you trying to solve here?

All I wanted is optional AR with a fallback-detection. In my case the fallback is a legacy(non-MARS), custom, gyroscope-based experience. I don’t want to re-invent this fallback (in other words re-build it on MARS) as it’ll fade away eventually.

MARS/ARF gives devs a good abstraction layer, so I tried to make the AR Subsystem detection abstract as well, but couldn’t get it to work (with non-ARKit iOS devices).

I’ll go with my workaround posted above, even if it’s not as abstract as MARS/ARF.

Anyways, thanks for your support!

Did a quick scan of the code, it looks like the ‘ARSessionState.Ready’ bug is known & fixed. It’s just making it’s way down the pipes. Edit: Fixed in 4.1.0 Preview 6, being backported.

As for initializing manually, what you are doing works. Another available option is to create an ‘ARSession’ Component on a Monobehaviour and start with it disabled. You can then enable the ARSession component in order to start up AR Foundation. You can also call yield return ARSession.CheckAvailability(); to refresh ARSession.state to check if it can be enabled without enabling it.

MARS should grab an existing ARSession Monobehaviour if one is already created, giving you manual control. ARSession also does a lazy initialization, so it will initialize and start subsystems if necessary, or use whatever was already initialized.

1 Like

That’s great to hear! Thanks for your excellent support!