mixedreality/com.microsoft.mixedreality..../Runtime/Subsystems/AnchorStore.cs

160 lines
6.3 KiB
C#

// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.
using System;
using System.Collections.Generic;
using System.Text;
using System.Threading.Tasks;
using UnityEngine;
using UnityEngine.XR.ARSubsystems;
namespace Microsoft.MixedReality.OpenXR
{
internal static class OpenXRAnchorStoreFactory
{
private static MixedRealityFeaturePlugin Feature => OpenXRFeaturePlugin<MixedRealityFeaturePlugin>.Feature;
private static Task<OpenXRAnchorStore> m_anchorStoreLoadTask = null;
private static ulong m_currentOpenxrSession;
public static Task<OpenXRAnchorStore> LoadAnchorStoreAsync(XRAnchorSubsystem anchorSubsystem)
{
if (!(anchorSubsystem is AnchorSubsystem))
{
Debug.LogWarning($"LoadAnchorStoreAsync: subsystem is not of type Microsoft.MixedReality.AnchorSubsystem. type: {anchorSubsystem.GetType()}");
return Task.FromResult<OpenXRAnchorStore>(null);
}
// Load anchor store only once per OpenXR session. And load again if the session changes. This is specifically added to support loading anchor store in subsequent app remoting connections.
if (m_anchorStoreLoadTask == null || (m_anchorStoreLoadTask.IsCompleted && m_anchorStoreLoadTask.Result == null) || m_currentOpenxrSession != OpenXRContext.Current.Session)
{
if (!Feature.IsValidAndEnabled())
{
Debug.LogWarning($"LoadAnchorStoreAsync: The anchor store is not supported; {MixedRealityFeaturePlugin.featureName} is not enabled.");
return Task.FromResult<OpenXRAnchorStore>(null);
}
if (OpenXRContext.Current.Session == 0)
{
Debug.LogWarning("LoadAnchorStoreAsync: Cannot load anchor store without a valid XR session.");
return Task.FromResult<OpenXRAnchorStore>(null);
}
m_currentOpenxrSession = OpenXRContext.Current.Session;
m_anchorStoreLoadTask = Task.Run(() =>
{
bool nativeAnchorStoreLoaded = NativeLib.LoadAnchorStore();// Blocking, potentially long call
if (!nativeAnchorStoreLoaded)
{
Debug.LogWarning("LoadAnchorStoreAsync: The anchor store is not supported; either the feature is not enabled, or the related OpenXR extensions are not supported");
return null;
}
return new OpenXRAnchorStore();
});
}
return m_anchorStoreLoadTask;
}
}
internal class OpenXRAnchorStore
{
internal static MixedRealityFeaturePlugin Feature => OpenXRFeaturePlugin<MixedRealityFeaturePlugin>.Feature;
private List<string> m_persistedAnchorNamesCache;
private bool m_persistedAnchorNamesCacheDirty = true;
private readonly object m_persistedAnchorNamesCacheLock = new object();
public IReadOnlyList<string> PersistedAnchorNames
{
get
{
lock (m_persistedAnchorNamesCacheLock)
{
if (m_persistedAnchorNamesCacheDirty)
{
UpdatePersistedAnchorNames();
m_persistedAnchorNamesCacheDirty = false;
}
return m_persistedAnchorNamesCache;
}
}
}
private void UpdatePersistedAnchorNames()
{
lock (m_persistedAnchorNamesCacheLock)
{
uint numPersisted = 0;
m_persistedAnchorNamesCache = new List<string>();
numPersisted = NativeLib.GetNumPersistedAnchorNames();
for (uint i = 0; i < numPersisted; i++)
{
// A persisted anchor with a name > 255 chars does not appear
// to be supported by the anchor store winrt implementation.
StringBuilder stringBuilder = new StringBuilder(255);
NativeLib.GetPersistedAnchorName(i, stringBuilder, (uint)stringBuilder.Capacity);
m_persistedAnchorNamesCache.Add(stringBuilder.ToString());
}
}
}
public TrackableId LoadAnchor(string name)
{
Guid persistedAnchor = NativeLib.LoadPersistedAnchor(name);
return FeatureUtils.ToTrackableId(persistedAnchor);
}
public bool TryPersistAnchor(string name, TrackableId trackableId)
{
bool anchorPersisted = false;
lock (m_persistedAnchorNamesCacheLock)
{
m_persistedAnchorNamesCacheDirty = true;
anchorPersisted = NativeLib.TryPersistAnchor(name, FeatureUtils.ToGuid(trackableId));
}
return anchorPersisted;
}
public void UnpersistAnchor(string name)
{
lock (m_persistedAnchorNamesCacheLock)
{
m_persistedAnchorNamesCacheDirty = true;
NativeLib.UnpersistAnchor(name);
}
}
public void Clear()
{
lock (m_persistedAnchorNamesCacheLock)
{
m_persistedAnchorNamesCacheDirty = true;
NativeLib.ClearPersistedAnchors();
}
}
public Task<bool> TryReloadAnchorStoreAsync()
{
return Task.Run(() =>
{
if (OpenXRContext.Current.Session == 0)
{
return false;
}
if (NativeLib.GetAnchorCount() != 0)
{
Debug.LogError("TryReloadAnchorsAsync cannot execute if anchors exist in scene. Remove all anchors before calling TryReloadNativeAnchorsAsync.");
return false;
}
lock (m_persistedAnchorNamesCacheLock)
{
m_persistedAnchorNamesCacheDirty = true;
NativeLib.ResetAnchorStore();
return NativeLib.LoadAnchorStore(); // Blocking, potentially long call
}
});
}
}
} // namespace Microsoft.MixedReality.OpenXR