From eae9489ba80a9c8012a555f388c6b64fae035308 Mon Sep 17 00:00:00 2001 From: Santiago Lo Coco Date: Sun, 13 Oct 2024 18:53:46 +0200 Subject: [PATCH] Add openxr package --- .../CHANGELOG.md | 382 ++++++++ .../CHANGELOG.md.meta | 7 + .../Documentation~/README.md | 13 + com.microsoft.mixedreality.openxr/Editor.meta | 8 + .../Editor/AssemblyInfo.cs | 9 + .../Editor/AssemblyInfo.cs.meta | 11 + .../Editor/BuildProcessors.meta | 8 + .../BuildProcessors/MetaBuildProcessor.cs | 150 ++++ .../MetaBuildProcessor.cs.meta | 11 + .../MixedRealityBuildProcessor.cs | 399 +++++++++ .../MixedRealityBuildProcessor.cs.meta | 3 + .../Editor/FeatureSets.meta | 8 + .../FeatureSets/AppRemotingFeatureSet.cs | 38 + .../FeatureSets/AppRemotingFeatureSet.cs.meta | 11 + .../Editor/FeatureSets/HoloLensFeatureSet.cs | 36 + .../FeatureSets/HoloLensFeatureSet.cs.meta | 11 + .../Editor/FeatureSets/WMRFeatureSet.cs | 36 + .../Editor/FeatureSets/WMRFeatureSet.cs.meta | 11 + .../Editor/Inspectors.meta | 8 + .../EyeLevelSceneOriginInspector.cs | 17 + .../EyeLevelSceneOriginInspector.cs.meta | 11 + .../PlayModeRemotingPluginInspector.cs | 33 + .../PlayModeRemotingPluginInspector.cs.meta | 11 + .../Inspectors/RemotingSettingsInspector.cs | 19 + .../RemotingSettingsInspector.cs.meta | 11 + ...icrosoft.MixedReality.OpenXR.Editor.asmdef | 44 + ...oft.MixedReality.OpenXR.Editor.asmdef.meta | 7 + .../Editor/PropertyDrawers.meta | 8 + .../PropertyDrawers/DocURLAttributeDrawer.cs | 40 + .../DocURLAttributeDrawer.cs.meta | 11 + ...awerVisibleToBuildTargetAttributeDrawer.cs | 29 + ...isibleToBuildTargetAttributeDrawer.cs.meta | 11 + .../LabelWidthAttributeDrawer.cs | 22 + .../LabelWidthAttributeDrawer.cs.meta | 11 + .../Editor/Settings.meta | 8 + .../Editor/Settings/PlatformValidation.cs | 752 ++++++++++++++++ .../Settings/PlatformValidation.cs.meta | 11 + .../Editor/Settings/PlayModeRemotingWindow.cs | 120 +++ .../Settings/PlayModeRemotingWindow.cs.meta | 11 + com.microsoft.mixedreality.openxr/LICENSE.md | 21 + .../LICENSE.md.meta | 7 + .../Runtime.meta | 8 + .../Runtime/API.meta | 8 + .../Runtime/API/ARMarker.meta | 8 + .../Runtime/API/ARMarker/ARMarker.cs | 179 ++++ .../Runtime/API/ARMarker/ARMarker.cs.meta | 11 + .../Runtime/API/ARMarker/ARMarkerManager.cs | 183 ++++ .../API/ARMarker/ARMarkerManager.cs.meta | 11 + .../Runtime/API/ARMarker/ARMarkerScale.cs | 39 + .../API/ARMarker/ARMarkerScale.cs.meta | 11 + .../API/ARMarker/ARMarkersChangedEventArgs.cs | 59 ++ .../ARMarkersChangedEventArgs.cs.meta | 11 + .../Runtime/API/ARMarker/XRMarker.cs | 119 +++ .../Runtime/API/ARMarker/XRMarker.cs.meta | 11 + .../Runtime/API/ARMarker/XRMarkerSubsystem.cs | 156 ++++ .../API/ARMarker/XRMarkerSubsystem.cs.meta | 11 + .../ARMarker/XRMarkerSubsystemDescriptor.cs | 53 ++ .../XRMarkerSubsystemDescriptor.cs.meta | 11 + .../Runtime/API/AnchorConverter.cs | 185 ++++ .../Runtime/API/AnchorConverter.cs.meta | 11 + .../Runtime/API/AppRemoting.cs | 842 ++++++++++++++++++ .../Runtime/API/AppRemoting.cs.meta | 11 + .../Runtime/API/ControllerModel.cs | 211 +++++ .../Runtime/API/ControllerModel.cs.meta | 11 + .../Runtime/API/ControllerModelArticulator.cs | 87 ++ .../API/ControllerModelArticulator.cs.meta | 11 + .../Runtime/API/EyeLevelSceneOrigin.cs | 102 +++ .../Runtime/API/EyeLevelSceneOrigin.cs.meta | 11 + .../Runtime/API/FrameTime.cs | 21 + .../Runtime/API/FrameTime.cs.meta | 11 + .../Runtime/API/GestureRecognizer.cs | 363 ++++++++ .../Runtime/API/GestureRecognizer.cs.meta | 11 + .../Runtime/API/HandMeshTracker.cs | 138 +++ .../Runtime/API/HandMeshTracker.cs.meta | 11 + .../Runtime/API/HandTracker.cs | 267 ++++++ .../Runtime/API/HandTracker.cs.meta | 11 + .../Runtime/API/MeshSettings.cs | 151 ++++ .../Runtime/API/MeshSettings.cs.meta | 11 + .../Runtime/API/OpenXRContext.cs | 52 ++ .../Runtime/API/OpenXRContext.cs.meta | 11 + .../Runtime/API/OpenXRTime.cs | 48 + .../Runtime/API/OpenXRTime.cs.meta | 11 + .../Runtime/API/PerceptionInterop.cs | 43 + .../Runtime/API/PerceptionInterop.cs.meta | 11 + .../Runtime/API/SelectKeywordRecognizer.cs | 79 ++ .../API/SelectKeywordRecognizer.cs.meta | 11 + .../Runtime/API/SpatialGraphNode.cs | 132 +++ .../Runtime/API/SpatialGraphNode.cs.meta | 11 + .../Runtime/API/TrackingMapAPI.cs | 172 ++++ .../Runtime/API/TrackingMapAPI.cs.meta | 11 + .../Runtime/API/ViewConfiguration.cs | 166 ++++ .../Runtime/API/ViewConfiguration.cs.meta | 11 + .../Runtime/API/XRAnchorStore.cs | 149 ++++ .../Runtime/API/XRAnchorStore.cs.meta | 11 + .../Runtime/API/XRAnchorTransferBatch.cs | 110 +++ .../Runtime/API/XRAnchorTransferBatch.cs.meta | 11 + .../Runtime/AssemblyInfo.cs | 12 + .../Runtime/AssemblyInfo.cs.meta | 11 + .../Runtime/FeaturePlugins.meta | 8 + .../FeaturePlugins/AppRemotingPlugin.cs | 104 +++ .../FeaturePlugins/AppRemotingPlugin.cs.meta | 11 + .../HPMixedRealityControllerProfile.cs | 548 ++++++++++++ .../HPMixedRealityControllerProfile.cs.meta | 11 + .../HandTrackingFeaturePlugin.cs | 159 ++++ .../HandTrackingFeaturePlugin.cs.meta | 11 + .../MixedRealityFeaturePlugin.cs | 144 +++ .../MixedRealityFeaturePlugin.cs.meta | 11 + .../MotionControllerFeaturePlugin.cs | 32 + .../MotionControllerFeaturePlugin.cs.meta | 11 + .../FeaturePlugins/OpenXRFeaturePlugin.cs | 262 ++++++ .../OpenXRFeaturePlugin.cs.meta | 11 + .../OpenXRFeaturePluginManager.cs | 72 ++ .../OpenXRFeaturePluginManager.cs.meta | 11 + .../FeaturePlugins/PlayModeRemotingPlugin.cs | 206 +++++ .../PlayModeRemotingPlugin.cs.meta | 11 + .../Runtime/FeatureValidators.meta | 8 + .../FeatureValidators/AppRemotingValidator.cs | 89 ++ .../AppRemotingValidator.cs.meta | 11 + .../MixedRealityFeatureValidator.cs | 88 ++ .../MixedRealityFeatureValidator.cs.meta | 11 + .../PlayModeRemotingValidator.cs | 164 ++++ .../PlayModeRemotingValidator.cs.meta | 11 + .../FeatureValidators/ValidationRuleset.cs | 188 ++++ .../ValidationRuleset.cs.meta | 11 + .../Microsoft.MixedReality.OpenXR.asmdef | 69 ++ .../Microsoft.MixedReality.OpenXR.asmdef.meta | 7 + .../Runtime/NativeLib.cs | 458 ++++++++++ .../Runtime/NativeLib.cs.meta | 11 + .../Runtime/Subsystems.meta | 8 + .../Runtime/Subsystems/ARMarker.meta | 8 + .../Subsystems/ARMarker/MarkerSubsystem.cs | 428 +++++++++ .../ARMarker/MarkerSubsystem.cs.meta | 11 + .../Runtime/Subsystems/AnchorStore.cs | 159 ++++ .../Runtime/Subsystems/AnchorStore.cs.meta | 11 + .../Runtime/Subsystems/AnchorSubsystem.cs | 176 ++++ .../Subsystems/AnchorSubsystem.cs.meta | 11 + .../Runtime/Subsystems/AnchorTransferBatch.cs | 171 ++++ .../Subsystems/AnchorTransferBatch.cs.meta | 11 + .../Subsystems/AppRemotingCoroutineRunner.cs | 43 + .../AppRemotingCoroutineRunner.cs.meta | 11 + .../Subsystems/AppRemotingSubsystem.cs | 593 ++++++++++++ .../Subsystems/AppRemotingSubsystem.cs.meta | 11 + .../Runtime/Subsystems/GestureSubsystem.cs | 190 ++++ .../Subsystems/GestureSubsystem.cs.meta | 11 + .../Subsystems/HandTrackingSubsystem.cs | 53 ++ .../Subsystems/HandTrackingSubsystem.cs.meta | 11 + .../Subsystems/InternalMeshSettings.cs | 16 + .../Subsystems/InternalMeshSettings.cs.meta | 11 + .../Runtime/Subsystems/MeshSubsystem.cs | 31 + .../Runtime/Subsystems/MeshSubsystem.cs.meta | 11 + .../Subsystems/OpenXRRuntimeRestartHandler.cs | 64 ++ .../OpenXRRuntimeRestartHandler.cs.meta | 11 + .../Runtime/Subsystems/PlaneSubsystem.cs | 204 +++++ .../Runtime/Subsystems/PlaneSubsystem.cs.meta | 11 + .../Runtime/Subsystems/PluginEnvironment.cs | 60 ++ .../Subsystems/PluginEnvironment.cs.meta | 11 + .../Runtime/Subsystems/RaycastSubsystem.cs | 151 ++++ .../Subsystems/RaycastSubsystem.cs.meta | 11 + .../SelectKeywordRecognizerProvider.cs | 215 +++++ .../SelectKeywordRecognizerProvider.cs.meta | 11 + .../Runtime/Subsystems/SessionSubsystem.cs | 172 ++++ .../Subsystems/SessionSubsystem.cs.meta | 11 + .../Runtime/Subsystems/SubsystemController.cs | 101 +++ .../Subsystems/SubsystemController.cs.meta | 11 + .../Subsystems/TrackingMapSubsystem.cs | 68 ++ .../Subsystems/TrackingMapSubsystem.cs.meta | 11 + .../Subsystems/ViewConfigurationSettings.cs | 131 +++ .../ViewConfigurationSettings.cs.meta | 11 + .../Runtime/Subsystems/XrSessionState.cs | 65 ++ .../Runtime/Subsystems/XrSessionState.cs.meta | 11 + .../Runtime/UnitySubsystemsManifest.json | 15 + .../Runtime/UnitySubsystemsManifest.json.meta | 7 + .../Runtime/Utilities.meta | 8 + .../Utilities/BuildProcessorHelpers.cs | 190 ++++ .../Utilities/BuildProcessorHelpers.cs.meta | 11 + .../Runtime/Utilities/Disposable.cs | 50 ++ .../Runtime/Utilities/Disposable.cs.meta | 11 + .../Runtime/Utilities/DocURLAttribute.cs | 19 + .../Runtime/Utilities/DocURLAttribute.cs.meta | 11 + ...itorDrawerVisibleToBuildTargetAttribute.cs | 24 + ...rawerVisibleToBuildTargetAttribute.cs.meta | 11 + .../Runtime/Utilities/FeatureUtils.cs | 32 + .../Runtime/Utilities/FeatureUtils.cs.meta | 11 + .../Runtime/Utilities/FindObjectUtility.cs | 64 ++ .../Utilities/FindObjectUtility.cs.meta | 11 + .../Runtime/Utilities/LabelWidthAttribute.cs | 19 + .../Utilities/LabelWidthAttribute.cs.meta | 11 + .../Runtime/Utilities/XRSettingsHelpers.cs | 75 ++ .../Utilities/XRSettingsHelpers.cs.meta | 11 + .../Runtime/android.meta | 8 + .../Runtime/android/arm64.meta | 8 + .../android/arm64/libMicrosoftOpenXRPlugin.so | Bin 0 -> 1255096 bytes .../arm64/libMicrosoftOpenXRPlugin.so.meta | 81 ++ .../Runtime/universalwindows.meta | 8 + .../Runtime/universalwindows/arm32.meta | 8 + .../arm32/MicrosoftOpenXRPlugin.dll | Bin 0 -> 653744 bytes .../arm32/MicrosoftOpenXRPlugin.dll.meta | 81 ++ .../Runtime/universalwindows/arm64.meta | 8 + .../arm64/MicrosoftOpenXRPlugin.dll | Bin 0 -> 791584 bytes .../arm64/MicrosoftOpenXRPlugin.dll.meta | 81 ++ .../Runtime/universalwindows/x64.meta | 8 + ...soft.Holographic.AppRemoting.OpenXr.SU.dll | Bin 0 -> 769976 bytes ...Holographic.AppRemoting.OpenXr.SU.dll.meta | 81 ++ ...crosoft.Holographic.AppRemoting.OpenXr.dll | Bin 0 -> 10213296 bytes ...ft.Holographic.AppRemoting.OpenXr.dll.meta | 81 ++ .../x64/MicrosoftOpenXRPlugin.dll | Bin 0 -> 832432 bytes .../x64/MicrosoftOpenXRPlugin.dll.meta | 81 ++ .../universalwindows/x64/PerceptionDevice.dll | Bin 0 -> 24496 bytes .../x64/PerceptionDevice.dll.meta | 81 ++ .../Runtime/windows.meta | 8 + .../Runtime/windows/x64.meta | 8 + .../Runtime/windows/x64/DebugPanel.dll | Bin 0 -> 1397128 bytes .../Runtime/windows/x64/DebugPanel.dll.meta | 63 ++ ...soft.Holographic.AppRemoting.OpenXr.SU.dll | Bin 0 -> 769976 bytes ...Holographic.AppRemoting.OpenXr.SU.dll.meta | 81 ++ ...crosoft.Holographic.AppRemoting.OpenXr.dll | Bin 0 -> 10215344 bytes ...ft.Holographic.AppRemoting.OpenXr.dll.meta | 81 ++ .../windows/x64/MicrosoftOpenXRPlugin.dll | Bin 0 -> 833968 bytes .../x64/MicrosoftOpenXRPlugin.dll.meta | 81 ++ .../Runtime/windows/x64/PerceptionDevice.dll | Bin 0 -> 27064 bytes .../windows/x64/PerceptionDevice.dll.meta | 81 ++ .../Runtime/windows/x64/RemotingXR.json | 6 + .../Runtime/windows/x64/RemotingXR.json.meta | 7 + .../package.json | 38 + .../package.json.meta | 7 + 225 files changed, 14557 insertions(+) create mode 100644 com.microsoft.mixedreality.openxr/CHANGELOG.md create mode 100644 com.microsoft.mixedreality.openxr/CHANGELOG.md.meta create mode 100644 com.microsoft.mixedreality.openxr/Documentation~/README.md create mode 100644 com.microsoft.mixedreality.openxr/Editor.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/AssemblyInfo.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/AssemblyInfo.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/BuildProcessors.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MetaBuildProcessor.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MetaBuildProcessor.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MixedRealityBuildProcessor.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MixedRealityBuildProcessor.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/FeatureSets.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/FeatureSets/AppRemotingFeatureSet.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/FeatureSets/AppRemotingFeatureSet.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/FeatureSets/HoloLensFeatureSet.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/FeatureSets/HoloLensFeatureSet.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/FeatureSets/WMRFeatureSet.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/FeatureSets/WMRFeatureSet.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/Inspectors.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/Inspectors/EyeLevelSceneOriginInspector.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/Inspectors/EyeLevelSceneOriginInspector.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/Inspectors/PlayModeRemotingPluginInspector.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/Inspectors/PlayModeRemotingPluginInspector.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/Inspectors/RemotingSettingsInspector.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/Inspectors/RemotingSettingsInspector.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/Microsoft.MixedReality.OpenXR.Editor.asmdef create mode 100644 com.microsoft.mixedreality.openxr/Editor/Microsoft.MixedReality.OpenXR.Editor.asmdef.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/PropertyDrawers.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/DocURLAttributeDrawer.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/DocURLAttributeDrawer.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/EditorDrawerVisibleToBuildTargetAttributeDrawer.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/EditorDrawerVisibleToBuildTargetAttributeDrawer.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/LabelWidthAttributeDrawer.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/LabelWidthAttributeDrawer.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/Settings.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/Settings/PlatformValidation.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/Settings/PlatformValidation.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Editor/Settings/PlayModeRemotingWindow.cs create mode 100644 com.microsoft.mixedreality.openxr/Editor/Settings/PlayModeRemotingWindow.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/LICENSE.md create mode 100644 com.microsoft.mixedreality.openxr/LICENSE.md.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarker.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarker.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerManager.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerManager.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerScale.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerScale.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkersChangedEventArgs.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkersChangedEventArgs.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarker.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarker.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystem.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystem.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystemDescriptor.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystemDescriptor.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/AnchorConverter.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/AnchorConverter.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/AppRemoting.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/AppRemoting.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ControllerModel.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ControllerModel.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ControllerModelArticulator.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ControllerModelArticulator.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/EyeLevelSceneOrigin.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/EyeLevelSceneOrigin.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/FrameTime.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/FrameTime.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/GestureRecognizer.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/GestureRecognizer.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/HandMeshTracker.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/HandMeshTracker.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/HandTracker.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/HandTracker.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/MeshSettings.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/MeshSettings.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/OpenXRContext.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/OpenXRContext.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/OpenXRTime.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/OpenXRTime.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/PerceptionInterop.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/PerceptionInterop.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/SelectKeywordRecognizer.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/SelectKeywordRecognizer.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/SpatialGraphNode.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/SpatialGraphNode.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/TrackingMapAPI.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/TrackingMapAPI.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ViewConfiguration.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/ViewConfiguration.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorStore.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorStore.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorTransferBatch.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorTransferBatch.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/AssemblyInfo.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/AssemblyInfo.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/AppRemotingPlugin.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/AppRemotingPlugin.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HPMixedRealityControllerProfile.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HPMixedRealityControllerProfile.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HandTrackingFeaturePlugin.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HandTrackingFeaturePlugin.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MixedRealityFeaturePlugin.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MixedRealityFeaturePlugin.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MotionControllerFeaturePlugin.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MotionControllerFeaturePlugin.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePlugin.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePlugin.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePluginManager.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePluginManager.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/PlayModeRemotingPlugin.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/PlayModeRemotingPlugin.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeatureValidators.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/AppRemotingValidator.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/AppRemotingValidator.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/MixedRealityFeatureValidator.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/MixedRealityFeatureValidator.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/PlayModeRemotingValidator.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/PlayModeRemotingValidator.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/ValidationRuleset.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/ValidationRuleset.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Microsoft.MixedReality.OpenXR.asmdef create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Microsoft.MixedReality.OpenXR.asmdef.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/NativeLib.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/NativeLib.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker/MarkerSubsystem.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker/MarkerSubsystem.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorStore.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorStore.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorSubsystem.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorSubsystem.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorTransferBatch.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorTransferBatch.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingCoroutineRunner.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingCoroutineRunner.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingSubsystem.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingSubsystem.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/GestureSubsystem.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/GestureSubsystem.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/HandTrackingSubsystem.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/HandTrackingSubsystem.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/InternalMeshSettings.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/InternalMeshSettings.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/MeshSubsystem.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/MeshSubsystem.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/OpenXRRuntimeRestartHandler.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/OpenXRRuntimeRestartHandler.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/PlaneSubsystem.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/PlaneSubsystem.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/PluginEnvironment.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/PluginEnvironment.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/RaycastSubsystem.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/RaycastSubsystem.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/SelectKeywordRecognizerProvider.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/SelectKeywordRecognizerProvider.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/SessionSubsystem.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/SessionSubsystem.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/SubsystemController.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/SubsystemController.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/TrackingMapSubsystem.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/TrackingMapSubsystem.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/ViewConfigurationSettings.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/ViewConfigurationSettings.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/XrSessionState.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Subsystems/XrSessionState.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/UnitySubsystemsManifest.json create mode 100644 com.microsoft.mixedreality.openxr/Runtime/UnitySubsystemsManifest.json.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/BuildProcessorHelpers.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/BuildProcessorHelpers.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/Disposable.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/Disposable.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/DocURLAttribute.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/DocURLAttribute.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/EditorDrawerVisibleToBuildTargetAttribute.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/EditorDrawerVisibleToBuildTargetAttribute.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/FeatureUtils.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/FeatureUtils.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/FindObjectUtility.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/FindObjectUtility.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/LabelWidthAttribute.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/LabelWidthAttribute.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/XRSettingsHelpers.cs create mode 100644 com.microsoft.mixedreality.openxr/Runtime/Utilities/XRSettingsHelpers.cs.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/android.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/android/arm64.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/android/arm64/libMicrosoftOpenXRPlugin.so create mode 100644 com.microsoft.mixedreality.openxr/Runtime/android/arm64/libMicrosoftOpenXRPlugin.so.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/arm32.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/arm32/MicrosoftOpenXRPlugin.dll create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/arm32/MicrosoftOpenXRPlugin.dll.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/arm64.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/arm64/MicrosoftOpenXRPlugin.dll create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/arm64/MicrosoftOpenXRPlugin.dll.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/x64.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/x64/Microsoft.Holographic.AppRemoting.OpenXr.SU.dll create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/x64/Microsoft.Holographic.AppRemoting.OpenXr.SU.dll.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/x64/Microsoft.Holographic.AppRemoting.OpenXr.dll create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/x64/Microsoft.Holographic.AppRemoting.OpenXr.dll.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/x64/MicrosoftOpenXRPlugin.dll create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/x64/MicrosoftOpenXRPlugin.dll.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/x64/PerceptionDevice.dll create mode 100644 com.microsoft.mixedreality.openxr/Runtime/universalwindows/x64/PerceptionDevice.dll.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/DebugPanel.dll create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/DebugPanel.dll.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/Microsoft.Holographic.AppRemoting.OpenXr.SU.dll create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/Microsoft.Holographic.AppRemoting.OpenXr.SU.dll.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/Microsoft.Holographic.AppRemoting.OpenXr.dll create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/Microsoft.Holographic.AppRemoting.OpenXr.dll.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/MicrosoftOpenXRPlugin.dll create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/MicrosoftOpenXRPlugin.dll.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/PerceptionDevice.dll create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/PerceptionDevice.dll.meta create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/RemotingXR.json create mode 100644 com.microsoft.mixedreality.openxr/Runtime/windows/x64/RemotingXR.json.meta create mode 100644 com.microsoft.mixedreality.openxr/package.json create mode 100644 com.microsoft.mixedreality.openxr/package.json.meta diff --git a/com.microsoft.mixedreality.openxr/CHANGELOG.md b/com.microsoft.mixedreality.openxr/CHANGELOG.md new file mode 100644 index 0000000..90d7e72 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/CHANGELOG.md @@ -0,0 +1,382 @@ +# Changelog + +All notable changes to this package will be documented in this file. + +The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) +and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.html). + +## [1.11.1] - Current Release +* Upgraded the OpenXR remoting runtime to 2.9.4 release. + * Fixed a deadlock issue that occurred when the GPU encoder was occupied for long durations. + * Fixed erroneous pinch gap values leading to unwanted interactions. +* Added new `OpenXRTime` API that contains methods to predict display times. +* Added `TryReloadAnchorStoreAsync` method to `XRAnchorStore` which can be used to reload the anchor store when the map or anchors have been modified outside the application. +* Fixed instances of null propagation on Unity objects + +## [1.11.0] - 2024-06-26 +* Internal test release + +## [1.10.1] - 2024-04-17 + +* Fixed a bug where RaycastSubsystem would search for a missing ARPlaneManager every frame. +* Fixed a bug where camera validation produced errors with no camera in the scene. +* Fixed a bug where the locatable camera was registered but not unregistered. + +## [1.10.0] - 2023-12-04 + +* Added new APIs `TrackingMapManager` and `TrackingMapType` to allow an application to opt into running in an Application-Exclusive tracking mode on HoloLens 2. +* Upgraded the OpenXR remoting runtime to 2.9.3 release. + * Holographic Remoting using the OpenXR API now supports the XR_MSFT_scene_marker extension. + * Holographic Remoting using the OpenXR API now supports GPU Adapter selection through the XrRemotingPreferredGraphicsAdapterMSFT extension struct. +* Updated the OpenXR MR Plugin package to be compatible with Unity's UPM Asset Store. +* Fixed an issue where ARMarkerManager was getting enabled even when marker functionality was unavailable. +* Fixed an issue where ARMarkers were not positioned correctly for some orientations when using TransformMode.Center. +* Added additional context to MR Plugin validation tooltips. + +## [1.9.0] - 2023-09-12 + +* Minimum requirement of Unity version is changed to Unity 2021.3 LTS. +* Added new APIs `ARMarkerManager`, `ARMarker` and others for QR code tracking on HoloLens 2. +* Added new API `AppRemoting.AudioCaptureMode` for remoting apps to choose audio capture mode between system-wide vs. within application. +* Added new APIs `AppRemoting.TryConvertToPlayerTime` and `AppRemoting.TryConvertToRemoteTime` to convert timestamp between remoting app and remoting player. +* Fixed a bug where the `AppRemoting.TryGetConnectionState` function sometimes did not return the correct `disconnectReason` when the session was unexpectedly lost. +* Added a new `Mixed Reality -> Project Setting Validations` menu to select 4 new rulesets to help users setup project settings easily for different types of Mixed Reality applications. +* The following APIs are deprecated and may be removed in future releases: + * Deprecated extension methods, such as `ARAnchorExtensions`, `XRAnchorExtensions`, and `MeshSubsystemExtensions`, in favor of their corresponding static functions. + * `EyeLevelSceneOrigin` is deprecated in favor of MRTK3 `Microsoft.MixedReality.Toolkit.Input.UnboundedTrackingMode` for HoloLens 2 application using unbounded space. For other XR applications use `Unity.​XR.​Core​Utils.XROrigin` instead. + +## [1.8.1] - 2023-06-21 + +* Restored Android support for hand tracking and controller models. + * These features are now deprecated instead of entirely removed. + * Using the OpenXR plugins from Unity and Meta is still recommended for these features. +* Depends on version 1.8.0 of Unity's OpenXR plugin. + * Deprecated our HP Reverb G2 controller bindings in the Mixed Reality OpenXR plugin. + * Recommend using the equivalent HP Reverb G2 controller bindings in Unity's OpenXR plugin with version 1.8.0 or above. +* Fixed a bug which prevented apps from building for platforms where this plugin was included, but not in use. +* Fixed a bug where errors about plugin initialization were printed when the plugin was included, but not in use. +* Fixed a bug where the EyeLevelSceneOrigin script behaved incorrectly when running in a scene without XR started. +* Fixed a bug where using this plugin with Unity 2022 and ARFoundation 5 would cause warnings. + +## [1.8.0] - 2023-04-04 + +* Removed Android related XR features for hand tracking and controller models. + * Recommend using the OpenXR plugin from Unity and Meta for these features. +* Upgraded the OpenXR remoting runtime to 2.9.1 release. + * Fixed a bug where ARPlaneManager does not report planes after some number of Unity PlayMode remoting sessions. + * Fixed a bug where anchors are not persisted after clearing the Anchor Store in Unity PlayMode remoting session. +* Fixed a bug where UWP project validation shows up incorrectly when OpenXR feature is disabled for HoloLens2. +* Fixed a bug where anchor creation failure leads to unnecessary spamming debugger logs. +* Fixed a bug that sometimes crashes an app on exiting due to a Unity's logging interface being used after released. +* Changed the XR validator rule based on the Unity versions with related bug fixes so that the HL2 apps can run properly without the "Run in Background" project settings. +* Fixed a bug where app remoting doesn't work when the user doesn't have permission to create regkey in HKCU. +* Fixed a bug where`AppRemoting.ReadyToStart` event is invoked incorrectly before `AppRemoting.StopListening` is used. +* Fixed a bug where some AppRemoting API methods were incorrectly enabled that should have been disabled + * When the app is used in Unity Editor with `Holographic Remoting for PlayMode` feature enabled in Unity Project Settings, AppRemoting API methods should be disabled. +* Fixed a bug where haptic binding is not available for HP Reverb G2. + +## [1.7.2] - 2023-03-03 + +* Fixed a bug where using some APIs from non-main threads could cause errors and incorrect behavior. + +## [1.7.1] - 2023-03-02 + +* Version 1.7.1 was packaged incorrectly and should be avoided. +* Version 1.7.2 is a replacement for 1.7.1. + +## [1.7.0] - 2022-12-15 + +* Fixed compatibility with the 1.6.0 release of Unity's OpenXR plugin. +* Fixed bugs where Unity application crashes due to unhandled exceptions in MR plugin. +* Fixed a bug that the Unity MR plugin was not properly cleaned up due to unbalanced module ref count. +* Added new API `AppRemoting.StartConnectingToPlayer` for connect mode app remoting. It replaces the deprecated `AppRemoting.Connect'. +* Added new API `AppRemoting.StartListeningForPlayer` for listen mode app remoting. It replaces the deprecated `AppRemoting.Listen'. +* Added new API `AppRemoting.StopListening` to stop listening on the remote app for incoming connections. +* Added new API `AppRemoting.IsReadyToStart` to indicate when app remoting is ready to be started. +* Added new APIs for secure mode app remoting connections, e.g. `AppRemoting.SecureRemotingConnectConfiguration` and `AppRemoting.SecureRemotingListenConfiguration` +* Added new events `AppRemoting.Connected`, `AppRemoting.Disconnecting`, and `AppRemoting.ReadyToStart` in addition to existing `AppRemoting.TryGetConnectionState` function. + +## [1.6.0] - 2022-11-02 + +* Depends on version 1.5.3 of Unity's OpenXR plugin. + * Fixed a bug where Holographic Remoting remote app may fail connection to remoting player +* Update the remoting OpenXR runtime to 2.8.1 release. + * Added support for XR_MSFT_spatial_anchor_export extension in remoting OpenXR runtime. + * Added better support for `SpatialGraphNode.FromStaticNodeId` in remoting OpenXR runtime. +* Added new dependency to com.unity.xr.core-utils package + * Changed project settings recommendation for HoloLens 2 to use [Unity's project validation system](https://docs.unity3d.com/Packages/com.unity.xr.core-utils@2.1/manual/project-validation.html) +* Added new API `AnchorConverter.CreateFromOpenXRHandle` for creating ARAnchor from OpenXR handle. +* Added new API `ViewConfiguration.StereoSeparationAdjustment` for adjusting the stereo separation on HoloLens 2. +* Supports running Holographic Remoting remote app in elevated process. +* Fixed a bug where remoting app build may fail if building into a non-standard exe name or building a Standalone build with "Create Visual Studio Solution" enabled. +* Fixed a bug in validator which incorrectly recommend disabling "Run in Background" settings. Enabling this setting can workaround a Unity bug so that Unity app can continue rendering when the app lost keyboard focus. +* Fixed a bug where the Mixed Reality OpenXR Plugin DLL wasn't being included in the build when specific features (Hand Tracking and Mixed Reality Features) weren't checked. + +## [1.5.1] - 2022-09-15 + +* Fixed a bug where apps may be deadlocked and stop rendering due to a race condition when using ARMeshManager to acquire meshes. + +## [1.5.0] - 2022-08-31 + +* Added new API `AppRemoting.TryLocateUserReferenceSpace` to locate the [XR_REMOTING_REFERENCE_SPACE_TYPE_USER_MSFT reference space](https://docs.microsoft.com/windows/mixed-reality/develop/native/holographic-remoting-coordinate-system-synchronization-openxr) in Unity's scene origin space in the remote app. +* The [ControllerModel](https://docs.microsoft.com/dotnet/api/microsoft.mixedreality.openxr.controllermodel) API now also supports loading Quest controller models. +* Fixed a bug where MeshProvider.AcquireMesh might crash in a rare race condition. +* Added warning message for the user to know that app remoting failure was due to the app running in elevated mode. +* Added new [`HandTracker.MotionRange`](https://docs.microsoft.com/dotnet/api/microsoft.mixedreality.openxr.handtracker.motionrange) API to support [hand joints motion range](https://registry.khronos.org/OpenXR/specs/1.0/html/xrspec.html#XR_EXT_hand_joints_motion_range). +* Fixed a bug where some new anchors were not tracked before their first update. +* Added a validator to remove "Run In Background" project setting for HoloLens 2 apps. +* Added a [SelectKeywordRecognizer](https://docs.microsoft.com/dotnet/api/microsoft.mixedreality.openxr.selectkeywordrecognizer) to allow developers to get notification when the "select" keyword is said on HoloLens 2. +* Added a new property [`ControllerModel.IsSupported`](https://docs.microsoft.com/dotnet/api/microsoft.mixedreality.openxr.controllermodel.issupported) to ControllerModel class. +* Renamed the API `AnchorProvider.FromPerceptionSpatialAnchor` to `AnchorProvider.CreateFromPerceptionSpatialAnchor` and the old method is now deprecated. +* Added support to [`CommonUsages.trackingState`](https://docs.unity3d.com/ScriptReference/XR.CommonUsages-trackingState.html) for hand tracking input device. +* Added support for Unity's [KeywordRecognizer](https://docs.unity3d.com/ScriptReference/Windows.Speech.KeywordRecognizer.html) over Holographic Remoting for Play Mode. +* Added new [`ControllerModelArticulator`](https://docs.microsoft.com/dotnet/api/microsoft.mixedreality.openxr.controllermodelarticulator) API for rendering controller model parts articulation. +* Fixed a potential null reference exception when calling the app remoting APIs on unsupported platforms. +* Removed dependency on Unity.Subsystem.Registration assembly. + +## [1.4.4] - 2022-08-08 + +* Fixed a bug caused by some runtimes reporting an active hand tracker while returning invalid hand joint poses. + +* Fixed a bug caused by some runtimes reporting an active hand tracker while returning invalid hand joint poses. + +## [1.4.3] - 2022-07-21 + +* Newly added ARAnchors will now report at least one update through the ARAnchorManager anchorsChanged event. + +## [1.4.2] - 2022-07-06 + +* Fixed a bug where the anchors are located incorrectly after user clear all holograms in user's settings page. +* Fixed a bug where rendering framerate might be dropped due to hand and controller tracking cost. +* Fixed a bug where ARPlanes might be incorrectly reported as updated or removed before they are reported as added. +* Added feature validation to warn developer about missing internet capabilities in appxmanifest for remoting apps. + +## [1.4.1] - 2022-06-07 + +* Depends on version 1.4.2 of Unity's OpenXR plugin. + * Fixed unnecessary destroying session on pause and resume. + * Fixed a bug where ARAnchor doesn't relocate properly after suspend and resume on HL2. + * Fixed an editor crash issue when updating OpenXR package version and then enter Playmode. +* Fixed a bug where AR Subsystems would clear trackables on subsystem stop/restart. Trackables will now only be cleared in this way on subsystem destroy/recreate. +* Fixed a bug where meshes provided through the XRMeshSubsystem could have the wrong 'updated' value. +* Fixed a bug where occlusion-optimized meshes were being computed at the wrong cadence. +* Fixed a bug where the app or Unity editor could crash when an OpenXR extension was not available. +* Fixed a bug where the XRAnchorSubsystem would not report anchors as removed when a remoting session was ended. +* Fixed a bug where the XRAnchorStore could not be loaded after a remoting session had disconnected and reconnected. +* Fixed a bug where loading an anchor from the XRAnchorStore multiple times would result in multiple ARAnchors. +* Upgraded the OpenXR remoting runtime to 2.8.0 release. + +## [1.4.0] - 2022-04-05 + +* Added SpatialGraphNode.FromDynamicNodeId() function to support interop with PV camera tracking. +* Added SpatialGraphNode.TryLocate(long qpcTime) function to support locating space at a historical time. +* Deprecated the "IDisposable" usage of GestureRecognizer, and replaced with "Destroy" function. +* Deprecated the MeshComputeSettings.MeshType property. +* Deprecated the XRAnchorStore.LoadAsync in favor of extension methods LoadAnchorStoreAsync. +* Fixed the ViewConfigurationType enum values to match OpenXR standard. +* Added a setting for application to choose MRC rendering between extra render pass with better hologram alignment using first person observer, or less render pass with better performance but compromise on hologram alignment. +* Support PlayMode remoting when the Unity project turned off "initialize XR at start up" setting, typically for Holographic remoting app. + * Note: ARFoundation trackable managers will not connect to XR subsystems if the trackable managers are active in the Unity scene before XR initialization. The application must disable then reenable these trackable managers after XR initialization, or wait to add active trackable managers to the scene until after XR initialization. +* Improved the performance by reducing the update events of ARAnchor and ARPlane when there's no location updates from the runtime. +* Upgraded the OpenXR remoting runtime to 2.7.5 release + +## [1.3.1] - 2022-02-16 + +* Fixed a bug where Unity editor sometimes crashes after upgrading the MR OpenXR plugin package. + +## [1.3.0] - 2022-02-09 + +* Fixed a bug where input system sometimes reports identity rotation for controller pose when the Hand Tracking feature was enabled. +* Enabled the "Hand Tracking feature" when it's used together with Unity's Oculus Quest feature. +* Fixed a crash on app resume when using plane finding. + +## [1.2.1] - 2021-12-03 + +* Depends on version 1.3.1 of Unity's OpenXR plugin. + * Fixed a bug where UWP remoting app won't render desktop view after XR session is started. + * Fixed a bug where a restart of XR session prevent future restart to happen. + * Fixed incorrect negative values on controller linear velocities. +* Fixed a bug that prevent UWP app to resume after suspend to background. + +## [1.2.0] - 2021-11-18 + +* Depends on version 1.3.0 of Unity's OpenXR plugin. + * Supports better HoloLens hand interaction action binding + * Fixed a crash during app suspend/resume when taking MRC video +* Depends on version 4.2.0 of XR management package. +* Fixed a bug where sometimes the the project settings assets are not created before being used. +* Added Microsoft.MixedReality.OpenXR.Remoting.AppRemoting.Listen function to support listen mode for a Holographic Remoting remote app. +* Added new enum value HandshakePermissionDenied to enum type RemotingDisconnectReason. +* Fixed a bug where after a failed remoting connection the XR session automatically restarted and repeat the failure. +* When hand tracking becomes untracked or out of view, the corresponding InputDevice for hand joints will remain valid and report `isTracked = false`, instead of invalidating the InputDevice. + +## [1.1.2] - 2021-10-27 + +* Fixed a bug where Unity Editor sometimes cannot quit after an unsuccessful connection to Holographic Remoting player in Play Mode. +* Update the OpenXR remoting runtime to 2.7.1 release. + +## [1.1.1] - 2021-10-15 + +* Fixed a bug where projects would fail to build if the project also referenced DotNetWinRT package. + +## [1.1.0] - 2021-10-07 + +* Added new APIs for spatial anchor transfer batch: Microsoft.MixedReality.OpenXR.XRAnchorTransferBatch +* Supports the XRMeshSubsystem through OpenXR scene understanding extensions. +* Supports OpenXR remoting runtime 2.7, with Spatial Anchor Store and Surface mapping in Holographic Remoting apps. +* Removed direct package dependency to ARSubsystem package. It's now implicit through ARFoundation package. +* Fixed a bug where Unity's UI froze briefly when Holographic Remoting failed to connect to remote player app. +* Fixed a bug where persisted anchor may lead to error saying "An item with the same key has already been added." +* Supports the ratified "XR_MSFT_scene_understanding" extension instead of "_preview3" version. +* Fixed a bug where projects would fail to build with Windows XR Plugin installed and app remoting enabled. + +## [1.0.3] - 2021-09-07 + +* Supports the OpenXR spatial anchor persistence MSFT extension. +* Fixed a bug where Editor Remoting settings are present in PackageSettings instead of UserSettings. +* Fixed a bug where some anchors could fail to be persisted after clearing the XRAnchorStore. +* Fixed a bug where extra anchors were created when switching between Unity scenes. + +## [1.0.2] - 2021-08-05 + +* Depends on Unity's 1.2.8 OpenXR plugin. +* Fixed a bug where ARAnchors were occasionally not removed properly. +* Fixed a bug where invalid ARAnchor changes were occasionally reported after restarting Holographic Remoting for Play Mode. +* Fixed a bug where view configurations were not properly reported when using Holographic Remoting for Play Mode. +* Added more specific settings validation with more precise messages when using Holographic Remoting for Play Mode. +* Added validation for "Initialize XR on Startup" setting when using Holographic Remoting for Play Mode. + +## [1.0.1] - 2021-07-13 + +* Depends on Unity's 1.2.3 OpenXR plugin. +* Updated Holographic Remoting runtime to 2.6.0 +* Removed the "Holographic Remoting for Play Mode" feature group from Unity settings UX and kept the feature independent. +* Fixed a bug where build process cannot find the app.cpp when building a XAML type unity project. + +## [1.0.0] - 2021-06-18 + +* Fixed a bug where a the XRAnchorSubsystem was always started on app start regardless ARAnchorManager's present. +* Fixed a bug where the reprojection mode didn't work properly. + +## [1.0.0-preview.2] - 2021-06-14 + +* Depends on Unity's 1.2.2 OpenXR plugin. +* Changed Holographic Remoting features in to individual feature groups. +* Fixed a bug where "Apply HoloLens 2 project settings" changes project color space. This is no longer needed after Unity OpenXR 1.2.0 plugin. +* Fixed a bug where a input device get connected without disconnect after application suspended and resumed. +* Added support for detecting plugin and current tracking states via ARSession. +* Fixed a bug where the "AR Default Plane" ARFoundation prefab wouldn't be visible. + +## [1.0.0-preview.1] - 2021-06-02 + +* Supports OpenXR scene understanding MSFT extensions instead of preview extensions. +* Plane detection on HoloLens 2 no longer requires preview versions of the Mixed Reality OpenXR runtimes. + +## [0.9.5] - 2021-05-21 + +* Depends on Unity's 1.2.0 OpenXR Plugin +* Adapted to the new feature UI (in OpenXR Plugin 1.2.0) for configuration. +* Fixed a bug where the locatable camera provider wasn't properly unregistering. +* Cleaned up some extra usages of `[Preserve]`. +* Update "HP Reverb G2 Controller (OpenXR)" name in the input system UI. + +## [0.9.4] - 2021-05-20 + +* Depends on Unity's 1.2.0 OpenXR Plugin. +* Added new C# API to get motion controller glTF model. +* Added new C# API to get enabled view configurations and set reprojection settings. +* Added new C# API to set additional settings for computing meshes with XRMeshSubsystem. +* Added new C# API to configure and subscribe to gesture recognition events. +* Added Windows->XR->Editor Remoting settings dialog. +* Added ARM support for HoloLens UWP applications. + +## [0.9.3] - 2021-04-29 + +* Fixed a bug where Holographic remoting connection is not reliable +* Fixed a bug where the VR rendering performance is sub-optimum after upgrade to Unity's 1.1.1 OpenXR plugin. + +## [0.9.2] - 2021-04-21 + +* Plane detection on HoloLens 2 in plugin version 0.9.1 will work with version 105 of the Mixed Reality OpenXR preview runtime. +* Plane detection on HoloLens 2 in plugin version 0.9.2 will work with version 106 of the Mixed Reality OpenXR preview runtime. +* Removed some unused callbacks from InputProvider to prevent calls like XRInputSubsystem.GetTrackingOriginMode (which aren't managed by our input system) from returning success with misleading values. +* Split out deprecated version of XRAnchorStore into its own file to prevent Unity console warning. + +## [0.9.1] - 2021-04-20 + +* Depends on Unity's 1.1.1 OpenXR Plugin. +* Added support for [Holographic Remoting application](https://aka.ms/openxr-unity-app-remoting) for UWP platform. +* Fix UnityException where XRAnchorStore was trying to get a settings instance outside the main thread. + +## [0.9.0] - 2021-03-29 + +* Added support for spatial mapping via XRMeshSubsystem and ARMeshManager. +* Added new C# API to get OpenXR handles to support other Unity packages consumes OpenXR extensions. +* Added new C# API to interop with Windows.Perception APIs to support other Unity packages consuming Perception WinRT APIs. +* Removed interaction profiles from required features in Windows Mixed Reality feature set, so developers can choose the motion controllers they tested with. +* Added Holographic editor remoting feature validator to help users to setup editor remoting properly. +* Fixed a bug where Unity editor crashes when exiting Holographic editor remoting mode after connection failure. +* Fixed a bug where unpremultipled alpha textures leads to sub-optimum performance on HoloLens 2. +* Fixed a bug where hand tracking was not located correctly when the scene origin was at floor level. +* Fixed a bug where hand mesh tracking disappear after leaving and loading a new scene. +* Fixed a bug where locatable camera provider didn't properly clean up. +* Revised the namespace of XRAnchorStore API into Microsoft.MixedReality.OpenXR. + +## [0.2.0] - 2021-03-24 + +* Depends on Unity's 1.0.3 OpenXR Plugin. +* Removed deprecated preview APIs. +* Supports new API "EyeLevelSceneOrigin" for easily setup eye level experience for HoloLens 2. +* Supports plane detection using the ARPlaneSubsystem on HoloLens 2. +* Supports single raycasts for planes using the ARRaycastSubsystem on HoloLens 2. +* Supports new HandMeshTracker API for hand mesh tracking inputs on HoloLens 2. +* Fixed a bug where ARAnchor is not properly reporting tracking state. + +## [0.1.5] - 2021-03-15 + +* Fixed a bug where using an HP Reverb G2 controller lead to errors in the Unity plugin. +* Fixed a bug that the Unity's "Input Debugger" window is blank when using Mixed Reality plugin. + +## [0.1.4] - 2021-03-02 + +* Depends on Unity's 1.0.2 OpenXR Plugin. +* Fixed a bug where SpatialGraphNode's TryLocateSpace's FrameTime parameter was ignored. +* Fixed a bug where hand tracking could occasionally cause a crash. + +## [0.1.3] - 2021-02-11 + +* Adds support for [desktop app holographic remoting](https://aka.ms/openxr-unity-app-remoting). +* Adds support for "SpatialGraphNode" API that bridges to other Mixed Reality tracking libraries, such as QR code tracking. +* Promote "FrameTime" concept from Preview API to supported API. +* Fixed a bug where eye tracking device capability is duplicated in manifest file. +* Fixed a bug where the plugin doesn't compile in Unity 2021.1+. + +## [0.1.2] - 2021-01-08 + +* Depends on Unity's 0.1.2-preview.2 +* Fixed unnecessary error message in XRAnchorStore before XR plugin is initialized. +* Fixed a bug where HandTracker's `TryLocateHandJoints` method might throw a `DllNotFoundException` if the DLL wasn't properly loaded. It now returns `false` instead. + +## [0.1.1] - 2020-12-18 + +* Fixed a bug where non-existent sources were being reported disconnected on shutdown, possibly causing errors. +* Fixed a bug that the menu button on HP Reverb G2 didn't bind correctly. +* Changed the SetSceneOrigin script to focus on overriding eye level experience instead. +* Fixed a bug that returns incorrect room boundary on Mixed Reality headset. +* Fixed a bug where sample scene anchor scenarios didn't work with ARFoundation before 4.1.1. + +## [0.1.0] - 2020-12-16 + +### Initial release + +This is initial release of *Mixed Reality OpenXR Plugin \*. + +* Supports both UWP applications for HoloLens 2 and Win32 VR applications for Windows Mixed Reality headsets. +* Optimizes UWP package and CoreWindow interaction for HoloLens 2 applications. +* Supports motion controller and hand interactions, including the new HP Reverb G2 controller. +* Supports articulated hand tracking using 26 joints and joint radius inputs. +* Supports eye gaze interaction on HoloLens 2. +* Supports locating PV camera on HoloLens 2. +* Supports mixed reality capture using 3rd eye rendering through PV camera. +* Supports "Play" to HoloLens 2 using Holographic Remoting app, allow developers to debug scripts without build and deploy to the device. +* Compatible with MRTK Unity 2.5.2 through MRTK OpenXR adapter package. diff --git a/com.microsoft.mixedreality.openxr/CHANGELOG.md.meta b/com.microsoft.mixedreality.openxr/CHANGELOG.md.meta new file mode 100644 index 0000000..0c10852 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/CHANGELOG.md.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: a4ea1fbe6b269e3408e0189f6ac7da45 +TextScriptImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Documentation~/README.md b/com.microsoft.mixedreality.openxr/Documentation~/README.md new file mode 100644 index 0000000..4455343 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Documentation~/README.md @@ -0,0 +1,13 @@ +# Mixed Reality OpenXR Plugin for Unity + +This "Mixed Reality OpenXR Plugin" package is an extension to Unity's "OpenXR Plugin" +to support a suite of features for HoloLens 2 and Windows Mixed Reality headsets + +This package requires Unity 2021.3+ and the "OpenXR Plugin" package from Unity. + +Please reference [online documents](https://aka.ms/openxr-unity) to learn more details +about setting up a Unity project and using this plugin to build Unity applications +for HoloLens 2 and Windows Mixed Reality headsets. + +Please reference [OpenXR-Unity-MixedReality-Samples](https://github.com/microsoft/OpenXR-Unity-MixedReality-Samples) +to find sample projects using this package to build Unity applications on HoloLens 2. diff --git a/com.microsoft.mixedreality.openxr/Editor.meta b/com.microsoft.mixedreality.openxr/Editor.meta new file mode 100644 index 0000000..d8a46ff --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 822a5747a975a80449ceb592310add2b +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/AssemblyInfo.cs b/com.microsoft.mixedreality.openxr/Editor/AssemblyInfo.cs new file mode 100644 index 0000000..4d70494 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/AssemblyInfo.cs @@ -0,0 +1,9 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System.Reflection; +using System.Runtime.CompilerServices; + +[assembly: InternalsVisibleTo("Microsoft.MixedReality.OpenXR.Internal.Editor")] + +[assembly: AssemblyVersion("1.11.1")] diff --git a/com.microsoft.mixedreality.openxr/Editor/AssemblyInfo.cs.meta b/com.microsoft.mixedreality.openxr/Editor/AssemblyInfo.cs.meta new file mode 100644 index 0000000..a065664 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/AssemblyInfo.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c40b7f5aac6defb49823fda257e5ea40 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/BuildProcessors.meta b/com.microsoft.mixedreality.openxr/Editor/BuildProcessors.meta new file mode 100644 index 0000000..b879b42 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/BuildProcessors.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 88556967a2633cf44afc6adf3e8c5fa8 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MetaBuildProcessor.cs b/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MetaBuildProcessor.cs new file mode 100644 index 0000000..76266c4 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MetaBuildProcessor.cs @@ -0,0 +1,150 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Xml; +using UnityEditor.Build.Reporting; +using UnityEditor.XR.OpenXR.Features; +using static Microsoft.MixedReality.OpenXR.Editor.BuildProcessorHelpers; +using static Microsoft.MixedReality.OpenXR.Editor.BuildProcessorHelpers.AndroidManifest; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + internal class MetaBuildProcessor : OpenXRFeatureBuildHooks + { + public override int callbackOrder => 1; + + public override Type featureType => +#if UNITY_OPENXR_1_6_OR_NEWER + typeof(UnityEngine.XR.OpenXR.Features.MetaQuestSupport.MetaQuestFeature); +#else + typeof(UnityEngine.XR.OpenXR.Features.OculusQuestSupport.OculusQuestFeature); +#endif + + protected override void OnPreprocessBuildExt(BuildReport report) { } + + protected override void OnPostGenerateGradleAndroidProjectExt(string path) + { + HandTrackingFeaturePlugin handTrackingFeaturePlugin = GetOpenXRFeature(); + bool handTrackingEnabled = handTrackingFeaturePlugin != null && handTrackingFeaturePlugin.enabled; + bool motionControllerModelEnabled = IsFeatureEnabled(); + + if (!handTrackingEnabled && !motionControllerModelEnabled) + { + return; + } + + AndroidManifest androidManifest = new AndroidManifest(GetManifestPath(path)); + + if (handTrackingEnabled) + { + androidManifest.EnsurePermission("com.oculus.permission.HAND_TRACKING"); + androidManifest.EnsureFeature("oculus.software.handtracking", false); + + if (handTrackingFeaturePlugin.QuestHandTrackingMode == HandTrackingFeaturePlugin.QuestHandTracking.v2) + { + androidManifest.EnsureMetaData("com.oculus.handtracking.version", "V2.0"); + } + else if (handTrackingFeaturePlugin.QuestHandTrackingMode == HandTrackingFeaturePlugin.QuestHandTracking.v1) + { + androidManifest.EnsureMetaData("com.oculus.handtracking.version", "V1.0"); + } + } + + if (motionControllerModelEnabled) + { + androidManifest.EnsurePermission("com.oculus.permission.RENDER_MODEL"); + androidManifest.EnsureFeature("com.oculus.feature.RENDER_MODEL"); + } + + androidManifest.Save(); + } + + protected override void OnPostprocessBuildExt(BuildReport report) { } + } + + internal static class AndroidManifestExtensions + { + internal static void EnsurePermission(this AndroidManifest manifest, string permissionString) + { + XmlNode usesPermission = null; + foreach (XmlNode child in manifest.RootElement.ChildNodes) + { + if (child.Name == "uses-permission" && + HasAttribute(child, "android:name", permissionString)) + { + usesPermission = child; + + if (usesPermission != null) + { + break; + } + } + } + + if (usesPermission == null) + { + usesPermission = manifest.RootElement.AppendChild(manifest.CreateElement("uses-permission")); + usesPermission.Attributes.Append(manifest.CreateAndroidAttribute("name", permissionString)); + } + } + + internal static void EnsureFeature(this AndroidManifest manifest, string featureString, bool? required = null) + { + XmlNode usesFeature = null; + foreach (XmlNode child in manifest.RootElement.ChildNodes) + { + if (child.Name == "uses-feature" && + HasAttribute(child, "android:name", featureString)) + { + usesFeature = child; + + if (usesFeature != null) + { + break; + } + } + } + + if (usesFeature == null) + { + usesFeature = manifest.RootElement.AppendChild(manifest.CreateElement("uses-feature")); + usesFeature.Attributes.Append(manifest.CreateAndroidAttribute("name", featureString)); + } + + if (required.HasValue && !SetAttribute(usesFeature, "android:required", required.Value.ToString())) + { + usesFeature.Attributes.Append(manifest.CreateAndroidAttribute("required", required.Value.ToString())); + } + } + + internal static void EnsureMetaData(this AndroidManifest manifest, string nameString, string valueString) + { + XmlNode metaData = null; + foreach (XmlNode child in manifest.ApplicationElement.ChildNodes) + { + if (child.Name == "meta-data" && + HasAttribute(child, "android:name", nameString)) + { + metaData = child; + + if (metaData != null) + { + break; + } + } + } + + if (metaData == null) + { + metaData = manifest.ApplicationElement.AppendChild(manifest.CreateElement("meta-data")); + metaData.Attributes.Append(manifest.CreateAndroidAttribute("name", nameString)); + } + + if (!SetAttribute(metaData, "android:value", valueString)) + { + metaData.Attributes.Append(manifest.CreateAndroidAttribute("value", valueString)); + } + } + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MetaBuildProcessor.cs.meta b/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MetaBuildProcessor.cs.meta new file mode 100644 index 0000000..6ccc2bc --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MetaBuildProcessor.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e33bc45f8c21ce746aa8116a3318d611 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MixedRealityBuildProcessor.cs b/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MixedRealityBuildProcessor.cs new file mode 100644 index 0000000..50fd277 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MixedRealityBuildProcessor.cs @@ -0,0 +1,399 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR + +using Microsoft.MixedReality.OpenXR.Remoting; +using System; +using System.Collections.Generic; +using System.IO; +using System.Xml; +using System.Xml.Linq; +using UnityEditor; +using UnityEditor.Build.Reporting; +using UnityEditor.XR.OpenXR.Features; +using UnityEngine; +using UnityEngine.XR.OpenXR.Features; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + // Customize code for UWP (aka WSA platform) + // Enable the "holographic window attachment" and slightly modify the app.cpp file in app. + internal class MixedRealityBuildProcessor : OpenXRFeatureBuildHooks + { + private const string MixedRealityPluginName = "MicrosoftOpenXRPlugin.dll"; + private const string PerceptionDeviceName = "PerceptionDevice.dll"; + private const string RemotingRuntimeName = "Microsoft.Holographic.AppRemoting.OpenXr.dll"; + private const string RemotingSceneUnderstandingName = "Microsoft.Holographic.AppRemoting.OpenXr.SU.dll"; + + private const string RemotingJsonName = "RemotingXR.json"; + private const string RemotingJsonGuid = "db1217138e9d063459fa78b3e75f4f93"; + private const string OpenXR = "com.microsoft.mixedreality.openxr"; + private const string SamplesRepoURL = "https://github.com/microsoft/OpenXR-Unity-MixedReality-Samples"; + + private static readonly Dictionary BootVars = new Dictionary() + { + {"force-primary-window-holographic", "1"}, + {"vr-enabled", "1"}, + {"xrsdk-windowsmr-library", NativeLib.DllName + ".dll"}, + {"early-boot-windows-holographic", "1"}, + }; + + public override void OnPreprocessBuild(BuildReport report) + { + base.OnPreprocessBuild(report); + + PluginImporter[] allPlugins = PluginImporter.GetAllImporters(); + foreach (PluginImporter plugin in allPlugins) + { + if (plugin.isNativePlugin && plugin.assetPath.Contains(OpenXR)) + { + if (plugin.assetPath.Contains(MixedRealityPluginName)) + { + plugin.SetIncludeInBuildDelegate(IsMixedRealityNativePluginRequired); + } + else if (plugin.assetPath.Contains(RemotingRuntimeName)) + { + plugin.SetIncludeInBuildDelegate(IsRemotingRuntimeRequired); + } + else if (plugin.assetPath.Contains(RemotingSceneUnderstandingName)) + { + plugin.SetIncludeInBuildDelegate(IsRemotingRuntimeRequired); + } + else if (plugin.assetPath.Contains(PerceptionDeviceName)) + { + plugin.SetIncludeInBuildDelegate(IsRemotingRuntimeRequired); + } + } + } + } + + protected override void OnPreprocessBuildExt(BuildReport report) + { + if (report.summary.platformGroup == BuildTargetGroup.WSA) + { + PreprocessBuildForWSA(report); + } + } + + private void PreprocessBuildForWSA(BuildReport report) + { + // Write boot settings before build + BootConfig bootConfig = new BootConfig(report); + bootConfig.ReadBootConfig(); + + foreach (KeyValuePair entry in BootVars) + { + if (entry.Key == "force-primary-window-holographic" && IsRemotingRuntimeRequired()) + { + // When AppRemoting is enabled, skip the flag to force primary corewindow to be holographic (it won't be). + // If this flag exist, Unity might hit a bug that it skips rendering into the CoreWindow on the desktop. + continue; + + } + bootConfig.SetValueForKey(entry.Key, entry.Value); + } + + bootConfig.WriteBootConfig(); + } + + protected override void OnPostprocessBuildExt(BuildReport report) + { + if (report.summary.platformGroup == BuildTargetGroup.WSA) + { + PostprocessBuildForWSA(report); + } + + CheckRemotingJson(report, IsRemotingRuntimeRequired()); + } + + private void PostprocessBuildForWSA(BuildReport report) + { + // Clean up boot settings after build + BootConfig bootConfig = new BootConfig(report); + bootConfig.ReadBootConfig(); + + foreach (KeyValuePair entry in BootVars) + { + bootConfig.ClearEntryForKeyAndValue(entry.Key, entry.Value); + } + + bootConfig.WriteBootConfig(); + + if (IsRemotingRuntimeRequired()) + { + AddRemotingJsonToData(Path.Combine(report.summary.outputPath, PlayerSettings.productName)); + VerifyPackageManifestCapabilities(Path.Combine(report.summary.outputPath, PlayerSettings.productName)); + } + + RemoveSuppressSystemOverlays(report); + } + + /// + /// Copies RemotingXR.json to or deletes from the build folder, depending on shouldExist. + /// + /// The build report from Unity's build hooks events. + /// If the file should be present in the build. + private void CheckRemotingJson(BuildReport report, bool shouldExist) + { + string path = report.summary.outputPath; + + if (report.summary.platform == BuildTarget.WSAPlayer) + { + path = Path.Combine(path, PlayerSettings.productName, RemotingJsonName); + } + else if (report.summary.platform == BuildTarget.StandaloneWindows64) + { + string folderName = Path.GetFileNameWithoutExtension(report.summary.outputPath); + + // When building with "Create Visual Studio Solution", this API still reports a .exe + // in the output path (but it doesn't exist, so we can assume we're building a .sln) + if (path.EndsWith(".exe") && !File.Exists(path)) + { + path = Path.Combine(path, "..", "build", "bin"); + } + else + { + path = Path.Combine(path, ".."); + } + + path = Path.Combine(path, folderName + "_Data", "Plugins", "x86_64", RemotingJsonName); + } + else + { + // Other platforms aren't supported + return; + } + + string folderPath = Directory.GetParent(path).FullName; + if (!Directory.Exists(folderPath)) + { + Debug.LogError($"The {nameof(MixedRealityBuildProcessor)} could not find the Plugins folder (looked in {folderPath}).\n" + + $"Please file an issue on {SamplesRepoURL}, as this is unexpected. Holographic Remoting may not work as a result."); + } + else if (shouldExist) + { + File.Copy(Path.GetFullPath(AssetDatabase.GUIDToAssetPath(RemotingJsonGuid)), path, true); + } + else if (File.Exists(path)) + { + File.Delete(path); + } + } + + /// + /// Adds a line in the Unity data file project to include the remoting file in the build. + /// + /// The path to the folder that contains Unity Data.vcxitems. + private void AddRemotingJsonToData(string path) + { + const string UnityDataPath = "Unity Data.vcxitems"; + path = Path.Combine(path, UnityDataPath); + + XElement root = XElement.Load(path); + foreach (XElement itemGroup in root.Elements(root.GetDefaultNamespace() + "ItemGroup")) + { + foreach (XElement remotingDll in itemGroup.Elements(root.GetDefaultNamespace() + "None")) + { + XAttribute includeDll = remotingDll.Attribute("Include"); + if (includeDll != null && includeDll.Value.Contains(RemotingRuntimeName)) + { + XElement jsonElement = new XElement(remotingDll); + // Update "Include" to point to the json, but leave "Condition" alone so it's still dependent on the remoting binary existing + jsonElement.Attribute("Include").Value = jsonElement.Attribute("Include").Value.Replace(RemotingRuntimeName, RemotingJsonName); + itemGroup.Add(jsonElement); + root.Save(path); + return; + } + } + } + } + + /// + /// Verifies if the internet capabilities in Package.appxmanifest match the ones in the Unity Player Settings for Holographic App remoting to work properly. + /// + /// The path to the folder that contains Package.appxmanifest. + private void VerifyPackageManifestCapabilities(string path) + { + bool internetClientEnabled = false, internetClientServerEnabled = false, privateNetworkClientServerEnabled = false, microphone = false; + + const string PackageManifestPath = "Package.appxmanifest"; + path = Path.Combine(path, PackageManifestPath); + XmlDocument manifest = new XmlDocument(); + manifest.Load(path); + XmlNode root = manifest.DocumentElement; + + // Get the internet capabilities that are enabled from manifest + foreach (XmlNode childNode in root.ChildNodes) + { + if (childNode.Name == "Capabilities") + { + foreach (XmlNode capability in childNode.ChildNodes) + { + if (capability.Name == "Capability") + { + foreach (XmlAttribute attribute in capability.Attributes) + { + if (attribute.Name == "Name" && attribute.Value == "internetClient") + { + internetClientEnabled = true; + } + if (attribute.Name == "Name" && attribute.Value == "internetClientServer") + { + internetClientServerEnabled = true; + } + if (attribute.Name == "Name" && attribute.Value == "privateNetworkClientServer") + { + privateNetworkClientServerEnabled = true; + } + if (attribute.Name == "Name" && attribute.Value == "microphone") + { + microphone = true; + } + } + } + } + } + } + + // Generate a warning, if the capabilities in the manifest don't match the ones in Unity Player Settings + if (PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.InternetClient) != internetClientEnabled) + { + Debug.LogWarning("The InternetClient capability in the existing Package.appxmanifest does not match the capability in this project's Player settings." + + " To update the existing Package.appxmanifest to reflect your project's current Player settings, please edit the file manually" + + " or regenerate the Package.appxmanifest."); + } + if (PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.InternetClientServer) != internetClientServerEnabled) + { + Debug.LogWarning("The InternetClientServer capability in the existing Package.appxmanifest does not match the capability in this project's Player settings." + + " To update the existing Package.appxmanifest to reflect your project's current Player settings, please edit the file manually" + + " or regenerate the Package.appxmanifest."); + } + if (PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.PrivateNetworkClientServer) != privateNetworkClientServerEnabled) + { + Debug.LogWarning("The PrivateNetworkClientServer capability in the existing Package.appxmanifest does not match the capability in this project's Player settings." + + " To update the existing Package.appxmanifest to reflect your project's current Player settings, please edit the file manually" + + " or regenerate the Package.appxmanifest."); + } + if (PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.Microphone) != microphone) + { + Debug.LogWarning("The Microphone capability in the existing Package.appxmanifest does not match the capability in this project's Player settings." + + " To update the existing Package.appxmanifest to reflect your project's current Player settings, please edit the file manually" + + " or regenerate the Package.appxmanifest."); + } + + } + + // Remove the deprecated usage of SuppressSystemOverlays from Unity UWP project template. + private void RemoveSuppressSystemOverlays(BuildReport report) + { + string appCppPath = Path.Combine(report.summary.outputPath, PlayerSettings.productName, "App.cpp"); + + // For certain types of builds, like XAML builds, App.cpp won't exist. App.xaml.cpp does though. + if (!File.Exists(appCppPath)) + { + appCppPath = Path.Combine(report.summary.outputPath, PlayerSettings.productName, "App.xaml.cpp"); + } + + string appCppLines = File.ReadAllText(appCppPath); + const string Pattern = @"\r?\n.*SuppressSystemOverlays.*\r?\n"; + string modifiedAppCppLines = System.Text.RegularExpressions.Regex.Replace(appCppLines, Pattern, ""); + File.WriteAllText(appCppPath, modifiedAppCppLines); + } + + private static bool IsRemotingRuntimeRequired(string path = "") => BuildProcessorHelpers.IsFeatureEnabled(); + private static bool IsMixedRealityNativePluginRequired(string path = "") + { + if (IsRemotingRuntimeRequired(path)) + { + return true; + } + + foreach (OpenXRFeature feature in BuildProcessorHelpers.GetOpenXRFeatures()) + { + if (feature.IsValidAndEnabled() && Attribute.IsDefined(feature.GetType(), typeof(RequiresNativePluginDLLsAttribute))) + { + return true; + } + } + + return false; + } + + /// + /// Small utility class for reading, updating and writing boot config. + /// + private class BootConfig + { + private const string XrBootSettingsKey = "xr-boot-settings"; + + private readonly Dictionary bootConfigSettings; + private readonly string buildTargetName; + + public BootConfig(BuildReport report) + { + bootConfigSettings = new Dictionary(); + buildTargetName = BuildPipeline.GetBuildTargetName(report.summary.platform); + } + + public void ReadBootConfig() + { + bootConfigSettings.Clear(); + + string xrBootSettings = EditorUserBuildSettings.GetPlatformSettings(buildTargetName, XrBootSettingsKey); + if (!string.IsNullOrEmpty(xrBootSettings)) + { + // boot settings string format + // :[;:]* + var bootSettings = xrBootSettings.Split(';'); + foreach (var bootSetting in bootSettings) + { + var setting = bootSetting.Split(':'); + if (setting.Length == 2 && !string.IsNullOrEmpty(setting[0]) && !string.IsNullOrEmpty(setting[1])) + { + bootConfigSettings.Add(setting[0], setting[1]); + } + } + } + } + + public void SetValueForKey(string key, string value) => bootConfigSettings[key] = value; + + public void ClearEntryForKeyAndValue(string key, string value) + { + if (bootConfigSettings.TryGetValue(key, out string dictValue) && dictValue == value) + { + bootConfigSettings.Remove(key); + } + } + + public void WriteBootConfig() + { + // boot settings string format + // :[;:]* + bool firstEntry = true; + var sb = new System.Text.StringBuilder(); + foreach (var kvp in bootConfigSettings) + { + if (!firstEntry) + { + sb.Append(";"); + } + sb.Append($"{kvp.Key}:{kvp.Value}"); + firstEntry = false; + } + + EditorUserBuildSettings.SetPlatformSettings(buildTargetName, XrBootSettingsKey, sb.ToString()); + } + } + + public override int callbackOrder => 1; + public override Type featureType => typeof(MixedRealityFeaturePlugin); + + protected override void OnPostGenerateGradleAndroidProjectExt(string path) + { + } + } +} + +#endif \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MixedRealityBuildProcessor.cs.meta b/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MixedRealityBuildProcessor.cs.meta new file mode 100644 index 0000000..729d83b --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/BuildProcessors/MixedRealityBuildProcessor.cs.meta @@ -0,0 +1,3 @@ +fileFormatVersion: 2 +guid: e6ccea71264b4942ab62948680084dbb +timeCreated: 1590606496 \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Editor/FeatureSets.meta b/com.microsoft.mixedreality.openxr/Editor/FeatureSets.meta new file mode 100644 index 0000000..7fcc3e1 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/FeatureSets.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: b20cca0dd54f8174997d18765356a110 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/FeatureSets/AppRemotingFeatureSet.cs b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/AppRemotingFeatureSet.cs new file mode 100644 index 0000000..96c5eb1 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/AppRemotingFeatureSet.cs @@ -0,0 +1,38 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using Microsoft.MixedReality.OpenXR.Remoting; +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + [OpenXRFeatureSet( + FeatureSetId = featureSetId, + FeatureIds = new string[] + { + AppRemotingPlugin.featureId, + MixedRealityFeaturePlugin.featureId, + HandTrackingFeaturePlugin.featureId, + }, + RequiredFeatureIds = new string[] + { + AppRemotingPlugin.featureId, + MixedRealityFeaturePlugin.featureId, + }, + DefaultFeatureIds = new string[] + { + AppRemotingPlugin.featureId, + MixedRealityFeaturePlugin.featureId, + HandTrackingFeaturePlugin.featureId, + }, + UiName = "Holographic Remoting remote app", + // This will appear as a tooltip for the (?) icon in the loader UI. + Description = "Enable the Holographic Remoting remote app features.", + SupportedBuildTargets = new BuildTargetGroup[] { BuildTargetGroup.Standalone, BuildTargetGroup.WSA } + )] + sealed class AppRemotingFeatureSet + { + internal const string featureSetId = "com.microsoft.openxr.featureset.appremoting"; + } +} diff --git a/com.microsoft.mixedreality.openxr/Editor/FeatureSets/AppRemotingFeatureSet.cs.meta b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/AppRemotingFeatureSet.cs.meta new file mode 100644 index 0000000..eab4f66 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/AppRemotingFeatureSet.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6cf0805250c68bd44a6e3268dfffc641 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/FeatureSets/HoloLensFeatureSet.cs b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/HoloLensFeatureSet.cs new file mode 100644 index 0000000..762cef6 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/HoloLensFeatureSet.cs @@ -0,0 +1,36 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + [OpenXRFeatureSet( + FeatureSetId = featureSetId, + FeatureIds = new string[] + { + MixedRealityFeaturePlugin.featureId, + HandTrackingFeaturePlugin.featureId, + MotionControllerFeaturePlugin.featureId, + }, + RequiredFeatureIds = new string[] + { + MixedRealityFeaturePlugin.featureId, + }, + DefaultFeatureIds = new string[] + { + MixedRealityFeaturePlugin.featureId, + HandTrackingFeaturePlugin.featureId, + MotionControllerFeaturePlugin.featureId, + }, + UiName = "Microsoft HoloLens", + // This will appear as a tooltip for the (?) icon in the loader UI. + Description = "Enable the full suite of features for Microsoft HoloLens 2.", + SupportedBuildTargets = new BuildTargetGroup[] { BuildTargetGroup.WSA } + )] + sealed class HoloLensFeatureSet + { + internal const string featureSetId = "com.microsoft.openxr.featureset.hololens"; + } +} diff --git a/com.microsoft.mixedreality.openxr/Editor/FeatureSets/HoloLensFeatureSet.cs.meta b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/HoloLensFeatureSet.cs.meta new file mode 100644 index 0000000..643314e --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/HoloLensFeatureSet.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 52029f146b94c2c4ebdf267f82595a4a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/FeatureSets/WMRFeatureSet.cs b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/WMRFeatureSet.cs new file mode 100644 index 0000000..9da99e5 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/WMRFeatureSet.cs @@ -0,0 +1,36 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + [OpenXRFeatureSet( + FeatureSetId = featureSetId, + FeatureIds = new string[] + { + MixedRealityFeaturePlugin.featureId, + MotionControllerFeaturePlugin.featureId, + HandTrackingFeaturePlugin.featureId, + }, + RequiredFeatureIds = new string[] + { + MixedRealityFeaturePlugin.featureId + }, + DefaultFeatureIds = new string[] + { + MixedRealityFeaturePlugin.featureId, + MotionControllerFeaturePlugin.featureId, + HandTrackingFeaturePlugin.featureId, + }, + UiName = "Windows Mixed Reality", + // This will appear as a tooltip for the (?) icon in the loader UI. + Description = "Enable the full suite of features for Windows Mixed Reality headsets.", + SupportedBuildTargets = new BuildTargetGroup[] { BuildTargetGroup.Standalone } + )] + sealed class WMRFeatureSet + { + internal const string featureSetId = "com.microsoft.openxr.featureset.wmr"; + } +} diff --git a/com.microsoft.mixedreality.openxr/Editor/FeatureSets/WMRFeatureSet.cs.meta b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/WMRFeatureSet.cs.meta new file mode 100644 index 0000000..ca97cc6 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/FeatureSets/WMRFeatureSet.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 35604464464fafe47b8f22e922fdf1ec +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/Inspectors.meta b/com.microsoft.mixedreality.openxr/Editor/Inspectors.meta new file mode 100644 index 0000000..fb839c1 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Inspectors.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 69905536a2c895b4d9f370188e743dc8 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/Inspectors/EyeLevelSceneOriginInspector.cs b/com.microsoft.mixedreality.openxr/Editor/Inspectors/EyeLevelSceneOriginInspector.cs new file mode 100644 index 0000000..1a880f1 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Inspectors/EyeLevelSceneOriginInspector.cs @@ -0,0 +1,17 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEditor; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ +#pragma warning disable CS0618 + [CustomEditor(typeof(EyeLevelSceneOrigin))] + internal class EyeLevelSceneOriginInspector : UnityEditor.Editor + { + public override void OnInspectorGUI() + { + EditorGUILayout.HelpBox("Enable this override if the XR experience assumes the scene origin at eye level.", MessageType.Info); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Editor/Inspectors/EyeLevelSceneOriginInspector.cs.meta b/com.microsoft.mixedreality.openxr/Editor/Inspectors/EyeLevelSceneOriginInspector.cs.meta new file mode 100644 index 0000000..cef8a85 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Inspectors/EyeLevelSceneOriginInspector.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c975b3665c7ed4d468e229dcd90c1eac +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/Inspectors/PlayModeRemotingPluginInspector.cs b/com.microsoft.mixedreality.openxr/Editor/Inspectors/PlayModeRemotingPluginInspector.cs new file mode 100644 index 0000000..75266fd --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Inspectors/PlayModeRemotingPluginInspector.cs @@ -0,0 +1,33 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using Microsoft.MixedReality.OpenXR.Remoting; +using UnityEditor; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + [CustomEditor(typeof(PlayModeRemotingPlugin))] + internal class PlayModeRemotingPluginInspector : UnityEditor.Editor + { + private PlayModeRemotingPlugin m_playModeRemotingPlugin; + private UnityEditor.Editor m_remotingSettingsEditor; + + private void OnEnable() + { + m_playModeRemotingPlugin = target as PlayModeRemotingPlugin; + } + + public override void OnInspectorGUI() + { + CreateCachedEditor(m_playModeRemotingPlugin.GetOrLoadRemotingSettings(), null, ref m_remotingSettingsEditor); + + if (m_remotingSettingsEditor == null) + { + return; + } + + EditorGUI.BeginDisabledGroup(EditorApplication.isPlaying); + m_remotingSettingsEditor.OnInspectorGUI(); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Editor/Inspectors/PlayModeRemotingPluginInspector.cs.meta b/com.microsoft.mixedreality.openxr/Editor/Inspectors/PlayModeRemotingPluginInspector.cs.meta new file mode 100644 index 0000000..b35bdac --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Inspectors/PlayModeRemotingPluginInspector.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 139c58869652e4541be90a4ed4096c54 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/Inspectors/RemotingSettingsInspector.cs b/com.microsoft.mixedreality.openxr/Editor/Inspectors/RemotingSettingsInspector.cs new file mode 100644 index 0000000..cebf093 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Inspectors/RemotingSettingsInspector.cs @@ -0,0 +1,19 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using Microsoft.MixedReality.OpenXR.Remoting; +using UnityEditor; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + [CustomEditor(typeof(RemotingSettings))] + internal class RemotingSettingsInspector : UnityEditor.Editor + { + public override void OnInspectorGUI() + { + serializedObject.UpdateIfRequiredOrScript(); + DrawPropertiesExcluding(serializedObject, new string[] { "m_Script" }); + serializedObject.ApplyModifiedProperties(); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Editor/Inspectors/RemotingSettingsInspector.cs.meta b/com.microsoft.mixedreality.openxr/Editor/Inspectors/RemotingSettingsInspector.cs.meta new file mode 100644 index 0000000..c2fecdd --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Inspectors/RemotingSettingsInspector.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: fbf9a002e78616b429854e4ed5d179e1 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/Microsoft.MixedReality.OpenXR.Editor.asmdef b/com.microsoft.mixedreality.openxr/Editor/Microsoft.MixedReality.OpenXR.Editor.asmdef new file mode 100644 index 0000000..415f7de --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Microsoft.MixedReality.OpenXR.Editor.asmdef @@ -0,0 +1,44 @@ +{ + "name": "Microsoft.MixedReality.OpenXR.Editor", + "rootNamespace": "Microsoft.MixedReality.OpenXR.Editor", + "references": [ + "Microsoft.MixedReality.OpenXR", + "Unity.XR.ARFoundation", + "Unity.XR.Management", + "Unity.XR.Management.Editor", + "Unity.XR.OpenXR", + "Unity.XR.OpenXR.Editor", + "Unity.XR.OpenXR.Features.MetaQuestSupport", + "Unity.XR.OpenXR.Features.OculusQuestSupport", + "UnityEngine.SpatialTracking", + "Unity.XR.CoreUtils.Editor", + "Unity.InputSystem" + ], + "includePlatforms": [ + "Editor" + ], + "excludePlatforms": [], + "allowUnsafeCode": false, + "overrideReferences": false, + "precompiledReferences": [], + "autoReferenced": true, + "defineConstraints": [], + "versionDefines": [ + { + "name": "com.unity.xr.arfoundation", + "expression": "3.0.0", + "define": "USE_ARFOUNDATION" + }, + { + "name": "com.unity.xr.arfoundation", + "expression": "5.0.0", + "define": "USE_ARFOUNDATION_5_OR_NEWER" + }, + { + "name": "com.unity.xr.openxr", + "expression": "1.6.0", + "define": "UNITY_OPENXR_1_6_OR_NEWER" + } + ], + "noEngineReferences": false +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Editor/Microsoft.MixedReality.OpenXR.Editor.asmdef.meta b/com.microsoft.mixedreality.openxr/Editor/Microsoft.MixedReality.OpenXR.Editor.asmdef.meta new file mode 100644 index 0000000..f117fcb --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Microsoft.MixedReality.OpenXR.Editor.asmdef.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: c07adb8a81c425743b901c73a99547ae +AssemblyDefinitionImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers.meta b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers.meta new file mode 100644 index 0000000..cc25223 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 223bf5e4b8454e246ac28d8b95f42298 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/DocURLAttributeDrawer.cs b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/DocURLAttributeDrawer.cs new file mode 100644 index 0000000..7605b1c --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/DocURLAttributeDrawer.cs @@ -0,0 +1,40 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEditor; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + [CustomPropertyDrawer(typeof(DocURLAttribute))] + internal class DocURLAttributeDrawer : PropertyDrawer + { + private static readonly GUIContent ButtonContent = new GUIContent( + string.Empty, EditorGUIUtility.IconContent("_Help").image, "Click for documentation"); + + public override void OnGUI(Rect position, SerializedProperty property, GUIContent label) + { + DocURLAttribute labelWidthAttribute = attribute as DocURLAttribute; + + if (!string.IsNullOrEmpty(labelWidthAttribute.Url)) + { + using (new EditorGUILayout.HorizontalScope()) + { + const float Spacing = 5f; + Vector2 size = EditorStyles.label.CalcSize(ButtonContent); + position.width -= size.x + Spacing; + + EditorGUI.PropertyField(position, property, label); + + position.x = position.width + Spacing; + position.width = size.x; + + if (GUI.Button(position, ButtonContent, EditorStyles.label)) + { + Help.BrowseURL(labelWidthAttribute.Url); + } + } + } + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/DocURLAttributeDrawer.cs.meta b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/DocURLAttributeDrawer.cs.meta new file mode 100644 index 0000000..33ecb4c --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/DocURLAttributeDrawer.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6a08fd0f8c714814c99274a433599839 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/EditorDrawerVisibleToBuildTargetAttributeDrawer.cs b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/EditorDrawerVisibleToBuildTargetAttributeDrawer.cs new file mode 100644 index 0000000..59c3583 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/EditorDrawerVisibleToBuildTargetAttributeDrawer.cs @@ -0,0 +1,29 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR + +using System; +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + [CustomPropertyDrawer(typeof(EditorDrawerVisibleToBuildTargetAttribute))] + internal class EditorDrawerVisibleToBuildTargetAttributeDrawer : PropertyDrawer + { + public override void OnGUI(Rect position, SerializedProperty property, GUIContent label) + { + EditorDrawerVisibleToBuildTargetAttribute buildTargetAttribute = attribute as EditorDrawerVisibleToBuildTargetAttribute; + + if (Array.Exists(buildTargetAttribute.BuildTargetGroups, + x => x == OpenXRFeatureSetManager.activeBuildTarget)) + { + EditorGUI.PropertyField(position, property, label); + } + } + } +} + +#endif \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/EditorDrawerVisibleToBuildTargetAttributeDrawer.cs.meta b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/EditorDrawerVisibleToBuildTargetAttributeDrawer.cs.meta new file mode 100644 index 0000000..7bbf23c --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/EditorDrawerVisibleToBuildTargetAttributeDrawer.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 7632f69fe6c5ecf4f81e01d55ae62ea8 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/LabelWidthAttributeDrawer.cs b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/LabelWidthAttributeDrawer.cs new file mode 100644 index 0000000..8fa762e --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/LabelWidthAttributeDrawer.cs @@ -0,0 +1,22 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEditor; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + [CustomPropertyDrawer(typeof(LabelWidthAttribute))] + internal class LabelWidthAttributeDrawer : PropertyDrawer + { + public override void OnGUI(Rect position, SerializedProperty property, GUIContent label) + { + LabelWidthAttribute labelWidthAttribute = attribute as LabelWidthAttribute; + + float oldLabelWidth = EditorGUIUtility.labelWidth; + EditorGUIUtility.labelWidth = labelWidthAttribute.Width; + EditorGUI.PropertyField(position, property, label); + EditorGUIUtility.labelWidth = oldLabelWidth; + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/LabelWidthAttributeDrawer.cs.meta b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/LabelWidthAttributeDrawer.cs.meta new file mode 100644 index 0000000..891771b --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/PropertyDrawers/LabelWidthAttributeDrawer.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 20a708cf09c10b64388afaf5e4d004c7 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/Settings.meta b/com.microsoft.mixedreality.openxr/Editor/Settings.meta new file mode 100644 index 0000000..a0dcf82 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Settings.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 78a6c9cec2c299048a64cc37382c6a56 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/Settings/PlatformValidation.cs b/com.microsoft.mixedreality.openxr/Editor/Settings/PlatformValidation.cs new file mode 100644 index 0000000..7257103 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Settings/PlatformValidation.cs @@ -0,0 +1,752 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR + +using Microsoft.MixedReality.OpenXR.Remoting; +using System.Collections.Generic; +using System.Linq; +using Unity.XR.CoreUtils.Editor; +using UnityEditor; +using UnityEditor.SceneManagement; +using UnityEditor.XR.Management.Metadata; +using UnityEditor.XR.OpenXR.Features; +using UnityEngine; +using UnityEngine.SpatialTracking; +using UnityEngine.XR.Management; +using UnityEngine.XR.OpenXR; +using UnityEngine.XR.OpenXR.Features; +using UnityEngine.XR.OpenXR.Features.Interactions; +using static Microsoft.MixedReality.OpenXR.MixedRealityFeaturePlugin; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + /// + /// Provides a menu item for configuring settings according to specified OpenXR devices. + /// + internal static class PlatformValidation + { + private static readonly IEnumerable ValidationRulesetsForRuleGeneration = ((ValidationRuleset[]) System.Enum.GetValues(typeof(ValidationRuleset))).Where((ruleset) => ruleset != ValidationRuleset.None); + + private const string OpenXRProjectValidationSettingsPath = "Project/XR Plug-in Management/Project Validation"; + + private const string Win32StandaloneRulesetMenuPath = "Mixed Reality/Project Validation Settings/Win32 Application (Standalone)"; + private const string HoloLens2RulesetMenuPath = "Mixed Reality/Project Validation Settings/HoloLens 2 Application (UWP)"; + private const string Win32AppRemotingRulesetMenuPath = "Mixed Reality/Project Validation Settings/Win32 Remoting App for HoloLens 2 (Standalone)"; + private const string UWPAppRemotingRulesetMenuPath = "Mixed Reality/Project Validation Settings/Win32 Remoting App for HoloLens 2 (UWP)"; + private const string DisableValidationRulesetMenuPath = "Mixed Reality/Project Validation Settings/General Validation Rules Only"; + + private const string PerformanceHelpLinkText = "Click this icon for more info on recommended performance settings for HoloLens 2."; + private const string PerformanceHelpLink = "https://aka.ms/HoloLens2PerfSettings"; + private const string SetupHelpLink = "https://aka.ms/HoloLens2OpenXRConfig"; + private static readonly string CannotAutoSetupForHL2 = "Could not automatically apply recommended settings for HoloLens 2. " + + $"Please see {SetupHelpLink} for manual set up instructions"; + private static readonly string CannotAutoOptimizeForHL2 = "Could not automatically apply recommended settings for HoloLens 2. " + + $"Please see {PerformanceHelpLink} for manual optimization instructions"; + + private const string AboutValidationRulesetsHelpText = "This rule has been enabled because a specific application type has been targeted for this project. \r\n" + + "To change or disable these additional validation rules, use the top-level menu:\r\n \"Mixed Reality > Project Validation Settings\""; + + [InitializeOnLoadMethod] + private static void InitializePlatformValidation() + { + // These rules are generated for every validation ruleset on every platform - e.g. trying to validate HL2 apps on Standalone will redirect users to UWP. + BuildValidator.AddRules(BuildTargetGroup.WSA, GenerateBuildTargetRules(BuildTargetGroup.WSA)); + BuildValidator.AddRules(BuildTargetGroup.Standalone, GenerateBuildTargetRules(BuildTargetGroup.Standalone)); + + // These rules are generated for every validation ruleset, but only on the platforms where each ruleset is supported. + // If a project is on the wrong platform, that needs to be fixed before these rules will show. + BuildValidator.AddRules(BuildTargetGroup.WSA, GenerateOpenXRLoaderRules(BuildTargetGroup.WSA)); + BuildValidator.AddRules(BuildTargetGroup.Standalone, GenerateOpenXRLoaderRules(BuildTargetGroup.Standalone)); + + BuildValidator.AddRules(BuildTargetGroup.WSA, GenerateRequiredFeatureSetsRules(BuildTargetGroup.WSA)); + BuildValidator.AddRules(BuildTargetGroup.Standalone, GenerateRequiredFeatureSetsRules(BuildTargetGroup.Standalone)); + + BuildValidator.AddRules(BuildTargetGroup.WSA, GenerateInitializeXROnStartRules(BuildTargetGroup.WSA)); + BuildValidator.AddRules(BuildTargetGroup.Standalone, GenerateInitializeXROnStartRules(BuildTargetGroup.Standalone)); + + BuildValidator.AddRules(BuildTargetGroup.WSA, GenerateCameraRules()); + + // These rules are all HL2 specific, and are only enabled with the HL2 validation ruleset while targeting UWP. + BuildValidationRule[] wsaValidationRules = new BuildValidationRule[] { GenerateHL2RealtimeGIRule(), GenerateHL2QualityRule(), + GenerateHL2RenderAndDepthSubmissionModeRule() }; + BuildValidator.AddRules(BuildTargetGroup.WSA, wsaValidationRules); + + // These deprecation warnings are always enabled. + BuildValidationRule[] androidDeprecationRules = new BuildValidationRule[] { GenerateAndroidHandTrackingRule(), GenerateAndroidMotionControllerRule() }; + BuildValidator.AddRules(BuildTargetGroup.Android, androidDeprecationRules); + } + + private static void ChangeValidationRuleset(ValidationRuleset validationRuleset) + { + BuildTargetGroup buildTargetGroup = validationRuleset.GetBuildTargetGroup(); + BuildTarget buildTarget = validationRuleset.GetBuildTarget(); + string scenarioName = validationRuleset.GetScenarioName(); + + if (buildTargetGroup == BuildTargetGroup.WSA && !BuildPipeline.IsBuildTargetSupported(BuildTargetGroup.WSA, BuildTarget.WSAPlayer)) + { + EditorUtility.DisplayDialog("UWP support not found", "The UWP build support is not currently installed. " + + "Please add the Universal Windows Platform Build Support module to your Unity installation.", "OK"); + return; + } + + BuildTargetGroup previousBuildTargetGroup = EditorUserBuildSettings.selectedBuildTargetGroup; + + // Make sure to select the correct platform, which also select the correct tab in the validation window. + // NOTE: must do this selected group change before switching build target below + // otherwise this selection change will not function properly in Unity editor. + if (EditorUserBuildSettings.selectedBuildTargetGroup != buildTargetGroup) + { + EditorUserBuildSettings.selectedBuildTargetGroup = buildTargetGroup; + } + + if (EditorUserBuildSettings.activeBuildTarget != buildTarget) + { + if (EditorUtility.DisplayDialog("Change build target platform?", $"This project is currently targeting a platform which does not support {scenarioName} apps. " + + $"To build {scenarioName} applications, the build target platform in Build Settings must be {buildTarget}.\n\n" + + $"Click `Continue` to switch the build target platform to {buildTarget} and open the Project Validation window to review other validation messages.", + "Continue", "Cancel")) + { + EditorUserBuildSettings.SwitchActiveBuildTarget(buildTargetGroup, buildTarget); + } + else + { + EditorUserBuildSettings.selectedBuildTargetGroup = previousBuildTargetGroup; + return; + } + } + + ValidationSettings.CurrentRuleset = validationRuleset; + ShowProjectValidationSettings(); + } + + private static void ShowProjectValidationSettings() + { + // Need to call OpenProjectSettings twice since the first call may not properly bring up the requested page + // Possibly due to the generation of XR settings related files on the fly + SettingsService.OpenProjectSettings(OpenXRProjectValidationSettingsPath); + EditorApplication.delayCall += () => + { + SettingsService.OpenProjectSettings(OpenXRProjectValidationSettingsPath); + }; + } + + [MenuItem(Win32StandaloneRulesetMenuPath, isValidateFunction: false, priority: 10)] + private static void ChangeValidationRulesetWin32Standalone() => + ChangeValidationRuleset(ValidationRuleset.Win32Standalone); + + [MenuItem(Win32StandaloneRulesetMenuPath, isValidateFunction: true)] + private static bool ChangeValidationRulesetWin32StandaloneValidator() => + UpdateRulesetMenuItemChecked(Win32StandaloneRulesetMenuPath, ValidationRuleset.Win32Standalone); + + [MenuItem(HoloLens2RulesetMenuPath, isValidateFunction: false, priority: 11)] + private static void ChangeValidationRulesetHoloLens2() => + ChangeValidationRuleset(ValidationRuleset.HoloLens2); + + [MenuItem(HoloLens2RulesetMenuPath, isValidateFunction: true)] + private static bool ChangeValidationRulesetHoloLens2Validator() => + UpdateRulesetMenuItemChecked(HoloLens2RulesetMenuPath, ValidationRuleset.HoloLens2); + + [MenuItem(Win32AppRemotingRulesetMenuPath, isValidateFunction: false, priority: 12)] + private static void ChangeValidationRulesetWin32AppRemoting() => + ChangeValidationRuleset(ValidationRuleset.Win32AppRemoting); + + [MenuItem(Win32AppRemotingRulesetMenuPath, isValidateFunction: true)] + private static bool ChangeValidationRulesetWin32AppRemotingValidator() => + UpdateRulesetMenuItemChecked(Win32AppRemotingRulesetMenuPath, ValidationRuleset.Win32AppRemoting); + + [MenuItem(UWPAppRemotingRulesetMenuPath, isValidateFunction: false, priority: 13)] + private static void ChangeValidationRulesetUWPAppRemoting() => + ChangeValidationRuleset(ValidationRuleset.UWPAppRemoting); + + [MenuItem(UWPAppRemotingRulesetMenuPath, isValidateFunction: true)] + private static bool ChangeValidationRulesetUWPAppRemotingValidator() => + UpdateRulesetMenuItemChecked(UWPAppRemotingRulesetMenuPath, ValidationRuleset.UWPAppRemoting); + + // MenuItems with a difference in priority > 10 will have a horizontal line between them in the menu UI. + [MenuItem(DisableValidationRulesetMenuPath, isValidateFunction: false, priority: 25)] + private static void RemoveValidationRulesets() => ValidationSettings.CurrentRuleset = ValidationRuleset.None; + + [MenuItem(DisableValidationRulesetMenuPath, isValidateFunction: true)] + private static bool RemoveValidationRulesetsValidator() => + UpdateRulesetMenuItemChecked(DisableValidationRulesetMenuPath, ValidationRuleset.None); + + + #region Validation ruleset rules + + private static BuildValidationRule[] GenerateBuildTargetRules(BuildTargetGroup buildTargetGroup) + { + List validationRules = new List(); + + foreach(ValidationRuleset validationRuleset in ValidationRulesetsForRuleGeneration) + { + if(validationRuleset.GetBuildTargetGroup() == buildTargetGroup) + { + // No need for a rule on this build target group, since it must already be correct + continue; + } + + string scenarioName = validationRuleset.GetScenarioName(); + string platformShortName = validationRuleset.GetPlatformShortName(); + validationRules.Add(new BuildValidationRule() + { + // The build target should always be enabled if plugin validation has been enabled. + IsRuleEnabled = () => ValidationSettings.CurrentRuleset == validationRuleset, + Category = $"Mixed Reality OpenXR - {scenarioName} Ruleset", + Message = $"The project needs to target the {platformShortName} platform to build {scenarioName} applications.", + CheckPredicate = () => EditorUserBuildSettings.activeBuildTarget == validationRuleset.GetBuildTarget(), + FixIt = () => { + EditorUserBuildSettings.selectedBuildTargetGroup = validationRuleset.GetBuildTargetGroup(); + EditorUserBuildSettings.SwitchActiveBuildTarget(validationRuleset.GetBuildTargetGroup(), validationRuleset.GetBuildTarget()); + }, + FixItMessage = $"Switch the build target to {platformShortName}", + Error = true, + FixItAutomatic = true, + HelpText = AboutValidationRulesetsHelpText + }); + } + + return validationRules.ToArray(); + } + + private static BuildValidationRule[] GenerateOpenXRLoaderRules(BuildTargetGroup buildTargetGroup) + { + List validationRules = new List(); + + foreach(ValidationRuleset validationRuleset in ValidationRulesetsForRuleGeneration) + { + if(validationRuleset.GetBuildTargetGroup() != buildTargetGroup) + { + continue; + } + + string scenarioName = validationRuleset.GetScenarioName(); + string platformShortName = validationRuleset.GetPlatformShortName(); + validationRules.Add(new BuildValidationRule() + { + // The OpenXR loader should always be enabled if a validation ruleset has been enabled. + IsRuleEnabled = () => ValidationSettings.CurrentRuleset == validationRuleset, + Category = $"Mixed Reality OpenXR - {scenarioName} Ruleset", + Message = $"For {scenarioName} applications, the OpenXR loader must be enabled for {platformShortName} in XR plugin management settings.", + CheckPredicate = () => XRPackageMetadataStore.IsLoaderAssigned(typeof(OpenXRLoader).FullName, buildTargetGroup), + FixIt = () => EnableOpenXRLoader(buildTargetGroup), + FixItMessage = $"Assign the OpenXR loader for {platformShortName}", + Error = false, + HelpText = AboutValidationRulesetsHelpText + }); + } + + return validationRules.ToArray(); + } + + private static BuildValidationRule[] GenerateRequiredFeatureSetsRules(BuildTargetGroup buildTargetGroup) + { + List validationRules = new List(); + + foreach (ValidationRuleset validationRuleset in ValidationRulesetsForRuleGeneration) + { + if (validationRuleset.GetBuildTargetGroup() != buildTargetGroup) + { + continue; + } + + string scenarioName = validationRuleset.GetScenarioName(); + string platformShortName = validationRuleset.GetPlatformShortName(); + + System.Type[] featureSets = validationRuleset.GetRequiredFeatureSets(); + List featureSetUINames = new List(); + List featureSetIds = new List(); + + GetIdsAndUserNamesForFeatureSets(featureSets, ref featureSetIds, ref featureSetUINames); + + validationRules.Add(new BuildValidationRule() + { + // The necessary feature set should always be enabled if a validation ruleset has been enabled. + IsRuleEnabled = () => ValidationSettings.CurrentRuleset == validationRuleset, + Category = $"Mixed Reality OpenXR - {scenarioName} Ruleset", + Message = $"For {scenarioName} apps, the following feature sets must be enabled for {platformShortName} in OpenXR settings: {string.Join(", ", featureSetUINames)}", + CheckPredicate = () => CheckFeatureSets(buildTargetGroup, featureSetIds), + FixIt = () => EnableFeatureSets(buildTargetGroup, featureSetIds), + FixItMessage = $"Enable the following feature sets for {platformShortName}: {string.Join(", ", featureSetUINames)}", + Error = true, + HelpText = AboutValidationRulesetsHelpText + }); + + System.Type[] notRequiredfeatureSets = validationRuleset.GetNotRequiredFeatureSets(); + List notRequiredFeatureSetUINames = new List(); + List notRequiredFeatureSetIds = new List(); + + GetIdsAndUserNamesForFeatureSets(notRequiredfeatureSets, ref notRequiredFeatureSetIds, ref notRequiredFeatureSetUINames); + + if (notRequiredFeatureSetIds.Count > 0) + { + validationRules.Add(new BuildValidationRule() + { + // The necessary feature set should always be enabled if a validation ruleset has been enabled. + IsRuleEnabled = () => ValidationSettings.CurrentRuleset == validationRuleset, + Category = $"Mixed Reality OpenXR - {scenarioName} Ruleset", + Message = $"For {scenarioName} apps, the following feature sets must be disabled for {platformShortName} in OpenXR settings: {string.Join(", ", notRequiredFeatureSetUINames)}", + CheckPredicate = () => !CheckFeatureSets(buildTargetGroup, notRequiredFeatureSetIds), + FixIt = () => DisableFeatureSets(buildTargetGroup, notRequiredFeatureSetIds), + FixItMessage = $"Disable the following feature sets for {platformShortName}: {string.Join(", ", notRequiredFeatureSetUINames)}", + Error = true, + HelpText = AboutValidationRulesetsHelpText + }); + } + } + + return validationRules.ToArray(); + } + + private static BuildValidationRule[] GenerateInitializeXROnStartRules(BuildTargetGroup buildTargetGroup) + { + List validationRules = new List(); + + foreach (ValidationRuleset validationRuleset in ValidationRulesetsForRuleGeneration) + { + if (validationRuleset.GetBuildTargetGroup() != buildTargetGroup) + { + continue; + } + + string scenarioName = validationRuleset.GetScenarioName(); + string platformShortName = validationRuleset.GetPlatformShortName(); + bool remotingEnabled = validationRuleset.GetRemotingEnabled(); + + validationRules.Add(new BuildValidationRule() + { + // This rule will run if a validation ruleset has been enabled. If there is no validation ruleset, we fallback to the validator in AppRemotingValidator.cs + IsRuleEnabled = () => ValidationSettings.CurrentRuleset == validationRuleset, + Category = $"Mixed Reality OpenXR - {scenarioName} Ruleset", + Message = $"For {scenarioName} applications, XR initialization should {(remotingEnabled ? "be delayed until a specific IP address is entered" : "not be delayed")}", + CheckPredicate = () => + { + XRGeneralSettings settings = XRSettingsHelpers.GetOrCreateXRGeneralSettings(buildTargetGroup); + return settings != null && settings.InitManagerOnStart != remotingEnabled; + }, + FixIt = () => + { + XRGeneralSettings settings = XRSettingsHelpers.GetOrCreateXRGeneralSettings(buildTargetGroup); + if (settings != null) + { + settings.InitManagerOnStart = !remotingEnabled; + } + }, + FixItMessage = $"{ (remotingEnabled ? "Disable" : "Enable") } XR initialization on startup", + // If remoting is enabled, this must be enabled. If remoting is not enabled, this should likely be disabled, but it's not required. + Error = remotingEnabled, + HelpText = AboutValidationRulesetsHelpText + }); + } + + return validationRules.ToArray(); + } + + #endregion + + + #region HoloLens 2 rules + + private static BuildValidationRule GenerateHL2RealtimeGIRule() + { + return new BuildValidationRule() + { + IsRuleEnabled = IsHL2RulesetEnabled, + Category = "Mixed Reality OpenXR - HoloLens 2 Ruleset", + Message = $"Realtime GI has a negative performance impact on HoloLens 2 applications.", + CheckPredicate = () => !Lightmapping.TryGetLightingSettings(out LightingSettings lightingSettings) || !lightingSettings.realtimeGI, + FixIt = () => + { + if (Lightmapping.TryGetLightingSettings(out LightingSettings lightingSettings)) + { + lightingSettings.realtimeGI = false; + EditorUtility.SetDirty(lightingSettings); + } + else + { + Debug.LogError(CannotAutoOptimizeForHL2); + } + }, + FixItMessage = $"Disable realtime GI in lighting settings", + Error = false, + HelpText = PerformanceHelpLinkText + "\r\n\r\n" + AboutValidationRulesetsHelpText, + HelpLink = PerformanceHelpLink + }; + } + + private static BuildValidationRule GenerateHL2QualityRule() + { + return new BuildValidationRule() + { + // Currently this rule doesn't work as Unity use the "default quality" for a platform to determine the quality level to use + // Setting the "current active quality" does not impact the application running on HoloLens 2 + IsRuleEnabled = () => false, + Category = "Mixed Reality OpenXR - HoloLens 2 Ruleset", + Message = $"High quality settings have a negative performance impact on HoloLens 2 applications.", + CheckPredicate = () => QualitySettings.GetQualityLevel() == 0, + FixIt = () => QualitySettings.SetQualityLevel(0, true), + FixItMessage = $"Set quality settings to very low", + Error = false, + HelpText = PerformanceHelpLinkText + "\r\n\r\n" + AboutValidationRulesetsHelpText, + HelpLink = PerformanceHelpLink + }; + } + + private static BuildValidationRule GenerateHL2GPUSkinningRule() + { + return new BuildValidationRule() + { + IsRuleEnabled = IsHL2RulesetEnabled, + Category = "Mixed Reality OpenXR - HoloLens 2 Ruleset", + Message = $"GPU skinning negatively impacts the performance of HoloLens 2 applications.", + CheckPredicate = () => !PlayerSettings.gpuSkinning, + FixIt = () => PlayerSettings.gpuSkinning = false, + FixItMessage = $"Disable GPU skinning", + Error = false, + HelpText = PerformanceHelpLinkText + "\r\n\r\n" + AboutValidationRulesetsHelpText, + HelpLink = PerformanceHelpLink + }; + } + + private static BuildValidationRule GenerateHL2RenderAndDepthSubmissionModeRule() + { + return new BuildValidationRule() + { + IsRuleEnabled = IsHL2RulesetEnabled, + Category = "Mixed Reality OpenXR - HoloLens 2 Ruleset", + Message = $"Single pass instanced is recommended for render mode and depth 16 bit is recommended for depth submission mode settings.", + CheckPredicate = () => + { + if (TryGetOpenXRSetting(BuildTargetGroup.WSA, out OpenXRSettings settings)) + { + return settings.depthSubmissionMode == OpenXRSettings.DepthSubmissionMode.Depth16Bit && settings.renderMode == OpenXRSettings.RenderMode.SinglePassInstanced; + } + return true; + }, + FixIt = () => + { + if (TryGetOpenXRSetting(BuildTargetGroup.WSA, out OpenXRSettings settings)) + { + settings.depthSubmissionMode = OpenXRSettings.DepthSubmissionMode.Depth16Bit; + settings.renderMode = OpenXRSettings.RenderMode.SinglePassInstanced; + EditorUtility.SetDirty(settings); + } + else + { + Debug.LogError(CannotAutoOptimizeForHL2); + } + }, + FixItMessage = $"Switch the render mode to single pass instanced and depth submission mode to depth 16 bit", + Error = false, + HelpText = PerformanceHelpLinkText + "\r\n\r\n" + AboutValidationRulesetsHelpText, + HelpLink = PerformanceHelpLink + }; + } + + #endregion HoloLens 2 rules + + #region Camera setup rules + + // Guidelines for using a camera which will render output for a HoloLens 2 + private static BuildValidationRule[] GenerateCameraRules() + { + return new BuildValidationRule[] { GenerateCameraFlagsRule(ValidationRuleset.HoloLens2), GenerateCameraFlagsRule(ValidationRuleset.UWPAppRemoting), + GenerateCameraBackgroundColorRule(ValidationRuleset.HoloLens2), GenerateCameraBackgroundColorRule(ValidationRuleset.UWPAppRemoting), + GenerateCameraPoseDriverRule(ValidationRuleset.HoloLens2), GenerateCameraPoseDriverRule(ValidationRuleset.UWPAppRemoting) }; + } + + private static BuildValidationRule GenerateCameraFlagsRule(ValidationRuleset targetRuleset) + { + return new BuildValidationRule() + { + IsRuleEnabled = () => ValidationSettings.CurrentRuleset == targetRuleset, + Category = $"Mixed Reality OpenXR - {targetRuleset.GetScenarioName()} Ruleset (Scene specific)", + Message = $"It is recommended for the main camera to be cleared with a solid color.", + CheckPredicate = () => Camera.main == null || Camera.main.clearFlags == CameraClearFlags.SolidColor, + FixIt = () => + { + Camera.main.clearFlags = CameraClearFlags.SolidColor; + EditorSceneManager.MarkSceneDirty(EditorSceneManager.GetActiveScene()); + }, + FixItMessage = $"Set the main camera's clearFlags to CameraClearFlags.SolidColor", + Error = false, + FixItAutomatic = false, + SceneOnlyValidation = true, + HelpText = AboutValidationRulesetsHelpText + }; + } + + private static BuildValidationRule GenerateCameraBackgroundColorRule(ValidationRuleset targetRuleset) + { + return new BuildValidationRule() + { + IsRuleEnabled = () => ValidationSettings.CurrentRuleset == targetRuleset, + Category = $"Mixed Reality OpenXR - {targetRuleset.GetScenarioName()} Ruleset (Scene specific)", + Message = $"It is recommended for the main camera to have a clear background color.", + CheckPredicate = () => Camera.main == null || Camera.main.backgroundColor == Color.clear, + FixIt = () => + { + Camera.main.backgroundColor = Color.clear; + EditorSceneManager.MarkSceneDirty(EditorSceneManager.GetActiveScene()); + }, + FixItMessage = $"Set the main camera background color to clear", + Error = false, + FixItAutomatic = false, + SceneOnlyValidation = true, + HelpText = AboutValidationRulesetsHelpText + }; + } + + private static BuildValidationRule GenerateCameraPoseDriverRule(ValidationRuleset targetRuleset) + { + return new BuildValidationRule() + { + IsRuleEnabled = () => ValidationSettings.CurrentRuleset == targetRuleset, + Category = $"Mixed Reality OpenXR - {targetRuleset.GetScenarioName()} Ruleset (Scene specific)", + Message = $"It is recommended for the main camera to have a PoseDriver component.", + CheckPredicate = () => + Camera.main == null + || Camera.main.gameObject.GetComponent() != null + || Camera.main.gameObject.GetComponent() != null +#if USE_ARFOUNDATION && !USE_ARFOUNDATION_5_OR_NEWER + || Camera.main.gameObject.GetComponent() != null +#endif + , + FixIt = () => + { + if (Camera.main != null + && !Camera.main.gameObject.GetComponent() + && !Camera.main.gameObject.GetComponent() +#if USE_ARFOUNDATION && !USE_ARFOUNDATION_5_OR_NEWER + && !Camera.main.gameObject.GetComponent() +#endif + ) + { + Camera.main.gameObject.AddComponent(); + } + EditorSceneManager.MarkSceneDirty(EditorSceneManager.GetActiveScene()); + }, + FixItMessage = $"Ensure the main camera has a PoseDriver component", + Error = false, + FixItAutomatic = false, + SceneOnlyValidation = true, + HelpText = AboutValidationRulesetsHelpText + }; + } + + #endregion + + + #region Android rules + + private static BuildValidationRule GenerateAndroidHandTrackingRule() + { + var handTrackingFeaturePlugins = new System.Type[] { typeof(HandTrackingFeaturePlugin) }; + return new BuildValidationRule() + { + IsRuleEnabled = () => true, + Category = "Mixed Reality OpenXR - Android", + Message = $"Hand tracking on Android with the Mixed Reality OpenXR Plugin has been deprecated. " + + "For Android applications using hand tracking, we recommend transitioning to the OpenXR plugins from Unity and Meta.", + CheckPredicate = () => !CheckFeatures(BuildTargetGroup.Android, handTrackingFeaturePlugins), + FixIt = () => SettingsService.OpenProjectSettings("Project/XR Plug-in Management/OpenXR"), + FixItMessage = $"Open Project Settings to disable the Hand Tracking feature for the Android build target.", + FixItAutomatic = false, + Error = false, + HelpText = AboutValidationRulesetsHelpText + }; + } + + private static BuildValidationRule GenerateAndroidMotionControllerRule() + { + var motionControllerFeaturePlugins = new System.Type[] { typeof(MotionControllerFeaturePlugin) }; + return new BuildValidationRule() + { + IsRuleEnabled = () => true, + Category = "Mixed Reality OpenXR - Android", + Message = $"The Motion Controller Model feature plugin from the Mixed Reality OpenXR Plugin has been deprecated on Android. " + + "For Android applications using motion controller models, we recommend transitioning to the OpenXR plugins from Unity and Meta.", + CheckPredicate = () => !CheckFeatures(BuildTargetGroup.Android, motionControllerFeaturePlugins), + FixIt = () => SettingsService.OpenProjectSettings("Project/XR Plug-in Management/OpenXR"), + FixItMessage = $"Open Project Settings to disable the Motion Controller feature for the Android build target.", + FixItAutomatic = false, + Error = false, + HelpText = AboutValidationRulesetsHelpText + }; + } + #endregion Android rules + + + #region Helpers + + private static bool UpdateRulesetMenuItemChecked(string menuPath, ValidationRuleset ruleset) + { + Menu.SetChecked(menuPath, ValidationSettings.CurrentRuleset == ruleset); + return true; + } + + private static bool IsHL2RulesetEnabled() + { + // Ensure the OpenXR loader is enabled on WSA - otherwise this is a flat UWP app on HL2 that shouldn't be validated with XR rules. + XRGeneralSettings wsaSettings = XRSettingsHelpers.GetOrCreateXRGeneralSettings(BuildTargetGroup.WSA); + if (!wsaSettings.Manager.activeLoaders.Any(loader => loader is UnityEngine.XR.OpenXR.OpenXRLoader)) + { + return false; + } + + // Ensure the HL2 feature plugin is enabled, with ValidationRuleTarget == HL2 - otherwise these specific rules are not necessary. + MixedRealityFeaturePlugin plugin = BuildProcessorHelpers.GetOpenXRFeature(BuildTargetGroup.WSA, false); + return plugin != null && plugin.enabled && (ValidationSettings.CurrentRuleset == ValidationRuleset.HoloLens2); + } + + private static void EnableOpenXRLoader(BuildTargetGroup targetGroup) + { + if (XRSettingsHelpers.GetOrCreateXRManagerSettings(targetGroup) is XRManagerSettings settings && settings != null + && XRPackageMetadataStore.AssignLoader(settings, nameof(OpenXRLoader), targetGroup)) + { + EditorUtility.SetDirty(settings); + } + else + { + Debug.LogError(CannotAutoSetupForHL2); + } + } + + private static bool CheckFeatureSets(BuildTargetGroup targetGroup, List featureIds) + { + if (XRSettingsHelpers.GetOrCreateXRManagerSettings(targetGroup) is XRManagerSettings settings && settings != null) + { + foreach (var featureSet in OpenXRFeatureSetManager.FeatureSetsForBuildTarget(targetGroup)) + { + if (featureIds.Contains(featureSet.featureSetId) && !featureSet.isEnabled) + { + return false; + } + } + } + return true; + } + + private static void EnableFeatureSets(BuildTargetGroup targetGroup, List featureIds) + { + foreach (var featureSet in OpenXRFeatureSetManager.FeatureSetsForBuildTarget(targetGroup)) + { + if (featureIds.Contains(featureSet.featureSetId) && !featureSet.isEnabled) + { + featureSet.isEnabled = true; + } + } + OpenXRFeatureSetManager.SetFeaturesFromEnabledFeatureSets(targetGroup); + } + + private static void DisableFeatureSets(BuildTargetGroup targetGroup, List featureIds) + { + foreach (var featureSet in OpenXRFeatureSetManager.FeatureSetsForBuildTarget(targetGroup)) + { + if (featureIds.Contains(featureSet.featureSetId) && featureSet.isEnabled) + { + featureSet.isEnabled = false; + } + } + OpenXRFeatureSetManager.SetFeaturesFromEnabledFeatureSets(targetGroup); + } + + // This extension method must be defined from within the Editor package, as it depends on the feature set types. + internal static System.Type[] GetRequiredFeatureSets(this ValidationRuleset validationRuleset) + { + switch (validationRuleset) + { + case ValidationRuleset.Win32Standalone: + return new System.Type[] { typeof(WMRFeatureSet) }; + case ValidationRuleset.HoloLens2: + return new System.Type[] { typeof(HoloLensFeatureSet) }; + case ValidationRuleset.Win32AppRemoting: + return new System.Type[] { typeof(AppRemotingFeatureSet) }; + case ValidationRuleset.UWPAppRemoting: + return new System.Type[] { typeof(AppRemotingFeatureSet) }; + } + Debug.LogError($"RequiredFeatureSets of ValidationRuleset \"{validationRuleset}\" are not defined."); + return new System.Type[] { }; + } + + internal static System.Type[] GetNotRequiredFeatureSets(this ValidationRuleset validationRuleset) + { + switch (validationRuleset) + { + case ValidationRuleset.Win32Standalone: + return new System.Type[] { typeof(AppRemotingFeatureSet) }; + case ValidationRuleset.HoloLens2: + return new System.Type[] { typeof(AppRemotingFeatureSet) }; + case ValidationRuleset.Win32AppRemoting: + return new System.Type[] { }; + case ValidationRuleset.UWPAppRemoting: + return new System.Type[] { }; + } + Debug.LogError($"RequiredFeatureSets of ValidationRuleset \"{validationRuleset}\" are not defined."); + return new System.Type[] { }; + } + + private static bool TryGetOpenXRSetting(BuildTargetGroup targetGroup, out OpenXRSettings openXRSettings) + { + if (EditorBuildSettings.TryGetConfigObject(Constants.k_SettingsKey, out Object obj) && obj is IPackageSettings packageSettings + && packageSettings.GetSettingsForBuildTargetGroup(targetGroup) is OpenXRSettings settings && settings != null) + { + openXRSettings = settings; + return true; + } + else + { + openXRSettings = null; + return false; + } + } + + private static bool CheckFeatures(BuildTargetGroup targetGroup, IEnumerable featureTypes) + { + if (TryGetOpenXRSetting(targetGroup, out OpenXRSettings settings)) + { + foreach (OpenXRFeature feature in settings.GetFeatures()) + { + if (featureTypes.Contains(feature.GetType()) && !feature.enabled) + { + return false; + } + } + } + return true; + } + + private static void EnableFeatures(BuildTargetGroup targetGroup, IEnumerable featureTypes) + { + if (TryGetOpenXRSetting(targetGroup, out OpenXRSettings settings)) + { + foreach (OpenXRFeature feature in settings.GetFeatures()) + { + if (featureTypes.Contains(feature.GetType()) && !feature.enabled) + { + feature.enabled = true; + } + } + EditorUtility.SetDirty(settings); + } + } + + private static void GetIdsAndUserNamesForFeatureSets(System.Type[] featureSets, ref List featureSetIds, ref List featureSetUINames) + { + for(int i = 0; i < featureSets.Length; i++) { + var attribute = featureSets[i].GetCustomAttributes(typeof(OpenXRFeatureSetAttribute), true).FirstOrDefault() as OpenXRFeatureSetAttribute; + if (attribute == null) + { + Debug.LogError($"Could not generate Mixed Reality OpenXR feature set validator - feature set attribute not found for {featureSets[i]}"); + continue; + } + + featureSetIds.Add(attribute.FeatureSetId); + featureSetUINames.Add(attribute.UiName); + } + } + #endregion Helpers + } +} + +#endif \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Editor/Settings/PlatformValidation.cs.meta b/com.microsoft.mixedreality.openxr/Editor/Settings/PlatformValidation.cs.meta new file mode 100644 index 0000000..1eb4099 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Settings/PlatformValidation.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 519ca35bfc54b8b4ca24b48e14a4bd88 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Editor/Settings/PlayModeRemotingWindow.cs b/com.microsoft.mixedreality.openxr/Editor/Settings/PlayModeRemotingWindow.cs new file mode 100644 index 0000000..5fcf7c8 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Settings/PlayModeRemotingWindow.cs @@ -0,0 +1,120 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR + +using Microsoft.MixedReality.OpenXR.Remoting; +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; +using UnityEngine; +using UnityEngine.XR.OpenXR; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + /// + /// Represents a standalone window for accessing holographic remoting for play mode settings. + /// + internal class PlayModeRemotingWindow : EditorWindow + { + private static PlayModeRemotingPlugin Feature => OpenXRFeaturePlugin.Feature; + + private const string ConnectionInfo = "Clicking the \"Play\" button will connect Unity editor to the Holographic Remoting Player running on above IP address."; + + private static readonly GUIContent FeatureEnabledLabel = EditorGUIUtility.TrTextContent($"Disable {PlayModeRemotingPlugin.featureName}"); + private static readonly GUIContent FeatureDisabledLabel = EditorGUIUtility.TrTextContent($"Enable {PlayModeRemotingPlugin.featureName}"); + private static readonly GUIContent FixLabel = EditorGUIUtility.TrTextContent("Fix"); + + private UnityEditor.Editor m_playModeRemotingPluginEditor; + private Vector2 m_scrollPos; + + /// + /// Initializes the Remoting Window class + /// + [MenuItem(PlayModeRemotingValidator.PlayModeRemotingMenuPath)] + [MenuItem(PlayModeRemotingValidator.PlayModeRemotingMenuPath2)] + private static void Init() + { + GetWindow(PlayModeRemotingPlugin.featureName); + } + + private void OnGUI() + { + using (var scroll = new EditorGUILayout.ScrollViewScope(m_scrollPos)) + { + m_scrollPos = scroll.scrollPosition; + + if (Feature == null) + { + FeatureHelpers.RefreshFeatures(BuildTargetGroup.Standalone); + } + + UnityEditor.Editor.CreateCachedEditor(Feature, null, ref m_playModeRemotingPluginEditor); + + if (m_playModeRemotingPluginEditor == null) + { + EditorGUILayout.Space(); + EditorGUILayout.HelpBox($"An instance of {PlayModeRemotingPlugin.featureName} could not be found. Please open Project Settings > XR Plug-in Management > OpenXR to ensure it's properly loaded.", MessageType.Error); + return; + } + + EditorGUILayout.Space(); + m_playModeRemotingPluginEditor.OnInspectorGUI(); + + if (Feature.IsValidAndEnabled()) + { + EditorGUILayout.Space(); + bool hasValidSettings = Feature.HasValidSettings(); + bool isLoaderAssigned = PlayModeRemotingValidator.IsLoaderAssigned(); + bool areDependenciesEnabled = PlayModeRemotingValidator.AreDependenciesEnabled(); + + if (hasValidSettings && isLoaderAssigned && areDependenciesEnabled) + { + EditorGUILayout.HelpBox(ConnectionInfo, MessageType.Info); + } + else + { + if (!hasValidSettings) + { + EditorGUILayout.HelpBox(PlayModeRemotingValidator.RemotingNotConfigured, MessageType.Error); + } + + if (!isLoaderAssigned) + { + EditorGUILayout.HelpBox(PlayModeRemotingValidator.OpenXRLoaderNotAssigned, MessageType.Error); + if (GUILayout.Button(FixLabel)) + { + PlayModeRemotingValidator.AssignLoader(); + } + } + + if (!areDependenciesEnabled) + { + EditorGUILayout.HelpBox(PlayModeRemotingValidator.DependenciesNotEnabled, MessageType.Error); + if (GUILayout.Button(FixLabel)) + { + PlayModeRemotingValidator.EnableDependencies(); + } + } + } + } + + // Disable the "enable/disable" button when editor is already playing + using (new EditorGUI.DisabledScope(EditorApplication.isPlaying)) + { + EditorGUILayout.Space(); + if (GUILayout.Button(Feature.enabled ? FeatureEnabledLabel : FeatureDisabledLabel)) + { + Feature.enabled = !Feature.enabled; + if (Feature.enabled) + { + // If the user turned on the feature, try to enable dependencies as well. + PlayModeRemotingValidator.EnableDependencies(); + } + } + } + } + } + } +} + +#endif \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Editor/Settings/PlayModeRemotingWindow.cs.meta b/com.microsoft.mixedreality.openxr/Editor/Settings/PlayModeRemotingWindow.cs.meta new file mode 100644 index 0000000..2687292 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Editor/Settings/PlayModeRemotingWindow.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c48d5a99fb62eca48991105bd00f5f58 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/LICENSE.md b/com.microsoft.mixedreality.openxr/LICENSE.md new file mode 100644 index 0000000..63447fd --- /dev/null +++ b/com.microsoft.mixedreality.openxr/LICENSE.md @@ -0,0 +1,21 @@ +Copyright (c) Microsoft Corporation. + +MIT License + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED *AS IS*, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/LICENSE.md.meta b/com.microsoft.mixedreality.openxr/LICENSE.md.meta new file mode 100644 index 0000000..661edb4 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/LICENSE.md.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: 7e6fe1fc560cb7d49b50e5e450d58106 +TextScriptImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime.meta b/com.microsoft.mixedreality.openxr/Runtime.meta new file mode 100644 index 0000000..8c1b701 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 5b18f44b41699104c9fe4439d520c688 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API.meta b/com.microsoft.mixedreality.openxr/Runtime/API.meta new file mode 100644 index 0000000..2cd2d40 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 57cb24452de471d4b9fb2cec976acf3a +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker.meta b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker.meta new file mode 100644 index 0000000..5ad8f73 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 38032beb01b8ffe4f98b93e468204d7e +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarker.cs b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarker.cs new file mode 100644 index 0000000..5a147b0 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarker.cs @@ -0,0 +1,179 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using Microsoft.MixedReality.OpenXR.ARSubsystems; +using System; +using Unity.Collections; +using UnityEngine; +using UnityEngine.XR.ARFoundation; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Types of markers that can be tracked. + /// + public enum ARMarkerType + { + /// + /// A marker of QRCode type. + /// Equivalent to XR_SCENE_MARKER_TYPE_QR_CODE_MSFT. + /// + QRCode = 1 + } + + /// + /// Represents types of QRCodes that can be detected. + /// + public enum QRCodeType + { + /// + /// A QRCode type of QRCode. + /// Equivalent to XR_SCENE_MARKER_QR_CODE_SYMBOL_TYPE_QR_CODE_MSFT. + /// + QRCode = 1, + + /// + /// A QRCode type of MicroQRCode. + /// Equivalent to XR_SCENE_MARKER_QR_CODE_SYMBOL_TYPE_MICRO_QR_CODE_MSFT. + /// + MicroQRCode = 2 + } + + /// + /// Types of transforms that can be applied to markers. + /// + public enum TransformMode + { + /// + /// Marker centered at origin. + /// + MostStable = 0, + + /// + /// Marker centered at geometric center. + /// + Center = 1, + } + + /// + /// Represents properties of detected QRCodes. + /// + /// See https://github.com/yansikeim/QR-Code/blob/master/ISO%20IEC%2018004%202015%20Standard.pdf for more information. + public struct QRCodeProperties + { + /// + /// Version of the QRCode. + /// + /// + /// The version is in the range of 1-40 if the type is QRCode. + /// The version is in the range or 1-4 if the type is microQRCode. + /// + public uint version; + + /// + /// Type of the QRCode. + /// + public QRCodeType type; + } + + /// + /// Represents a marker detected by an AR device. + /// + /// + /// Generated by the when an AR device detects + /// a marker in the environment. + /// + [DefaultExecutionOrder(ARUpdateOrder.k_Plane)] + [DisallowMultipleComponent] + public sealed class ARMarker : ARTrackable + { + /// + /// The time when the marker was last seen. Comparable to . + /// + public float lastSeenTime => sessionRelativeData.lastSeenTime; + + /// + /// The center relative to the pose in the X, Y plane. + /// + public Vector2 center => sessionRelativeData.center; + + /// + /// The physical size (dimensions) of the marker in meters. + /// + public Vector2 size => sessionRelativeData.size; + + /// + /// The type of marker. Currently we only support markert os QRCode type. + /// + public ARMarkerType markerType => sessionRelativeData.markerType; + + /// + /// The type of transform to apply on the marker. + /// + public TransformMode transformMode + { + get => sessionRelativeData.transformMode; + set + { + if (ARMarkerManager.Instance != null) + { + ARMarkerManager.Instance.SetTransformMode(trackableId, value); + } + } + } + + /// + /// Get a native pointer associated with this marker. + /// + /// + /// The data pointed to by this member is implementation defined. + /// The lifetime of the pointed to object is also + /// implementation defined, but should be valid at least until the next + /// update. + /// + public IntPtr nativePtr => sessionRelativeData.nativePtr; + + /// + /// Get the decoded string associated with this marker. + /// + public string GetDecodedString() + { + if (ARMarkerManager.Instance != null) + { + return ARMarkerManager.Instance.GetDecodedString(trackableId); + } + return string.Empty; + } + + /// + /// Get the raw data associated with this marker. + /// + public NativeArray GetRawData(Allocator allocator) + { + if (ARMarkerManager.Instance != null) + { + return ARMarkerManager.Instance.GetRawData(trackableId, allocator); + } + return new NativeArray(0, allocator); + } + + /// + /// Get the properties associated with this QRCode marker. + /// + public QRCodeProperties GetQRCodeProperties() + { + if (markerType == ARMarkerType.QRCode) + { + if (ARMarkerManager.Instance != null) + { + return ARMarkerManager.Instance.GetQRCodeProperties(trackableId); + } + return new QRCodeProperties(); + } + else + { + throw new InvalidOperationException("GetQRCodeProperties() is only valid when markerType == ARMarkerType.QRCode"); + } + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarker.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarker.cs.meta new file mode 100644 index 0000000..8b4d934 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarker.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: cce13105eae40dc4193446e1a49607d9 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerManager.cs b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerManager.cs new file mode 100644 index 0000000..2cd9121 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerManager.cs @@ -0,0 +1,183 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using Microsoft.MixedReality.OpenXR.ARSubsystems; +using System; +using System.Collections.Generic; +using System.Linq; +using Unity.Collections; +using Unity.XR.CoreUtils; +using UnityEngine; +using UnityEngine.XR.ARFoundation; +using UnityEngine.XR.ARSubsystems; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// A manager for s. Creates, updates, and removes + /// GameObjects in response to detected surfaces in the physical + /// environment. + /// + [DefaultExecutionOrder(ARUpdateOrder.k_PlaneManager)] + [DisallowMultipleComponent] +#if USE_ARFOUNDATION_5_OR_NEWER + [RequireComponent(typeof(XROrigin))] +#else + [RequireComponent(typeof(ARSessionOrigin))] +#endif + public sealed class ARMarkerManager : ARTrackableManager< + XRMarkerSubsystem, + XRMarkerSubsystemDescriptor, + XRMarkerSubsystem.Provider, + XRMarker, + ARMarker> + { + private static ARMarkerManager m_instance = null; + + /// + /// Singleton instance for ARMarkerManager + /// + public static ARMarkerManager Instance => m_instance; + + /// + /// Getter or setter for the Marker Prefab. + /// + [Tooltip("If not null, instantiates this prefab for each created marker. Else, a default empty GameObject is created with the new ARMarker attached.")] + public GameObject markerPrefab; + + /// + /// The list of s that will be detected. + /// + [Tooltip("The list of ARMarker types that will be detected.")] + public ARMarkerType[] enabledMarkerTypes = { ARMarkerType.QRCode }; + + /// + /// Default for newly detected markers. + /// + [Tooltip("Default transform mode for newly detected markers.")] + public TransformMode defaultTransformMode = TransformMode.MostStable; + + /// + /// Invoked when markers have changed (been added, updated, or removed). + /// + public event Action markersChanged; + + /// + /// Attempt to retrieve an existing by . + /// + /// The of the marker to retrieve. + /// The with , or null if it does not exist. + public ARMarker GetMarker(TrackableId trackableId) => m_Trackables.TryGetValue(trackableId, out ARMarker marker) ? marker : null; + + /// + /// Set transform mode of an existing . + /// + /// The of the marker to be transformed. + /// The to be applied. + public void SetTransformMode(TrackableId trackableId, TransformMode transformMode) + { + if (enabled && subsystem != null) + { + subsystem.SetTransformMode(trackableId, transformMode); + } + } + + /// + /// Get raw data for an existing . + /// + /// The of the marker. + public NativeArray GetRawData(TrackableId trackableId, Allocator allocator) + { + if (enabled && subsystem != null) + { + return subsystem.GetRawData(trackableId, allocator); + } + return new NativeArray(); + } + + /// + /// Get decoded string for an existing . + /// + /// The of the marker. + public string GetDecodedString(TrackableId trackableId) + { + if (enabled && subsystem != null) + { + return subsystem.GetDecodedString(trackableId); + } + return null; + } + + /// + /// Get QR code properties for an existing of type . + /// + /// The of the QRCode marker. + public QRCodeProperties GetQRCodeProperties(TrackableId trackableId) + { + if (enabled && subsystem != null) + { + return subsystem.GetQRCodeProperties(trackableId); + } + return new QRCodeProperties(); + } + + /// + /// Get the Prefab which will be instantiated for each . Can be `null`. + /// + /// The Prefab which will be instantiated for each . + protected override GameObject GetPrefab() => markerPrefab; + + /// + /// Invoked when the base class detects trackable changes. + /// + /// The list of added s. + /// The list of updated s. + /// The list of removed s. + protected override void OnTrackablesChanged( + List added, + List updated, + List removed) + { + if (markersChanged != null) + { + using (new ScopedProfiler("OnMarkersChanged")) + markersChanged( + new ARMarkersChangedEventArgs( + added, + updated, + removed)); + } + } + + protected override void Update() + { + base.Update(); + if (enabled && subsystem != null) + { + Array.Sort(enabledMarkerTypes); + if (!enabledMarkerTypes.SequenceEqual(subsystem.EnabledMarkerTypes)) + { + subsystem.EnabledMarkerTypes = enabledMarkerTypes; + } + + if (defaultTransformMode != subsystem.DefaultTransformMode) + { + subsystem.DefaultTransformMode = defaultTransformMode; + } + } + } + + protected override void OnEnable() + { + base.OnEnable(); + + // Replicating behavior in ARFoundation to initialize singleton instance + m_instance = this; + } + + /// + /// The name to be used for the GameObject whenever a new marker object is created from . + /// + protected override string gameObjectName => "ARMarker"; + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerManager.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerManager.cs.meta new file mode 100644 index 0000000..58909f3 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerManager.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4df6e2a8f5c58cf4184b32d359111444 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerScale.cs b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerScale.cs new file mode 100644 index 0000000..73d3f85 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerScale.cs @@ -0,0 +1,39 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Monobehavior to help scale detected markers. + /// + [RequireComponent(typeof(ARMarker))] + public class ARMarkerScale : MonoBehaviour + { + private ARMarker m_arMarker; + + /// + /// Transform containing marker contents that needs to be scaled. + /// + [Tooltip("Transform containing marker contents that needs to be scaled.")] + public Transform markerScaleTransform; + + private void OnEnable() + { + m_arMarker = GetComponent(); + if (markerScaleTransform == null) + { + markerScaleTransform = gameObject.transform; + } + } + + private void Update() + { + // Scale the marker contents based on the computed scale factor. + float scaleFactor = (float)Math.Sqrt(m_arMarker.size.x * m_arMarker.size.y); + markerScaleTransform.transform.localScale = new Vector3(scaleFactor, scaleFactor, scaleFactor); + } + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerScale.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerScale.cs.meta new file mode 100644 index 0000000..49b5d6a --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkerScale.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 5e8c0e6a771419447a280526fce4012f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkersChangedEventArgs.cs b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkersChangedEventArgs.cs new file mode 100644 index 0000000..4a41e15 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkersChangedEventArgs.cs @@ -0,0 +1,59 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System.Collections.Generic; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Event arguments for the event. + /// Following design pattern set by + /// + public struct ARMarkersChangedEventArgs + { + /// + /// The list of s added since the last event. + /// + public IReadOnlyList added { get; } + + /// + /// The list of s udpated since the last event. + /// + public IReadOnlyList updated { get; } + + /// + /// The list of s removed since the last event. + /// + public IReadOnlyList removed { get; } + + /// + /// Default empty list of s. + /// + private static IReadOnlyList empty { get; } = new ARMarker[0]; + + /// + /// Constructs an . + /// + /// The list of s added since the last event. + /// The list of s updated since the last event. + /// The list of s removed since the last event. + internal ARMarkersChangedEventArgs( + IReadOnlyList added, + IReadOnlyList updated, + IReadOnlyList removed) + { + this.added = (added != null) ? added : empty; + this.updated = (updated != null) ? updated : empty; + this.removed = (removed != null) ? removed : empty; + } + + /// + /// Generates a string representation of this . + /// + /// A string representation of this . + public override string ToString() + { + return string.Format("Added: {0}, Updated: {1}, Removed: {2}", added.Count, updated.Count, removed.Count); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkersChangedEventArgs.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkersChangedEventArgs.cs.meta new file mode 100644 index 0000000..dd20eb9 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/ARMarkersChangedEventArgs.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c1c7fd4858958fb4fa6a7f24571915fa +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarker.cs b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarker.cs new file mode 100644 index 0000000..7bbc2a4 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarker.cs @@ -0,0 +1,119 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Runtime.InteropServices; +using UnityEngine; +using UnityEngine.XR.ARSubsystems; +using Pose = UnityEngine.Pose; + +namespace Microsoft.MixedReality.OpenXR.ARSubsystems +{ + /// + /// The session-relative data associated with a marker. + /// Following design pattern set by + /// + /// + [StructLayout(LayoutKind.Sequential)] + public struct XRMarker : ITrackable + { + private static readonly XRMarker s_Default = new XRMarker( + TrackableId.invalidId, + Pose.identity, + TrackingState.None, + Vector2.zero, + Vector2.zero, + 0.0f, + TransformMode.MostStable, + ARMarkerType.QRCode, + IntPtr.Zero); + + /// + /// Gets a default-initialized . This can be + /// different from the zero-initialized version, e.g., the + /// is Pose.identity instead of zero-initialized. + /// + internal static XRMarker defaultValue => s_Default; + + /// + /// Constructs a new . This is just a data container + /// for a marker's session relative data. These are typically created by + /// . + /// + /// The associated with the marker. + /// The Pose associated with the marker. + /// The associated with the marker. + /// The center of the marker, in marker space (relative to ). + /// The dimensions associated with the marker. + /// The time when the marker was last seen. + /// The type of the marker. Currently only markers of type QRCode are supported. + /// The native pointer associated with the marker. + internal XRMarker( + TrackableId trackableId, + Pose pose, + TrackingState trackingState, + Vector2 center, + Vector2 size, + float lastSeenTime, + TransformMode transformMode, + ARMarkerType markerType, + IntPtr nativePtr) + { + this.trackableId = trackableId; + this.pose = pose; + this.trackingState = trackingState; + this.center = center; + this.size = size; + this.lastSeenTime = lastSeenTime; + this.transformMode = transformMode; + this.markerType = markerType; + this.nativePtr = nativePtr; + } + + /// + /// The associated with this marker. + /// + public TrackableId trackableId { get; } + + /// + /// The Pose, in session space, of the marker. + /// + public Pose pose { get; internal set; } + + /// + /// The of the marker. + /// + public TrackingState trackingState { get; internal set; } + + /// + /// The center of the marker in marker space (relative to its ). + /// + public Vector2 center { get; internal set; } + + /// + /// The size (dimensions) of the marker in meters. + /// + public Vector2 size { get; internal set; } + + /// + /// The time the marker was last seen. + /// + public float lastSeenTime { get; internal set; } + + /// + /// The type of transform on the marker. + /// + public TransformMode transformMode { get; internal set; } + + /// + /// The type of the marker. Currently we only support markers of type QRCode. + /// + public ARMarkerType markerType { get; } + + /// + /// A native pointer associated with this marker. + /// The data pointer to by this pointer is implementation defined. + /// + public IntPtr nativePtr { get; } + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarker.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarker.cs.meta new file mode 100644 index 0000000..29a09d6 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarker.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 110427922f4c32d4caef07af374f874a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystem.cs b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystem.cs new file mode 100644 index 0000000..06e9c48 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystem.cs @@ -0,0 +1,156 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using Unity.Collections; +using UnityEngine.SubsystemsImplementation; +using UnityEngine.XR.ARSubsystems; + +namespace Microsoft.MixedReality.OpenXR.ARSubsystems +{ + /// + /// Base class for marker subsystems. + /// + /// + /// This subsystem surfaces information regarding the detection of markers such as QRCodes in the physical environment. + /// + public class XRMarkerSubsystem + : TrackingSubsystem + { + public XRMarkerSubsystem() { } + + /// + /// Get or set the list of marker types that will be detected. + /// + internal ARMarkerType[] EnabledMarkerTypes + { + get => provider.EnabledMarkerTypes; + set => provider.EnabledMarkerTypes = value; + } + + /// + /// Default for newly detected markers. + /// + public TransformMode DefaultTransformMode + { + get => provider.DefaultTransformMode; + set => provider.DefaultTransformMode = value; + } + + /// + /// Get the changes to markers (added, updated, and removed) since the last call to . + /// + /// An Allocator to use when allocating the returned NativeArrays. + /// + /// that describes the markers that have been added, updated, and removed + /// since the last call to . The caller owns the memory allocated with Allocator. + /// + public override TrackableChanges GetChanges(Allocator allocator) + { + var changes = provider.GetChanges(XRMarker.defaultValue, allocator); +#if DEVELOPMENT_BUILD || UNITY_EDITOR + m_ValidationUtility.ValidateAndDisposeIfThrown(changes); +#endif + return changes; + } + + /// + /// Set transform mode of an existing . + /// + /// The of the marker to be transformed. + /// The to be applied. + public void SetTransformMode(TrackableId trackableId, TransformMode transformMode) + { + provider.SetTransformMode(trackableId, transformMode); + } + + /// + /// Get raw data for an existing . + /// + /// The of the marker. + public NativeArray GetRawData(TrackableId trackableId, Allocator allocator) + { + return provider.GetRawData(trackableId, allocator); + } + + /// + /// Get decoded string for an existing . + /// + /// The of the marker to be transformed. + public string GetDecodedString(TrackableId trackableId) + { + return provider.GetDecodedString(trackableId); + } + + /// + /// Get QR code properties for an existing of type . + /// + /// The of the QRCode marker. + public QRCodeProperties GetQRCodeProperties(TrackableId trackableId) + { + return provider.GetQRCodeProperties(trackableId); + } + + /// + /// The API that derived classes must implement. + /// + public abstract class Provider : SubsystemProvider + { + /// + /// Get the changes to markers (added, updated, and removed) since the last call to + /// . + /// + /// + /// The default marker. This should be used to initialize the returned NativeArrays for backwards compatibility. + /// See . + /// + /// An Allocator to use when allocating the returned NativeArrays. + /// + /// describing the markers that have been added, updated, and removed + /// since the last call to . The changes should be allocated using + /// . + /// + public abstract TrackableChanges GetChanges(XRMarker defaultMarker, Allocator allocator); + + /// + /// Set transform mode of an existing . + /// + /// The of the marker to be transformed. + /// The to be applied. + public abstract void SetTransformMode(TrackableId trackableId, TransformMode transformMode); + + /// + /// Get raw data for an existing . + /// + /// The of the marker. + public abstract NativeArray GetRawData(TrackableId trackableId, Allocator allocator); + + /// + /// Get decoded string for an existing . + /// + /// The of the marker. + public abstract string GetDecodedString(TrackableId trackableId); + + /// + /// Get QR code properties for an existing of type . + /// + /// The of the QRCode marker. + public abstract QRCodeProperties GetQRCodeProperties(TrackableId trackableId); + + /// + /// Get or set the list of marker types that will be detected. + /// + internal abstract ARMarkerType[] EnabledMarkerTypes { get; set; } + + /// + /// Default for newly detected markers. + /// + internal abstract TransformMode DefaultTransformMode { get; set; } + } + +#if DEVELOPMENT_BUILD || UNITY_EDITOR + private ValidationUtility m_ValidationUtility = + new ValidationUtility(); +#endif + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystem.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystem.cs.meta new file mode 100644 index 0000000..2451cd1 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystem.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 69b633547077b49479b83ca0bea9508e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystemDescriptor.cs b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystemDescriptor.cs new file mode 100644 index 0000000..8804b01 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystemDescriptor.cs @@ -0,0 +1,53 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using UnityEngine.SubsystemsImplementation; + +namespace Microsoft.MixedReality.OpenXR.ARSubsystems +{ + public class XRMarkerSubsystemDescriptor : + SubsystemDescriptorWithProvider + { + internal struct Cinfo + { + /// + /// The string identifier for a specific implementation. + /// + internal string id { get; set; } + + /// + /// Specifies the provider implementation type to use for instantiation. + /// + /// + /// The provider implementation type to use for instantiation. + /// + internal Type providerType { get; set; } + + /// + /// Specifies the XRMarkerSubsystem-derived type that forwards casted calls to its provider. + /// + /// + /// The type of the subsystem to use for instantiation. If null, XRMarkerSubsystem will be instantiated. + /// + internal Type subsystemTypeOverride { get; set; } + } + + /// + /// Creates a new subsystem descriptor and registers it with the SubsystemManager. + /// + /// Construction info for the descriptor. + internal static void Create(Cinfo cinfo) + { + var descriptor = new XRMarkerSubsystemDescriptor(cinfo); + SubsystemDescriptorStore.RegisterDescriptor(descriptor); + } + + private XRMarkerSubsystemDescriptor(Cinfo cinfo) + { + id = cinfo.id; + providerType = cinfo.providerType; + subsystemTypeOverride = cinfo.subsystemTypeOverride; + } + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystemDescriptor.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystemDescriptor.cs.meta new file mode 100644 index 0000000..8f64ce2 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ARMarker/XRMarkerSubsystemDescriptor.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: ad94913995be62a4da5c8599be53c05d +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/AnchorConverter.cs b/com.microsoft.mixedreality.openxr/Runtime/API/AnchorConverter.cs new file mode 100644 index 0000000..0d1c487 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/AnchorConverter.cs @@ -0,0 +1,185 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Runtime.InteropServices; +using TrackableId = UnityEngine.XR.ARSubsystems.TrackableId; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Provides helper functions to convert an Unity Anchor object to underlying OpenXR anchor handle or SpatialAnchor COM object. + /// + public static class AnchorConverter + { + private static MixedRealityFeaturePlugin Feature => OpenXRFeaturePlugin.Feature; + + /// + /// Get the OpenXR handle of the given nativePtr from ARAnchor or XRAnchor object if available, or return 0. + /// + /// The nativePtr obtained from either XRAnchor.nativePtr or ARAnchor.nativePtr. + /// XrAnchorMSFT handle that represents the underlying OpenXR anchor of given nativePtr, or 0 when such associated handle cannot be found. + public static ulong ToOpenXRHandle(IntPtr nativePtr) + { + if (nativePtr == null) + return 0; + + NativeAnchorData data = Marshal.PtrToStructure(nativePtr); + if (data.version == 1) + { + return data.anchorHandle; + } + return 0; + } + + /// + /// Create a new ARAnchor from the given OpenXR XRSpatialAnchorMSFT handle. + /// + /// A valid OpenXR XRSpatialAnchorMSFT handle. + /// Returns the trackable id representing the Unity anchor or if the conversion was unsuccessful. + /// The newly created TrackableId will not be added to collection until the next frame's Update. + /// The app should listen to the event for the added ARAnchor object with the returned trackableId. + public static TrackableId CreateFromOpenXRHandle(ulong openxrAnchorHandle) + { + if (Feature.IsValidAndEnabled() && openxrAnchorHandle != 0) + { + Guid guid = NativeLib.TryCreateARAnchorFromOpenXRHandle(openxrAnchorHandle); + return FeatureUtils.ToTrackableId(guid); + } + return TrackableId.invalidId; + } + + /// + /// Get a COM wrapper object of Windows.Perception.Spatial.SpatialAnchor from the given ARAnchor's nativePtr. + /// + /// The nativePtr obtained from either XRAnchor.nativePtr or ARAnchor.nativePtr. + /// The COM wrapper object of Windows.Perception.Spatial.SpatialAnchor, or null when the conversion failed. + public static object ToPerceptionSpatialAnchor(IntPtr nativePtr) + { + if (Feature.IsValidAndEnabled() && nativePtr != IntPtr.Zero) + { + IntPtr unknown = NativeLib.TryAcquirePerceptionSpatialAnchor(ToOpenXRHandle(nativePtr)); + if (unknown != IntPtr.Zero) + { + object result = Marshal.GetObjectForIUnknown(unknown); + Marshal.Release(unknown); // Balance the ref count because "feature.TryAcquire" increment it on return. + return result; + } + } + return null; + } + + /// + /// Get a COM wrapper object of Windows.Perception.Spatial.SpatialAnchor from the given TrackableId. + /// If failed, the function returns nullptr. + /// + /// An existing XRAnchor or ARAnchor's ID. + /// The COM wrapper object of Windows.Perception.Spatial.SpatialAnchor, or null when the conversion failed. + public static object ToPerceptionSpatialAnchor(TrackableId trackableId) + { + if (Feature.IsValidAndEnabled() && trackableId != TrackableId.invalidId) + { + IntPtr unknown = NativeLib.TryAcquirePerceptionSpatialAnchor(FeatureUtils.ToGuid(trackableId)); + if (unknown != IntPtr.Zero) + { + object result = Marshal.GetObjectForIUnknown(unknown); + Marshal.Release(unknown); // Balance the ref count because "feature.TryAcquire" increment it on return. + return result; + } + } + return null; + } + + /// + /// Creating a new ARAnchor from the given Windows.Perception.Spatial.SpatialAnchor. + /// If failed, the function returns TrackableId.invalidId. + /// Creates an OpenXR anchor from a Windows.Perception.Spatial.SpatialAnchor and reports it to Unity. + /// + /// Must be a Windows.Perception.Spatial.SpatialAnchor. + /// Returns the trackable id representing the Unity anchor or if the conversion was unsuccessful. + /// The newly created TrackableId will not be added to collection until the next frame's Update. + /// The app should listen to the event for the added ARAnchor object with the returned trackableId. + [Obsolete("Obsolete and will be removed in future releases. Use the `CreateFromPerceptionSpatialAnchor` function instead.")] + public static TrackableId FromPerceptionSpatialAnchor(object spatialAnchor) + { + return CreateFromPerceptionSpatialAnchor(spatialAnchor); + } + + /// + /// Creating a new ARAnchor from the given Windows.Perception.Spatial.SpatialAnchor. + /// If failed, the function returns TrackableId.invalidId. + /// Creates an OpenXR anchor from a Windows.Perception.Spatial.SpatialAnchor and reports it to Unity. + /// + /// Must be a Windows.Perception.Spatial.SpatialAnchor. + /// Returns the trackable id representing the Unity anchor or if the conversion was unsuccessful. + /// The newly created TrackableId will not be added to collection until the next frame's Update. + /// The app should listen to the event for the added ARAnchor object with the returned trackableId. + public static TrackableId CreateFromPerceptionSpatialAnchor(object spatialAnchor) + { + if (Feature.IsValidAndEnabled() && spatialAnchor != null) + { + Guid guid = NativeLib.TryCreateARAnchorFromPerceptionAnchor(spatialAnchor); + return FeatureUtils.ToTrackableId(guid); + } + return TrackableId.invalidId; + } + + /// + /// Replaces the underlying platform anchor for an existing XRAnchor/ARAnchor represented by the + /// given TrackableId, so the Unity anchor will instead be located by the given SpatialAnchor. + /// + /// Use this function instead of to avoid creating new ARAnchor on every new platform anchor. + /// Must be a Windows.Perception.Spatial.SpatialAnchor. + /// An id representing an existing XRAnchor/ARAnchor. + /// Returns the trackable id representing the Unity anchor or if the conversion was unsuccessful. + public static TrackableId ReplaceSpatialAnchor(object spatialAnchor, TrackableId existingId) + { + if (Feature.IsValidAndEnabled() && spatialAnchor != null) + { + Guid guid = NativeLib.TryAcquireAndReplaceXrSpatialAnchor(spatialAnchor, FeatureUtils.ToGuid(existingId)); + return FeatureUtils.ToTrackableId(guid); + } + return TrackableId.invalidId; + } + } + + namespace ARSubsystems + { + /// + /// Provides extension function to convert an XRAnchor object to underlying OpenXR anchor handle. + /// + [System.Obsolete("Obsolete and will be removed in future releases. Use AnchorConverter.ToOpenXRHandle() function instead.", true)] + public static class XRAnchorExtensions + { + /// + /// Get the native OpenXR handle of the given XRAnchor object if available, or return 0. + /// + /// A valid object. + /// XrAnchorMSFT handle that represents the underlying OpenXR anchor, or 0 when such associated handle cannot be found. + public static ulong GetOpenXRHandle(this UnityEngine.XR.ARSubsystems.XRAnchor anchor) + { + return anchor == null ? 0 : AnchorConverter.ToOpenXRHandle(anchor.nativePtr); + } + } + } + + namespace ARFoundation + { + /// + /// Provides extension function to convert an ARAnchor object to underlying OpenXR anchor handle. + /// + [System.Obsolete("Obsolete and will be removed in future releases. Use AnchorConverter.ToOpenXRHandle() function instead.", true)] + public static class ARAnchorExtensions + { + /// + /// Get the native OpenXR handle of the given ARAnchor object if available, or return 0. + /// + /// A valid object. + /// XrAnchorMSFT handle that represents the underlying OpenXR anchor, or 0 when such associated handle cannot be found. + public static ulong GetOpenXRHandle(this UnityEngine.XR.ARFoundation.ARAnchor anchor) + { + return anchor == null ? 0 : AnchorConverter.ToOpenXRHandle(anchor.nativePtr); + } + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/AnchorConverter.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/AnchorConverter.cs.meta new file mode 100644 index 0000000..a8b5f77 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/AnchorConverter.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 7dd27931fc201ae46a11479a3446986c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/AppRemoting.cs b/com.microsoft.mixedreality.openxr/Runtime/API/AppRemoting.cs new file mode 100644 index 0000000..99ae084 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/AppRemoting.cs @@ -0,0 +1,842 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Runtime.InteropServices; +using System.Security.Cryptography.X509Certificates; +using Unity.Collections; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR.Remoting +{ + /// + /// Provides information and configuration for creating a Holographic Remoting remote app. + /// + /// + /// Please note that client/server and player/remote are orthogonal concepts in remoting. + /// Holographic Remoting remote app can act as either a + /// Server - when listening to incoming connection from a Holographic Remoting player app (Client) + /// (or) + /// Client - when connecting to Holographic Remoting player app (Server) that is listening to incoming connections. + /// For more details, please reference the Holographic Remoting Terminology. + /// + public static class AppRemoting + { + /// + /// Starts connecting with given Holographic Remoting remote app (Client) configuration and initializes XR. + /// + /// + /// The remote app (Client) will try and connect to remote player (Server) listening for incoming connections, + /// and after a successful connection, XR experience starts. If the connection fails for any reason, try to connect by calling the coroutine again. + /// This method must be run as a coroutine itself, as initializing XR has to happen in a coroutine. + /// + /// The set of parameters to use for remoting. + [Obsolete("This method is obsolete. Use the ConnectToPlayer() method instead.", false)] + public static System.Collections.IEnumerator Connect(RemotingConfiguration configuration) + { + AppRemotingSubsystem subsystem = AppRemotingSubsystem.GetCurrent(); + if (!subsystem.IsAppRemotingEnabled()) + { + Debug.LogWarning($"The Connect() function is not supported without enabling the \"Holographic Remoting remote app\" feature in OpenXR project settings."); + yield break; + } + if (subsystem.InPlayModeRemoting()) + { + Debug.LogWarning($"The Connect() function is not supported in PlayMode Remoting."); + yield break; + } + yield return subsystem.ConnectLegacy(configuration); + } + + /// + /// Starts connecting with given Holographic Remoting remote app (Client) configuration and initializes XR. + /// + /// + /// This remote app (Client) will try and connect to a remote player (Server) listening for incoming connections. + /// After a successful connection, the XR experience will be started. Apps can use to + /// monitor the ongoing connection and use to drive future connections. + /// This method will return quickly and is safe for use in UI threads. + /// + /// The set of parameters to use for remoting. + /// + /// During this connection, for the duration when is false, + /// calling will be a no-op and an error message will appear in the logs. + /// If the app wants to retry the connection, it should wait for "IsReadyToStart" to changed to true, + /// or monitor the "ReadyToStart" event. + /// + public static void StartConnectingToPlayer(RemotingConnectConfiguration connectConfiguration) + { + AppRemotingSubsystem subsystem = AppRemotingSubsystem.GetCurrent(); + if (!subsystem.IsAppRemotingEnabled()) + { + Debug.LogWarning($"The StartConnectingToPlayer() function is not supported without enabling the \"Holographic Remoting remote app\" feature in OpenXR project settings."); + return; + } + if (subsystem.InPlayModeRemoting()) + { + Debug.LogWarning($"The StartConnectingToPlayer() function is not supported in PlayMode Remoting."); + return; + } + subsystem.StartConnecting(connectConfiguration); + } + + /// + /// Starts listening with given Holographic Remoting remote app (Server) configuration and initializes XR. + /// + /// + /// The remote app (Server) will be waiting for remote player (Client) to connect, and after a successful connection, XR experience starts. + /// If the connection fails for any reason, it will retry listening for incoming connection until some other method is called. + /// This method must be run as a coroutine itself, as initializing XR has to happen in a coroutine. + /// + /// The set of parameters to use for remoting. + /// Action callback to signal listen complete. A new Connect or Listen coroutine can safely be started after this callback. + /// + /// During a Listen coroutine, if the remote app calls or function, + /// the second call will fail, because there can only be a single outstanding remoting connection. The Listen coroutine will wait indefinitely for new connections + /// until the remote app calls function to stop listening. During this coroutine, there could be multiple remoting connections + /// and the may change multiple times. If the remote app wants to know the completion of above listening session, + /// it can use the `onRemotingListenCompleted` callback here. After the callback, the Listen coroutine will complete, and the application can safely call the `Connect` or `Listen` functions again. + /// + [Obsolete("This method is obsolete. Use the StartListeningForPlayer() instead.", false)] + public static System.Collections.IEnumerator Listen(RemotingListenConfiguration listenConfiguration, Action onRemotingListenCompleted = null) + { + AppRemotingSubsystem subsystem = AppRemotingSubsystem.GetCurrent(); + if (!subsystem.IsAppRemotingEnabled()) + { + Debug.LogWarning($"The Listen() function is not supported without enabling the \"Holographic Remoting remote app\" feature in OpenXR project settings."); + if (onRemotingListenCompleted != null) + { + onRemotingListenCompleted.Invoke(); + } + yield break; + } + if (subsystem.InPlayModeRemoting()) + { + Debug.LogWarning($"The Listen() function is not supported in PlayMode Remoting."); + if (onRemotingListenCompleted != null) + { + onRemotingListenCompleted.Invoke(); + } + yield break; + } + yield return subsystem.ListenLegacy(listenConfiguration, ListenMode.LegacyListen, onRemotingListenCompleted); + } + + /// + /// Starts listening with given Holographic Remoting remote app (Server) configuration and initializes XR. + /// + /// + /// The remote app (Server) will be waiting for remote player (Client) to connect, and after a successful connection, XR experience starts. + /// This method will return quickly and is safe for use in UI threads. Apps can use to monitor the ongoing connection. + /// If the connection fails for any reason, it will retry listening for incoming connection until or is called. + /// + /// The set of parameters to use for remoting. + /// + /// During this connection, for the duration when is false, + /// calling will be a no-op and an error message will appear in the logs. + /// If the app wants to retry the connection, it should wait for "IsReadyToStart" to changed to true, + /// or monitor the "ReadyToStart" event. + /// + public static void StartListeningForPlayer(RemotingListenConfiguration listenConfiguration) + { + AppRemotingSubsystem subsystem = AppRemotingSubsystem.GetCurrent(); + if (!subsystem.IsAppRemotingEnabled()) + { + Debug.LogWarning($"The StartListeningForPlayer() function is not supported without enabling the \"Holographic Remoting remote app\" feature in OpenXR project settings."); + return; + } + if (subsystem.InPlayModeRemoting()) + { + Debug.LogWarning($"The StartListeningForPlayer() function is not supported in PlayMode Remoting."); + return; + } + subsystem.StartListening(listenConfiguration, ListenMode.Listen, null); + } + + /// + /// Disconnects the remote app (Client/Server) from the remote player (Client/Server) and stops the active XR session. + /// + /// + /// Disconnects network connection between remote app and remote player, stops , and coroutines. + /// It does not stop coroutine, use instead. `Disconnect` is not equivalent to the completion of `ConnectToPlayer` coroutine, + /// please use a wrapper coroutine, as explained in to identify completion. + /// + public static void Disconnect() + { + AppRemotingSubsystem subsystem = AppRemotingSubsystem.GetCurrent(); + if (!subsystem.IsAppRemotingEnabled()) + { + Debug.LogWarning($"The Disconnect() function is not supported without enabling the \"Holographic Remoting remote app\" feature in OpenXR project settings."); + return; + } + if (subsystem.InPlayModeRemoting()) + { + Debug.LogWarning($"The Disconnect() function is not supported in PlayMode Remoting."); + return; + } + subsystem.Disconnect(); + } + + /// + /// Stops listening on the remote app (Server) for incoming connections from the remote player (Client) and stops the active XR session. + /// + /// + /// Disconnects any outstanding Listen session and exits coroutine, throws an error if used with coroutine. + /// `StopListening` is not equivalent to the completion of `StartListeningForPlayer` coroutine, please use a wrapper coroutine as explained in + /// to identify completion. + /// + public static void StopListening() + { + AppRemotingSubsystem subsystem = AppRemotingSubsystem.GetCurrent(); + if (!subsystem.IsAppRemotingEnabled()) + { + Debug.LogWarning($"The StopListening() function is not supported without enabling the \"Holographic Remoting remote app\" feature in OpenXR project settings."); + return; + } + if (subsystem.InPlayModeRemoting()) + { + Debug.LogWarning($"The StopListening() function is not supported in PlayMode Remoting."); + return; + } + subsystem.StopListening(); + } + + /// + /// Indicates whether a remoting connection is ready to be started using + /// or . + /// + /// + /// This functionality needs the "Holographic Remoting remote app" feature in OpenXR project settings to be enabled and returns false otherwise. + /// This functionality always returns false in Holographic PlayMode Remoting. + /// + public static bool IsReadyToStart + { + get => AppRemotingSubsystem.GetCurrent().IsReadyToStart(); + } + + /// + /// Provides information on the current remoting session, if one exists. + /// + /// The current connection state of the remote session. + /// If the connection state is disconnected, this helps explain why. + /// Whether the information was successfully retrieved. + /// + /// This functionality needs the "Holographic Remoting remote app" feature in OpenXR project settings to be enabled and returns false otherwise. + /// This functionality always returns false in Holographic PlayMode Remoting. + /// + public static bool TryGetConnectionState(out ConnectionState connectionState, out DisconnectReason disconnectReason) + { + connectionState = ConnectionState.Disconnected; + disconnectReason = DisconnectReason.None; + AppRemotingSubsystem subsystem = AppRemotingSubsystem.GetCurrent(); + if (!subsystem.IsAppRemotingEnabled() || subsystem.InPlayModeRemoting()) + { + return false; + } + return subsystem.TryGetConnectionState(out connectionState, out disconnectReason); + } + + /// + /// To locate the `XR_REMOTING_REFERENCE_SPACE_TYPE_USER_MSFT` reference space in Unity's scene origin space in the remote app. For more details, reference the + /// Coordinate System Synchronization with Holographic Remoting. + /// + /// Specify the to locate the user reference space. + /// Output the pose of the user reference space in the Unity's scene origin space. + /// Returns true when the user reference space is tracking and output pose is valid to be used. + /// Returns false when the user reference space lost tracking or it's not properly set up. + /// + /// This functionality needs the "Holographic Remoting remote app" feature in OpenXR project settings to be enabled and returns false otherwise. + /// This functionality always returns false in Holographic PlayMode Remoting. + /// + public static bool TryLocateUserReferenceSpace(FrameTime frameTime, out Pose pose) + { + pose = Pose.identity; + AppRemotingSubsystem subsystem = AppRemotingSubsystem.GetCurrent(); + if (!subsystem.IsAppRemotingEnabled() || subsystem.InPlayModeRemoting()) + { + return false; + } + return subsystem.TryLocateUserReferenceSpace(frameTime, out pose); + } + + /// + /// Convert the time from a player app QPC time to the synchronized remote app QPC time. + /// + /// The performance count obtained in the player app using QueryPerformanceCounter. + /// Output the synchronized performance count as if it is using QueryPerformanceCounter in the remote app at the same time. + /// The output will be 0, indicating invalid time, if the function returns false. + /// Returns true when the time is successfully converted. + /// Returns false when the time synchronization between remote and player app is not yet established. + /// + /// + /// This functionality needs the "Holographic Remoting remote app" feature in OpenXR project settings to be enabled and returns false otherwise. + /// This functionality always returns false in Holographic PlayMode Remoting. + /// + public static bool TryConvertToRemoteTime(long playerPerformanceCount, out long remotePerformanceCount) + { + remotePerformanceCount = 0; // default to invalid timestamp + AppRemotingSubsystem subsystem = AppRemotingSubsystem.GetCurrent(); + if (!subsystem.IsAppRemotingEnabled() || subsystem.InPlayModeRemoting()) + { + return false; + } + return subsystem.TryConvertToRemoteTime(playerPerformanceCount, out remotePerformanceCount); + } + + /// + /// Convert the time from a remote app QPC time to the synchronized player app QPC time. + /// + /// The performance count obtained in the remote app using QueryPerformanceCounter. + /// Output the synchronized performance count as if it is using QueryPerformanceCounter in the player app at the same time. + /// The output will be 0, indicating invalid time, if the function returns false. + /// Returns true when the time is successfully converted. + /// Returns false when the time synchronization between remote and player app is not yet established. + /// + /// + /// This functionality needs the "Holographic Remoting remote app" feature in OpenXR project settings to be enabled and returns false otherwise. + /// This functionality always returns false in Holographic PlayMode Remoting. + /// + public static bool TryConvertToPlayerTime(long remotePerformanceCount, out long playerPerformanceCount) + { + playerPerformanceCount = 0; // default to invalid timestamp + AppRemotingSubsystem subsystem = AppRemotingSubsystem.GetCurrent(); + if (!subsystem.IsAppRemotingEnabled() || subsystem.InPlayModeRemoting()) + { + return false; + } + return subsystem.TryConvertToPlayerTime(remotePerformanceCount, out playerPerformanceCount); + } + + /// + /// Event triggered when changes from false to true. + /// + /// + /// Typically, applications can use this event to re-enable UX allowing the user to start a new remoting connection, as this + /// event indicates previous remoting sessions have fully completed and AppRemoting is ready for a new connection to start. + /// This event will only be triggered if the "Holographic Remoting remote app" feature in OpenXR project settings is enabled and + /// is never triggered in Holographic PlayMode Remoting. + /// + public static event ReadyToStartDelegate ReadyToStart + { + add + { + AppRemotingSubsystem.GetCurrent().ReadyToStart += value; + } + remove + { + AppRemotingSubsystem.GetCurrent().ReadyToStart -= value; + } + } + + /// + /// Event triggered when the connection between remote app (Client/Server) and player is successfully established. + /// + /// + /// This event will only be triggered if the "Holographic Remoting remote app" feature in OpenXR project settings is enabled and + /// is never triggered in Holographic PlayMode Remoting. + /// + public static event ConnectedDelegate Connected + { + add + { + AppRemotingSubsystem.GetCurrent().Connected += value; + } + remove + { + AppRemotingSubsystem.GetCurrent().Connected -= value; + } + } + + /// + /// Event triggered when the connection between remote app (Client/Server) and player is disconnecting. + /// + /// + /// This event might be triggered several times during the `StartListeningForPlayer` coroutine. + /// This may also be triggered without a corresponding "Connected" event. + /// This event will only be triggered if the "Holographic Remoting remote app" feature in OpenXR project settings is enabled and + /// is never triggered in Holographic PlayMode Remoting. + /// + public static event DisconnectingDelegate Disconnecting + { + add + { + AppRemotingSubsystem.GetCurrent().Disconnecting += value; + } + remove + { + AppRemotingSubsystem.GetCurrent().Disconnecting -= value; + } + } + } + + /// + /// Describes the event handler that can be implemented by remote app to get notified when changes from false to true. + /// + public delegate void ReadyToStartDelegate(); + + /// + /// Describes the event handler that can be implemented by remote app to get notified on a event. + /// + public delegate void ConnectedDelegate(); + + /// + /// Describes the event handler that can be implemented by remote app to get notified on a event. + /// + /// The reason for disconnecting + public delegate void DisconnectingDelegate(DisconnectReason disconnectReason); + + /// + /// Describes the preferred video codec to use for the connection. + /// + public enum RemotingVideoCodec + { + /// + /// Represents HEVC video codec preferred, fall back to H264 if HEVC is not supported by all participants. + /// + Auto = 0, + /// + /// Represents HEVC video codec. + /// + H265, + /// + /// Represents H264 video codec. + /// + H264, + } + + /// + /// Describes the preferred audio capture mode to use for the connection. + /// + /// + /// "EnableAudio" parameter in or has to be set to true for this selection to take effect. + /// This is only supported if the remote app runs on Windows 10 build 20348 or later.The default behavior with older Windows builds is capturing the entire system audio, regardless of the selection. + /// + public enum RemotingAudioCaptureMode + { + /// + /// Represents capturing the whole system wide audio on the remote app's end and stream it to the player app. + /// + SystemWideCapture = 0, + + /// + /// Represents capturing only the in app audio on the remote app's end and stream it to the player app. + /// + InAppCapture, + } + + /// + /// Specifies the configuration for the remote app (Client) to initiate a remoting connection. + /// + [Serializable, StructLayout(LayoutKind.Sequential, Pack = 8)] + [Obsolete("This struct is obsolete. Use 'RemotingConnectConfiguration' instead.", false)] + public struct RemotingConfiguration + { + /// + /// The host name or IP address of the player running in network server mode to connect to. + /// + public string RemoteHostName; + + /// + /// The port number of the server's handshake port. + /// + public ushort RemotePort; + + /// + /// The max bitrate in Kbps to use for the connection. + /// + public uint MaxBitrateKbps; + + /// + /// The video codec to use for the connection. + /// + public RemotingVideoCodec VideoCodec; + + /// + /// Enable/disable audio remoting. + /// + public bool EnableAudio; + } + + /// + /// Specifies the configuration for the remote app (Client) to initiate a remoting connection. + /// + [Serializable, StructLayout(LayoutKind.Sequential, Pack = 8)] + public struct RemotingConnectConfiguration + { + /// + /// The host name or IP address of the player running in network server mode to connect to. + /// + public string RemoteHostName; + + /// + /// The port number of the server's handshake port. + /// + public ushort RemotePort; + + /// + /// The max bitrate in Kbps to use for the connection. + /// + public uint MaxBitrateKbps; + + /// + /// The video codec to use for the connection. + /// + public RemotingVideoCodec VideoCodec; + + /// + /// Enable/disable audio remoting. + /// + public bool EnableAudio; + + /// + /// Audio capture mode for audio remoting. + /// + public RemotingAudioCaptureMode AudioCaptureMode; + + /// + /// Configuration to enable secure connection. + /// + public SecureRemotingConnectConfiguration? secureConnectConfiguration; + } + + /// + /// Specifies the configuration for the remote app (Client) to initiate a secure remoting connection. + /// + public struct SecureRemotingConnectConfiguration + { + /// + /// Shared token between the remote app (Client) and remote player (Server). + /// Used by remote player (Server) to validate the remote app (Client) + /// before establishing a secure connection. For more details, reference the + /// client to server authentication. + /// + public string AuthenticationToken; + + /// + /// Specify whether to request platform's default validation on the remote player's certificate using + /// the validation functions of the underlying operating system or cryptography library.This bool is + /// taken into account only when callback is implemented + /// by the remote app (Client). Otherwise, if the callback is not provided system validation is performed + /// regardless and is used for validating the remoting player's(Server) certificate. + /// + public bool PerformSystemValidation; + + /// + /// The callback function to validate the certificate chain provided by the remote player (Server). + /// + public SecureRemotingValidateServerCertificateDelegate ValidateServerCertificateCallback; + } + + /// + /// Defines the result of a certificate validation for secure remoting connection. + /// + [Serializable, StructLayout(LayoutKind.Sequential, Pack = 8)] + public struct SecureRemotingCertificateValidationResult + { + /// + /// Specifies whether the certificate can be traced back to a trusted root + /// + public bool TrustedRoot; + + /// + /// Specifies whether the certificate has been revoked + /// + public bool Revoked; + + /// + /// Specifies whether the certificate is outside of its validity period (expired or not yet valid) + /// + public bool Expired; + + /// + /// Specifies whether the allowed certificate usage is not compatible with its actual usage + /// + public bool WrongUsage; + + /// + /// Specifies whether the revocation check failed + /// + public bool RevocationCheckFailed; + + /// + /// Specifies whether the certificate to validate and/or certificate(s) in the certificate chain + /// contained invalid data and could not be examined + /// + public bool InvalidCertOrChain; + + /// + /// Specifies the result of name validation, if there is a name mismatch between + /// the name of the host presenting the certificate and the certificate subject + /// + public SecureRemotingNameValidationResult NameValidationResult; + } + + /// + /// Describes whether the name of the host presenting the certificate does not match the certificate subject. + /// + public enum SecureRemotingNameValidationResult + { + /// + /// Represents that the name match cannot be reliably determined. + /// + ResultIndeterminate = 0, + + /// + /// Represents the name of the host presenting the certificate matches the certificate subject. + /// + ResultMatch = 1, + + /// + /// Represents the name of the host presenting the certificate does not match the certificate subject. + /// + ResultMismatch = 2, + } + + /// + /// Defines the callback that can be provided by the remote app (Client) for custom validation of remote player's(Server) certificate chain. + /// + /// The name of the host the connection is being established with + /// The certificate chain that the server provides when initiating the secure connection + /// The result of system validation is provided by remoting runtime if system validation is requested by the remote app + /// Returns custom server certificated validation result. + /// System validation (as the name suggests) is certificate validation based on the underlying system’s cryptographic APIs and certificate stores. + /// Thus, results can vary depending on OS and local setup. The system validators are implemented in libbasix, which is owned by the RDV project. + /// System validation in this API will forward to libbasix’ default validators for the respective platform + /// + public delegate SecureRemotingCertificateValidationResult SecureRemotingValidateServerCertificateDelegate(string hostName, + X509Certificate2Collection serverCertificateChain, + SecureRemotingCertificateValidationResult? systemValidationResult = null); + + /// + /// Specifies the configuration for the remote app (Server) to initiate a remoting connection in listen mode. + /// + [Serializable, StructLayout(LayoutKind.Sequential, Pack = 8)] + public struct RemotingListenConfiguration + { + /// + /// The host name or IP address of the player running in network server mode to connect to. + /// + public string ListenInterface; + + /// + /// The port number of the server's handshake port. + /// + public ushort HandshakeListenPort; + + /// + /// The port number of the server's transport port. + /// + public ushort TransportListenPort; + + /// + /// The max bitrate in Kbps to use for the connection. + /// + public uint MaxBitrateKbps; + + /// + /// The video codec to use for the connection. + /// + public RemotingVideoCodec VideoCodec; + + /// + /// Enable/disable audio remoting. + /// + public bool EnableAudio; + + /// + /// Audio capture mode for audio remoting. + /// + public RemotingAudioCaptureMode AudioCaptureMode; + + /// + /// Configuration to enable secure connection. + /// + public SecureRemotingListenConfiguration? secureListenConfiguration; + } + + /// + /// Specifies the configuration for the remote app (Server) to initiate a secure remoting connection in linsten mode. + /// + public struct SecureRemotingListenConfiguration + { + /// + /// Byte array containing a certificate store in PKCS#12 format. This store must contain the server + /// certificate and the associated private key; optionally, it can also contain the certificate chain for the server certificate. + /// + public NativeArray Certificate; + + /// + /// The name of the server certificate which is used to identify it in the + /// certificate store. This is usually either the subject common name, or a friendly name assigned to the certificate. + /// + public string SubjectName; + + /// + /// The passphrase needed to decrypt the private key. Can be an empty string + /// if the private key is not encrypted. + /// + /// + /// The passphrase is passed on as UTF-8 encoded and thus, depending on the encoding used when + /// writing the certificate store, passphrases containing characters beyond the 7-bit ASCII range may not work as expected. + /// + public string KeyPassphrase; + + /// + /// The callback function to validate authentication token provided by the remoting player (Client). + /// + public SecureRemotingValidateAuthenticationTokenDelegate ValidateAuthenticationTokenCallback; + } + + /// + /// The callback that needs to be implemented by remote app (Server) in secure listen mode + /// to validate remote player's (Client) authentication token. + /// + /// shared secret between the client and server. + /// Used to validate the remote player to establish a secure connection. + /// Returns true if the token validation succeeds and false if not. + public delegate bool SecureRemotingValidateAuthenticationTokenDelegate(string authenticationTokenToCheck); + + /// + /// Describes the current connection state. + /// + public enum ConnectionState + { + /// + /// Represents that the state is not connected, and no connection attempt is + /// in progress (Client), or not listening for incoming connections (Server). + /// + Disconnected = 0, + /// + /// Represents connecting to server (Client), listening for incoming + /// connections (Server), or performing connection handshake (Client/Server). + /// + Connecting = 1, + /// + /// Represents fully connected, all communication channels established (Client/Server). + /// + Connected = 2, + } + + /// + /// Describes the reason for why the connection disconnected. + /// + public enum DisconnectReason + { + /// + /// The connection succeeded and there was no connection failure. + /// + None = 0, + /// + /// The connection failed for an unknown reason. + /// + Unknown = 1, + /// + /// The secure connection was enabled, but certificate was missing, invalid, or not usable (Server). + /// + NoServerCertificate = 2, + /// + /// The handshake port could not be opened for accepting connections (Server). + /// + HandshakePortBusy = 3, + /// + /// The handshake server is unreachable (Client). + /// + HandshakeUnreachable = 4, + /// + /// The handshake server closed the connection prematurely; likely due to TLS/Plain mismatch or invalid certificate (Client). + /// + HandshakeConnectionFailed = 5, + /// + /// The authentication with the handshake server failed (Client). + /// + AuthenticationFailed = 6, + /// + /// No common compatible remoting version could be determined during handshake (Client). + /// + RemotingVersionMismatch = 7, + /// + /// No common transport protocol could be determined during handshake (Client). + /// + IncompatibleTransportProtocols = 8, + /// + /// The handshake failed for any other reason (Client). + /// + HandshakeFailed = 9, + /// + /// The transport port could not be opened for accepting connections (Server). + /// + TransportPortBusy = 10, + /// + /// The transport server is unreachable (Client). + /// + TransportUnreachable = 11, + /// + /// The transport connection was closed before all communication channels had been set up (Client/Server). + /// + TransportConnectionFailed = 12, + /// + /// The transport connection was closed due to protocol version mismatch (Client/Server). + /// + ProtocolVersionMismatch = 13, + /// + /// A protocol error occurred that was severe enough to invalidate the current connection or connection attempt (Client/Server). + /// + ProtocolError = 14, + /// + /// The transport connection was closed due to the requested video codec not being available (Client/Server). + /// + VideoCodecNotAvailable = 15, + /// + /// The connection attempt has been canceled (Client/Server). + /// + Canceled = 16, + /// + /// The connection has been closed by peer (Client/Server). + /// + ConnectionLost = 17, + /// + /// The connection has been closed due to graphics device loss (Client/Server). + /// + DeviceLost = 18, + /// + /// The connection has been closed by request (Client/Server). + /// + DisconnectRequest = 19, + /// + /// The network is unreachable. This usually means the client knows no route to reach the remote host (Client). + /// + HandshakeNetworkUnreachable = 20, + /// + /// No connection could be made because the remote side actively refused it. Usually this means that no host application is running (Client). + /// + HandshakeConnectionRefused = 21, + /// + /// The transport connection was closed due to the requested video format not being available (Client/Server). + /// + VideoFormatNotAvailable = 22, + /// + /// Disconnected after receiving a disconnect request from the peer (Client/Server). + /// + PeerDisconnectRequest = 23, + /// + /// Timed out while waiting for peer to close connection (Client/Server). + /// + PeerDisconnectTimeout = 24, + /// + /// Timed out while waiting for transport session to be opened (Client/Server). + /// + SessionOpenTimeout = 25, + /// + /// Timed out while waiting for the remoting handshake to complete (Client/Server). + /// + RemotingHandshakeTimeout = 26, + /// + /// The connection failed due to an internal error (Client/Server). + /// + InternalError = 27, + /// + /// The handshake could not be opened due to insufficient permissions (Client). + /// + HandshakePermissionDenied = 28, + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/AppRemoting.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/AppRemoting.cs.meta new file mode 100644 index 0000000..341044a --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/AppRemoting.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 40dc7277be7d404408908051b509daf8 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModel.cs b/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModel.cs new file mode 100644 index 0000000..9ccc122 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModel.cs @@ -0,0 +1,211 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#pragma warning disable CS0618 // Suppress deprecation warnings + +using System; +using System.Runtime.InteropServices; +using System.Threading.Tasks; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Provides access to a byte stream representing a glTF model of the current controller. + /// +#if UNITY_ANDROID + [Obsolete("The Motion Controller Model feature plugin from the Mixed Reality OpenXR Plugin has been deprecated on Android. For apps using Android motion controller models, we recommend transitioning to the OpenXR plugins from Unity and Meta.", false)] +#endif + public class ControllerModel + { + private static MotionControllerFeaturePlugin Feature => OpenXRFeaturePlugin.Feature; + + /// + /// The user's left controller. + /// + public static ControllerModel Left { get; } = new ControllerModel(Handedness.Left); + + /// + /// The user's right controller. + /// + public static ControllerModel Right { get; } = new ControllerModel(Handedness.Right); + + private readonly Handedness m_handedness; + + internal ControllerModel(Handedness trackerHandedness) + { + m_handedness = trackerHandedness; + } + + /// + /// Returns true if the necessary OpenXR feature extensions are enabled on the current runtime. + /// + /// This value should not be assumed immutable and should be queried on a new XR session. + public static bool IsSupported => Feature.IsValidAndEnabled() && NativeLib.IsControllerModelSupported(); + + /// + /// Provides access to a model-specific key to either load a new model or use to cache loaded models. + /// + /// The unique key representing this controller's model, if one exists. + /// True if a valid key could be retrieved. False otherwise. + public bool TryGetControllerModelKey(out ulong modelKey) + { + if (!IsSupported || OpenXRContext.Current.Session == 0) + { + modelKey = 0; + return false; // Controller feature is not enabled. + } + + return NativeLib.TryGetControllerModelKey(m_handedness, out modelKey); + } + + /// + /// Provides a byte stream representing the glTF model of the controller, if available. + /// + /// + /// Needs to be passed into a glTF parser/loader to convert into a Unity GameObject. + /// This method allocates a byte buffer on every successful call. It's recommended to either cache it or the resulting GameObject locally instead of calling this multiple times. + /// + /// The unique key representing the desired controller's model. Can be queried using . + /// Task that triggers once the controller model stream is loaded, yielding the stream or null if there is no model available. + public Task TryGetControllerModel(ulong modelKey) + { + if (!IsSupported || OpenXRContext.Current.Session == 0) + { + return Task.FromResult(null); // Controller feature is not enabled. + } + + Task newTask = Task.Run(() => + { + if (NativeLib.TryGetControllerModel(modelKey, 0, out uint bufferCapacity)) + { + byte[] modelBuffer = new byte[bufferCapacity]; + if (NativeLib.TryGetControllerModel(modelKey, bufferCapacity, out _, modelBuffer)) + { + return modelBuffer; + } + } + return null; + }); + + return newTask; + } + + private uint m_lastNodeStateCount = 0; + + /// + /// Represents a set of animatable nodes in the controller model. Use to obtain the current animation values. + /// + /// The unique key representing the desired controller's model. Can be queried using . + /// The Transform representing the loaded controller model from . + /// A method-allocated array containing the animatable nodes in the current controller model, with the same indices as the Pose array data from . + /// + /// This method allocates a Transform array on every successful call. + /// It's recommended to cache it locally instead of calling this multiple times, as this won't change unless the model key changes. + /// + internal bool TryGetControllerModelProperties(ulong modelKey, Transform modelRoot, out Transform[] nodes) + { + if (IsSupported && OpenXRContext.Current.Session != 0 && + NativeLib.TryGetControllerModelProperties(modelKey, 0, out uint nodeCountOutput)) + { + m_lastNodeStateCount = nodeCountOutput; + ControllerModelNodeProperties[] properties = new ControllerModelNodeProperties[nodeCountOutput]; + if (NativeLib.TryGetControllerModelProperties(modelKey, nodeCountOutput, out _, properties)) + { + nodes = new Transform[nodeCountOutput]; + Transform[] children = modelRoot.GetComponentsInChildren(); + int nodesFound = 0; + // Iterates through all children of the model root in order to find the + // animatable nodes by name plus parent name (if provided). + foreach (Transform potentialNode in children) + { + // If we've found all named nodes, we can return early. + if (nodesFound == nodeCountOutput) + { + break; + } + + for (int i = 0; i < nodeCountOutput; i++) + { + // Because we iterate through all node names for each node, it's possible that this node has already been found. + if (nodes[i] != null) + { + continue; + } + + ControllerModelNodeProperties property = properties[i]; + if (potentialNode.name.Equals(property.NodeName, StringComparison.OrdinalIgnoreCase) + && (string.IsNullOrWhiteSpace(property.ParentNodeName) + || (potentialNode.parent != null && potentialNode.parent.name.Equals(property.ParentNodeName, StringComparison.OrdinalIgnoreCase)))) + { + nodes[i] = potentialNode; + nodesFound++; + break; + } + } + } + + // If we didn't find all nodes, log which ones are missing. + if (nodesFound != nodeCountOutput) + { + for (int i = 0; i < nodeCountOutput; i++) + { + if (nodes[i] == null) + { + Debug.LogError($"No corresponding node found for node name {properties[i].NodeName} and parent name {properties[i].ParentNodeName}."); + } + } + } + return nodesFound == nodeCountOutput; + } + } + + nodes = Array.Empty(); + return false; + } + + /// + /// Represents the current state of the controller model representing user's interaction to the controller, such as pressing a button or pulling a trigger. + /// + /// The unique key representing the desired controller's model. Can be queried using . + /// + /// The pose array must match the properties array size of the most recent call to . + internal bool TryGetControllerModelState(ulong modelKey, Pose[] poses) + { + if (!IsSupported || OpenXRContext.Current.Session == 0) + { + return false; + } + + if (poses.Length != m_lastNodeStateCount) + { + throw new ArgumentException("The poses array doesn't match the most recent array size from TryGetControllerModelProperties."); + } + + return NativeLib.TryGetControllerModelState(modelKey, m_lastNodeStateCount, out _, poses); + } + + /// + /// Describes properties of animatable nodes, including the node name and parent node name to locate a glTF node in the controller model that can be animated based on user's interactions on the controller. + /// + [StructLayout(LayoutKind.Sequential, Pack = 8, CharSet = CharSet.Ansi)] + internal readonly struct ControllerModelNodeProperties + { + // Represents the maximum name size defined by the OpenXR spec. Used for string marshaling. + private const int ControllerModelNodeNameSize = 64; + + /// + /// The name of the parent node in the provided glTF file. + /// + /// The parent name may be empty if it should not be used to locate this node. + [field: MarshalAs(UnmanagedType.ByValTStr, SizeConst = ControllerModelNodeNameSize)] + public string ParentNodeName { get; } + + /// + /// The name of this node in the provided glTF file. + /// + [field: MarshalAs(UnmanagedType.ByValTStr, SizeConst = ControllerModelNodeNameSize)] + public string NodeName { get; } + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModel.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModel.cs.meta new file mode 100644 index 0000000..8c6b7da --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModel.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6a764738c86a12741b0e3dab1be52f44 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModelArticulator.cs b/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModelArticulator.cs new file mode 100644 index 0000000..ef27b56 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModelArticulator.cs @@ -0,0 +1,87 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#pragma warning disable CS0618 // Suppress deprecation warnings + +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Handles articulation of the animatable parts of a controller model. + /// + public class ControllerModelArticulator : MonoBehaviour + { + private ControllerModel m_controllerModel = null; + private ulong m_modelKey = 0; + private Transform[] m_animationNodes = null; + private Pose[] m_poses = null; + + private bool IsArticulating => m_controllerModel != null && m_modelKey != 0; + + /// + /// Tries to start active articulation of this controller model. + /// + /// The reference to the controller model this component represents. See and . + /// The model key corresponding to the loaded controller model. See . + /// True if the controller model supports part articulation and articulation was actively started. + public bool TryStartArticulating(ControllerModel controllerModel, ulong modelKey) + { + if (controllerModel.TryGetControllerModelProperties(modelKey, transform, out m_animationNodes)) + { + m_controllerModel = controllerModel; + m_modelKey = modelKey; + // For updating the node poses in Update. This needs to be the same length as the number of nodes. + if (m_poses == null || m_poses.Length != m_animationNodes.Length) + { + m_poses = new Pose[m_animationNodes.Length]; + } + } + else + { + // Disable the built-in, auto-playing glTF animations in the Quest model. + Animation[] animations = GetComponentsInChildren(); + foreach (Animation animation in animations) + { + animation.enabled = false; + } + } + + return IsArticulating; + } + + /// + /// Stops any active articulation of this controller model. + /// + public void StopArticulating() + { + m_controllerModel = null; + m_modelKey = 0; + } + + /// + /// The MonoBehaviour Update() callback. + /// + protected void Update() + { + if (IsArticulating + && m_poses != null + && m_animationNodes != null + && m_controllerModel.TryGetControllerModelState(m_modelKey, m_poses)) + { + for (int i = 0; i < m_poses.Length; i++) + { + Transform node = m_animationNodes[i]; + Pose pose = m_poses[i]; + +#if UNITY_2021_3_11_OR_NEWER + node.SetLocalPositionAndRotation(pose.position, pose.rotation); +#else + node.localPosition = pose.position; + node.localRotation = pose.rotation; +#endif + } + } + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModelArticulator.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModelArticulator.cs.meta new file mode 100644 index 0000000..4ef4982 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ControllerModelArticulator.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 34f4fdd4ddfedb64e9e8ad700781dcff +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/EyeLevelSceneOrigin.cs b/com.microsoft.mixedreality.openxr/Runtime/API/EyeLevelSceneOrigin.cs new file mode 100644 index 0000000..1b3329c --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/EyeLevelSceneOrigin.cs @@ -0,0 +1,102 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEngine; +using UnityEngine.XR; +using UnityEngine.XR.Management; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Add the EyeLevelSceneOrigin component to the scene, it will automatically + /// switch the Unity's scene origin to an eye level experiences. + /// It will try to use "Unbounded" origin mode when it's supported. + /// + [System.Obsolete("Obsolete and will be removed in future releases. " + + "Use MRTK3 Microsoft.MixedReality.Toolkit.Input.UnboundedTrackingMode instead for HoloLens 2 application using Unbounded space." + + "Or use Unity.​XR.​Core​Utils.XROrigin for other XR applications.", false)] + public class EyeLevelSceneOrigin : MonoBehaviour + { + private XRInputSubsystem m_inputSubsystem; + private ulong m_currentSession = 0; + + private static XRInputSubsystem GetXRInputSubsystem() + { + XRGeneralSettings xrSettings = XRGeneralSettings.Instance; + if (xrSettings != null) + { + XRManagerSettings xrManager = xrSettings.Manager; + if (xrManager != null) + { + XRLoader xrLoader = xrManager.activeLoader; + if (xrLoader != null) + { + return xrLoader.GetLoadedSubsystem(); + } + } + } + return null; + } + + private void Update() + { + if (m_currentSession != OpenXRContext.Current.Session) + { + m_currentSession = OpenXRContext.Current.Session; + + if (m_inputSubsystem != null) + { + m_inputSubsystem.trackingOriginUpdated -= XrInputSubsystem_trackingOriginUpdated; + m_inputSubsystem = null; // reset input subsystem reference on a new OpenXR session. + } + } + + // Lazy initialize input subsystem. + if (m_inputSubsystem == null && OpenXRContext.Current.IsSessionRunning) + { + m_inputSubsystem = GetXRInputSubsystem(); + if (m_inputSubsystem != null) + { + EnsureSceneOriginAtEyeLevel(m_inputSubsystem); + m_inputSubsystem.trackingOriginUpdated += XrInputSubsystem_trackingOriginUpdated; + } + } + } + + private void XrInputSubsystem_trackingOriginUpdated(XRInputSubsystem xrInputSubsystem) + { + if (OpenXRContext.Current.IsSessionRunning && xrInputSubsystem == m_inputSubsystem) + { + EnsureSceneOriginAtEyeLevel(m_inputSubsystem); + } + } + + private static void EnsureSceneOriginAtEyeLevel(XRInputSubsystem xrInputSubsystem) + { + TrackingOriginModeFlags currentMode = xrInputSubsystem.GetTrackingOriginMode(); + TrackingOriginModeFlags desiredMode = GetDesiredTrackingOriginMode(xrInputSubsystem); + bool isEyeLevel = currentMode == TrackingOriginModeFlags.Device || currentMode == TrackingOriginModeFlags.Unbounded; + if (!isEyeLevel || currentMode != desiredMode) + { + Debug.Log($"EyeLevelSceneOrigin: TrySetTrackingOriginMode to {desiredMode}"); + if (!xrInputSubsystem.TrySetTrackingOriginMode(desiredMode)) + { + Debug.LogWarning($"EyeLevelSceneOrigin: Failed to set tracking origin to {desiredMode}."); + } + } + } + + private static TrackingOriginModeFlags GetDesiredTrackingOriginMode(XRInputSubsystem xrInputSubsystem) + { + TrackingOriginModeFlags supportedFlags = xrInputSubsystem.GetSupportedTrackingOriginModes(); + TrackingOriginModeFlags targetFlag = TrackingOriginModeFlags.Device; // All OpenXR runtime must support LOCAL space + + if (supportedFlags.HasFlag(TrackingOriginModeFlags.Unbounded)) + { + targetFlag = TrackingOriginModeFlags.Unbounded; + } + + return targetFlag; + } + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/EyeLevelSceneOrigin.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/EyeLevelSceneOrigin.cs.meta new file mode 100644 index 0000000..0392b3e --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/EyeLevelSceneOrigin.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: ce39d28887930cc46bb9a7bb6cd3e02d +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/FrameTime.cs b/com.microsoft.mixedreality.openxr/Runtime/API/FrameTime.cs new file mode 100644 index 0000000..06cad0a --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/FrameTime.cs @@ -0,0 +1,21 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Choose the predicted display time of a frame in pipelined rendering. + /// + public enum FrameTime + { + /// + /// The time in update thread using previous frame's predicted time + duration. + /// + OnUpdate = 0, + + /// + /// The time in render thread using current frame's predicted time. + /// + OnBeforeRender + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/FrameTime.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/FrameTime.cs.meta new file mode 100644 index 0000000..2cc7d13 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/FrameTime.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c4013a47da6b7964a988ac5cf3658cf6 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/GestureRecognizer.cs b/com.microsoft.mixedreality.openxr/Runtime/API/GestureRecognizer.cs new file mode 100644 index 0000000..160c634 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/GestureRecognizer.cs @@ -0,0 +1,363 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Runtime.InteropServices; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Represents the set of gestures that may be recognized by a GestureRecognizer. + /// + [Flags] + public enum GestureSettings + { + /// + /// An empty gesture setting + /// + None = 0, + + /// + /// A single air tap. + /// + Tap = 1, + + /// + /// A double air tap + /// + DoubleTap = 1 << 1, + + /// + /// Hold at the end of air tap + /// + Hold = 1 << 2, + + /// + /// Manipulate gesture to control X, Y, and Z translations. + /// + ManipulationTranslate = 1 << 3, + + /// + /// Navigation gesture at X direction. + /// + NavigationX = 1 << 4, + + /// + /// Navigation gesture at Y direction. + /// + NavigationY = 1 << 5, + + /// + /// Navigation gesture at Z direction. + /// + NavigationZ = 1 << 6, + + /// + /// Navigation gesture at X direction and suppress Y and Z direction. + /// + NavigationRailsX = 1 << 7, + + /// + /// Navigation gesture at Y direction and suppress X and Z direction. + /// + NavigationRailsY = 1 << 8, + + /// + /// Navigation gesture at Z direction and suppress X and Y direction. + /// + NavigationRailsZ = 1 << 9, + } + + /// + /// Represents the type of gesture recognizer event + /// + public enum GestureEventType + { + /// + /// Indicates a new recognition is started. + /// + RecognitionStarted, + + /// + /// Indicates the recognition is ended. + /// + RecognitionEnded, + + /// + /// Indicates a tap is detected. + /// + Tapped, + + /// + /// Indicates a hold gesture is detected. + /// + HoldStarted, + + /// + /// Indicates a hold gesture is completed. + /// + HoldCompleted, + + /// + /// Indicates a hold gesture is canceled. + /// + HoldCanceled, + + /// + /// Indicates a manipulation gesture is started. + /// + ManipulationStarted, + + /// + /// Indicates a manipulation gesture is updating the input location. + /// + ManipulationUpdated, + + /// + /// Indicates a manipulation gesture is completed. + /// + ManipulationCompleted, + + /// + /// Indicates a manipulation gesture is canceled. + /// + ManipulationCanceled, + + /// + /// Indicates a navigation gesture is started. + /// + NavigationStarted, + + /// + /// Indicates a navigation gesture is updating the input location. + /// + NavigationUpdated, + + /// + /// Indicates a navigation gesture is completed. + /// + NavigationCompleted, + + /// + /// Indicates a navigation gesture is canceled. + /// + NavigationCanceled, + } + + /// + /// Represents the hand that initiated the gesture. + /// + public enum GestureHandedness + { + /// + /// The gesture is not associated with any specific hand, for example when a gesture is triggered by voice command. + /// + Unspecified, + + /// + /// The gesture is initiated by left hand. + /// + Left, + + /// + /// The gesture is initiated by right hand. + /// + Right, + } + + /// + /// The data of a gesture event, include the event type, handedness, poses etc. + /// + [StructLayout(LayoutKind.Sequential, Pack = 8)] + public struct GestureEventData + { + /// + /// Get the type of gesture event. + /// + public GestureEventType EventType => nativeData.eventType; + + /// + /// Get which hand triggers this gesture event, or it's not related to specific hand. + /// + public GestureHandedness Handedness => nativeData.handedness; + + /// + /// Get the data for tap or double tap event. + /// It only has value if and only if the eventType == GestureEventType.Tapped + /// + public TappedEventData? TappedData => nativeData.Get(nativeData.tappedData, nativeData.IsTappedEvent()); + + /// + /// Get the data for manipulation gesture event. + /// It only has value if and only if the eventType == GestureEventType.ManipulationStarted/Updated/Completed + /// + public ManipulationEventData? ManipulationData => nativeData.Get(nativeData.manipulationData, nativeData.IsManipulationEvent()); + + /// + /// Get the data for navigation gesture event. + /// It only has value if and only if the eventType == GestureEventType.NavigationStarted/Updated/Completed + /// + public NavigationEventData? NavigationData => nativeData.Get(nativeData.navigationData, nativeData.IsNavigationEvent()); + + private readonly NativeGestureEventData nativeData; + } + + /// + /// The data of a tap gesture event + /// + [StructLayout(LayoutKind.Sequential, Pack = 8)] + public struct TappedEventData + { + /// + /// The tap number represented by this gesture, either 1 or 2. + /// + public uint TapCount; + } + + /// + /// The data of a manipulation gesture event + /// + [StructLayout(LayoutKind.Sequential, Pack = 8)] + public struct ManipulationEventData + { + /// + /// Get the relative translation of the hand since the start of a Manipulation gesture. + /// + public Vector3 CumulativeTranslation; + } + + /// + /// The data of a navigation gesture event + /// + [StructLayout(LayoutKind.Sequential, Pack = 8)] + public struct NavigationEventData + { + /// + /// Gets whether the navigation gesture the user is performing involves motion on the horizontal axis. + /// + public bool IsNavigatingX => m_directionFlags.HasFlag(NativeDirectionFlags.X); + /// + /// Gets whether the navigation gesture the user is performing involves motion on the vertical axis. + /// + public bool IsNavigatingY => m_directionFlags.HasFlag(NativeDirectionFlags.Y); + /// + /// Gets whether the navigation gesture the user is performing involves motion on the depth axis. + /// + public bool IsNavigatingZ => m_directionFlags.HasFlag(NativeDirectionFlags.Z); + + /// + /// Gets the normalized offset of the hand or motion controller within the unit cube for all axes for this Navigation gesture. + /// + /// X direction is from left to right. Y direction is from bottom to top. Z direction is from back to forward. + public Vector3 NormalizedOffset; + private NativeDirectionFlags m_directionFlags; + } + + /// + /// A gesture recognizer interprets user interactions from hands, motion controllers, and system voice commands + /// to surface spatial gesture events, which users target using their gaze or hand's pointing ray. + /// + public class GestureRecognizer + { + /// + /// Create a new GestureRecognizer using the given settings. + /// + /// If the given setting is not compatible, the new GestureRecognizer will still be created, + /// though it won't produce any gesture event. The setting can be corrected later through the GestureSettings property. + public GestureRecognizer(GestureSettings settings) + { + GestureSettings = settings; + } + + /// + /// Set the gesture settings to configure which gestures to recognize. + /// + public GestureSettings GestureSettings + { + get { return m_requestedSettings; } + set + { + if (m_requestedSettings != value) + { + m_requestedSettings = value; + if (m_gestureSubsystem != null) + { + m_gestureSubsystem.GestureSettings = value; + } + else + { + m_gestureSubsystem = GestureSubsystem.TryCreateGestureSubsystem(value); + } + } + } + } + + /// + /// Start monitor the user interactions and recognize the configured gestures. + /// + public void Start() + { + if (m_gestureSubsystem != null) + { + m_gestureSubsystem.Start(); + } + } + + /// + /// Stop monitor the user interactions + /// + public void Stop() + { + if (m_gestureSubsystem != null) + { + m_gestureSubsystem.Stop(); + } + } + + /// + /// Get the next gesture recognition event data, or return false when the event queue is empty. + /// + /// App allocated data struct to receive the data. + /// If function returns false, the content in eventData is undefined and should be avoided. + public bool TryGetNextEvent(ref GestureEventData eventData) + { + return m_gestureSubsystem != null && m_gestureSubsystem.TryGetNextEvent(ref eventData); + } + + /// + /// Cancel all pending gestures and reset to initial state. All events in the queue will be discarded. + /// + public void CancelPendingGestures() + { + if (m_gestureSubsystem != null) + { + m_gestureSubsystem.CancelPendingGestures(); + } + } + + /// + /// Destroy the GestureRecognizer when the application is done with it. + /// + public void Destroy() + { + if (m_gestureSubsystem != null) + { + m_gestureSubsystem.Dispose(); + m_gestureSubsystem = null; + } + } + + /// + /// Destroy the GestureRecognizer when the application is done with it. + /// + [Obsolete("Obsolete and will be removed in future releases. Use the Destroy() function at appropriated place instead.")] + public void Dispose() + { + Destroy(); + } + + private GestureSubsystem m_gestureSubsystem; + private GestureSettings m_requestedSettings; + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/GestureRecognizer.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/GestureRecognizer.cs.meta new file mode 100644 index 0000000..68e4560 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/GestureRecognizer.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c22e3d26edc7f2642ae3fb57db022f22 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/HandMeshTracker.cs b/com.microsoft.mixedreality.openxr/Runtime/API/HandMeshTracker.cs new file mode 100644 index 0000000..de2c515 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/HandMeshTracker.cs @@ -0,0 +1,138 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Represents different possible hand poses. + /// + /// See https://www.khronos.org/registry/OpenXR/specs/1.0/html/xrspec.html#XrHandPoseTypeMSFT for more information. + public enum HandPoseType + { + /// + /// Represents a hand pose provided by actual tracking of the user's hand. + /// + Tracked = 0, + + /// + /// Represents a stable reference hand pose in a relaxed open hand shape. + /// + ReferenceOpenPalm, + } + + /// + /// Represents a user's hand and the ability to render a hand mesh representation of it. + /// + public class HandMeshTracker + { + /// + /// The user's left hand. + /// + public static HandMeshTracker Left { get; } = new HandMeshTracker(Handedness.Left); + + /// + /// The user's right hand. + /// + public static HandMeshTracker Right { get; } = new HandMeshTracker(Handedness.Right); + + private HandTrackingFeaturePlugin Feature => OpenXRFeaturePlugin.Feature; + private readonly Handedness m_handedness; + + private Vector3[] m_handMeshVertices = null; + private Vector3[] m_handMeshNormals = null; + private int[] m_handMeshIndices = null; + + private Mesh m_currentMesh = null; + private uint m_indexBufferKey = 0; + private ulong m_vertexBufferkey = 0; + + + private HandMeshTracker(Handedness trackerHandedness) + { + m_handedness = trackerHandedness; + } + + /// + /// Tries to get the current location in world-space of the specified hand mesh. + /// + /// Specify the to locate the hand mesh. + /// The current pose of the specified hand mesh. + /// The type of hand mesh pose to request. The tracked pose represents the actively tracked hand. The reference pose represents a stable hand pose in a relaxed open hand shape. + /// Returns true when the returned pose is tracking and valid to be used. + /// Returns false when the hand mesh tracker lost tracking or it's not properly set up. + public bool TryLocateHandMesh(FrameTime frameTime, out Pose pose, HandPoseType handPoseType = HandPoseType.Tracked) + { + pose = Pose.identity; + return Feature.IsValidAndEnabled() && OpenXRContext.Current.IsSessionRunning + && NativeLib.TryLocateHandMesh(m_handedness, frameTime, handPoseType, out pose); + } + + /// + /// Retrieves the latest hand mesh information and build the current hand mesh in the passed-in mesh parameter. + /// + /// Specify the to locate the hand mesh. + /// The mesh object to build the hand mesh in. + /// The type of hand mesh to request. The tracked pose represents the actively tracked hand. The reference pose represents a stable hand pose in a relaxed open hand shape. + /// True if the mesh was retrievable. + public bool TryGetHandMesh(FrameTime frameTime, Mesh handMesh, HandPoseType handPoseType = HandPoseType.Tracked) + { + if (!Feature.IsValidAndEnabled() || !OpenXRContext.Current.IsSessionRunning) + { + return false; // Hand tracking feature is not enabled. Return the tracker not tracking. + } + + try + { + if (m_handMeshVertices == null || m_handMeshNormals == null || m_handMeshIndices == null) + { + if (NativeLib.TryGetHandMeshBufferSizes(out uint maxVertexCount, out uint maxIndexCount)) + { + m_handMeshVertices = new Vector3[maxVertexCount]; + m_handMeshNormals = new Vector3[maxVertexCount]; + m_handMeshIndices = new int[maxIndexCount]; + } + else + { + return false; + } + } + + if (m_currentMesh != handMesh) + { + m_currentMesh = handMesh; + m_indexBufferKey = 0; + m_vertexBufferkey = 0; + } + + if (NativeLib.TryGetHandMesh(m_handedness, frameTime, handPoseType, + ref m_vertexBufferkey, out uint vertexCount, m_handMeshVertices, m_handMeshNormals, + ref m_indexBufferKey, out uint indexCount, m_handMeshIndices)) + { + // The NativeLib call will return a count of 0 if no change was made + if (vertexCount > 0) + { + handMesh.SetVertices(m_handMeshVertices, 0, (int)vertexCount); + handMesh.SetNormals(m_handMeshNormals, 0, (int)vertexCount); + } + + if (indexCount > 0) + { + handMesh.SetTriangles(m_handMeshIndices, 0, (int)indexCount, 0); + } + + return true; + } + else + { + return false; + } + } + catch (System.DllNotFoundException) + { + return false; + } + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/HandMeshTracker.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/HandMeshTracker.cs.meta new file mode 100644 index 0000000..0f497b5 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/HandMeshTracker.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 075bd1956efb2194da10f92a17931bf0 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/HandTracker.cs b/com.microsoft.mixedreality.openxr/Runtime/API/HandTracker.cs new file mode 100644 index 0000000..83325d4 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/HandTracker.cs @@ -0,0 +1,267 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#pragma warning disable CS0618 // Suppress deprecation warnings + +using System; +using System.Runtime.InteropServices; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Represents a user's hand and the ability to track hand joints from it. + /// +#if UNITY_ANDROID + [Obsolete("Hand tracking on Android with the Mixed Reality OpenXR Plugin has been deprecated. " + + "For apps using Android hand tracking, we recommend transitioning to the OpenXR plugins from Unity and Meta.", false)] +#endif + public class HandTracker + { + /// + /// The user's left hand. + /// + public static HandTracker Left { get; } = new HandTracker(Handedness.Left); + + /// + /// The user's right hand. + /// + public static HandTracker Right { get; } = new HandTracker(Handedness.Right); + + private HandTrackingFeaturePlugin Feature = OpenXRFeaturePlugin.Feature; + private readonly Handedness m_handedness; + + internal HandTracker(Handedness trackerHandedness) + { + m_handedness = trackerHandedness; + } + + /// + /// The maximum number of hand joints that might be tracked. + /// + public const int JointCount = 26; + + /// + /// Fills the passed-in array with current hand joint locations, if possible. + /// + /// Specify the to locate the hand joints. + /// An array of HandJointLocations, indexed according to the HandJoint enum. + /// Returns true when the hand tracker is actively tracking the hands. + /// Returns false when the hand tracker is disabled or it's not properly set up. + /// + /// The return value matches the XrHandTrackingDataSourceStateEXT::isActive value in XR_EXT_hand_tracking_data_source extension. + /// It returns true if the extension is not supported by OpenXR runtime because Unity cannot observe the hand tracker active state. + /// + + public bool TryLocateHandJoints(FrameTime frameTime, HandJointLocation[] handJointLocations) + { + if (handJointLocations.Length != JointCount) + { + Debug.LogError($"LocateJoints requires an array of size {JointCount}. You can use HandTracker.JointCount for this."); + return false; + } + + return Feature.IsValidAndEnabled() && NativeLib.TryGetHandJointData(m_handedness, frameTime, handJointLocations); + } + + /// + /// Get or set the motion range for this hand tracker. + /// + /// + /// Setting the motion range will take effect immediately for subsequent function calls. + /// However, for Unity input system updates for hand joints, it will not take effect until next frame. + /// + /// If is used with an actual hand tracker, + /// joints will still be returned. + /// It's only valid when a runtime supports hand joints when using a physical controller. + /// + public HandJointsMotionRange MotionRange + { + get + { + return Feature.IsValidAndEnabled() && OpenXRContext.Current.IsSessionRunning + ? NativeLib.GetHandJointsMotionRange(m_handedness) + : HandJointsMotionRange.Unobstructed; + } + set + { + if (Feature.IsValidAndEnabled()) + { + NativeLib.SetHandJointsMotionRange(m_handedness, value); + } + } + } + } + + /// + /// Represents locational data for a hand joint. + /// + [StructLayout(LayoutKind.Sequential, Pack = 8)] + public readonly struct HandJointLocation + { + /// + /// Whether the corresponding hand joint is actively tracked. + /// + /// If not actively tracked, the pose may be inferred or last-known but otherwise still meaningful. + public bool IsTracked => Convert.ToBoolean(isTracked); + // bool isn't blittable, so marshal a byte across the P/Invoke layer instead + private readonly byte isTracked; + + /// + /// The world-space pose of the corresponding hand joint. + /// + public Pose Pose { get; } + + /// + /// The radius of the corresponding joint in units of meters. + /// + public float Radius { get; } + } + + /// + /// Describes which hand the current hand tracker represents. + /// + public enum Handedness + { + /// + /// Represents the user's left hand. + /// + Left = 0, + + /// + /// Represents the user's right hand. + /// + Right + } + + /// + /// The supported tracked hand joints in OpenXR. + /// + /// See https://registry.khronos.org/OpenXR/specs/1.0/html/xrspec.html#XrHandJointEXT for more information. + public enum HandJoint + { + /// + /// The palm. + /// + Palm, + /// + /// The wrist. + /// + Wrist, + /// + /// The lowest joint of the thumb. + /// + ThumbMetacarpal, + /// + /// The second joint of the thumb. + /// + ThumbProximal, + /// + /// The joint nearest the tip of the thumb. + /// + ThumbDistal, + /// + /// The tip of the thumb. + /// + ThumbTip, + /// + /// The lowest joint of the index finger. + /// + IndexMetacarpal, + /// + /// The knuckle joint of the index finger. + /// + IndexProximal, + /// + /// The middle joint of the index finger. + /// + IndexIntermediate, + /// + /// The joint nearest the tip of the index finger. + /// + IndexDistal, + /// + /// The tip of the index finger. + /// + IndexTip, + /// + /// The lowest joint of the middle finger. + /// + MiddleMetacarpal, + /// + /// The proximal joint of the middle finger. + /// + MiddleProximal, + /// + /// The middle joint of the middle finger. + /// + MiddleIntermediate, + /// + /// The joint nearest the tip of the middle finger. + /// + MiddleDistal, + /// + /// The tip of the middle finger. + /// + MiddleTip, + /// + /// The lowest joint of the ring finger. + /// + RingMetacarpal, + /// + /// The knuckle of the ring finger. + /// + RingProximal, + /// + /// The middle joint of the ring finger. + /// + RingIntermediate, + /// + /// The joint nearest the tip of the ring finger. + /// + RingDistal, + /// + /// The tip of the ring finger. + /// + RingTip, + /// + /// The lowest joint of the little finger. + /// + LittleMetacarpal, + /// + /// The knuckle joint of the little finger. + /// + LittleProximal, + /// + /// The middle joint of the little finger. + /// + LittleIntermediate, + /// + /// The joint nearest the tip of the little finger. + /// + LittleDistal, + /// + /// The tip of the little finger. + /// + LittleTip, + } + + /// + /// The requested hand joints' range of motion from a controller. + /// + /// See https://registry.khronos.org/OpenXR/specs/1.0/html/xrspec.html#XrHandJointsMotionRangeEXT for more information. + public enum HandJointsMotionRange + { + /// + /// The range of motion of a human hand, without any obstructions. + /// + Unobstructed = 1, + /// + /// The range of motion of the hand joints taking into account any physical limits imposed by the controller itself. + /// + /// + /// This will tend to be the most accurate pose compared to the user’s actual hand pose, but might not allow a closed fist for example. + /// + ConformingToController = 2, + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/HandTracker.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/HandTracker.cs.meta new file mode 100644 index 0000000..5eb0244 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/HandTracker.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 1c8311f14c6fff3499be26306dc09c81 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/MeshSettings.cs b/com.microsoft.mixedreality.openxr/Runtime/API/MeshSettings.cs new file mode 100644 index 0000000..96a8eaa --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/MeshSettings.cs @@ -0,0 +1,151 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Runtime.InteropServices; +using UnityEngine.XR; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// The type of mesh to request from the XRMeshSubsystem. + /// + public enum MeshType + { + /// + /// A mesh intended for visualization. + /// + Visual = 1, + + /// + /// A mesh intended for collision detection. + /// + Collider = 2, + } + + /// + /// The level of detail of visual mesh to request from the XRMeshSubsystem. + /// + /// Has no effect on the collider mesh. + public enum VisualMeshLevelOfDetail + { + /// + /// Coarse mesh level of detail with roughly 100 triangles per cubic meter. + /// + Coarse = 1, + /// + /// Medium mesh level of detail with roughly 400 triangles per cubic meter. + /// + Medium = 2, + /// + /// Fine mesh level of detail with roughly 2000 triangles per cubic meter. + /// + Fine = 3, + /// + /// Unlimited mesh level of detail with no guarantee as to the number of triangles per cubic meter. + /// + Unlimited = 4, + } + + /// + /// The compute consistency to request from the XRMeshSubsystem. + /// + public enum MeshComputeConsistency + { + /// + /// A watertight, globally consistent snapshot, not limited to observable objects in + /// the scanned regions. + /// + ConsistentSnapshotComplete = 1, + /// + /// A non-watertight snapshot, limited to observable objects in the scanned regions. + /// The returned mesh may not be globally optimized for completeness, and therefore + /// may be returned faster in some scenarios. + /// + ConsistentSnapshotIncompleteFast = 2, + /// + /// A mesh optimized for lower-latency occlusion uses. The returned mesh may not be + /// globally consistent and might be adjusted piecewise independently. + /// + OcclusionOptimized = 3, + } + + /// + /// Settings describing the quality and type of mesh to be provided. + /// + [StructLayout(LayoutKind.Sequential, Pack = 8)] + public struct MeshComputeSettings + { + private MeshType meshType; + private VisualMeshLevelOfDetail visualMeshLevelOfDetail; + private MeshComputeConsistency meshComputeConsistency; + + /// + /// Deprecated. The XRMeshSubsystem only supplies visual meshes. Use a or the method to get collider meshes as needed. + /// + /// Defaults to . + [Obsolete("Obsolete; only visual meshes are supplied through the mesh subsystem. Use a MeshCollider or the method XRMeshSubsystem.GenerateMeshAsync to get collider meshes as needed.")] + public MeshType MeshType + { + get => meshType != 0 ? meshType : MeshType.Visual; + set => meshType = MeshType.Visual; + } + + /// + /// Get or set the level of detail of visual mesh to request from the XRMeshSubsystem. + /// + /// Defaults to . + public VisualMeshLevelOfDetail VisualMeshLevelOfDetail + { + get => visualMeshLevelOfDetail != 0 ? visualMeshLevelOfDetail : VisualMeshLevelOfDetail.Coarse; + set => visualMeshLevelOfDetail = value; + } + + /// + /// Get or set the compute consistency to request from the XRMeshSubsystem. + /// + /// Defaults to . + public MeshComputeConsistency MeshComputeConsistency + { + get => meshComputeConsistency != 0 ? meshComputeConsistency : MeshComputeConsistency.OcclusionOptimized; + set => meshComputeConsistency = value; + } + } + + namespace ARSubsystems + { + /// + /// Additional functionality for the mesh subsystem. + /// + public static class MeshSubsystemExtensions + { + /// + /// Change the settings for future meshes given by the XRMeshSubsystem. + /// + /// The to receive the settings + /// The mesh compute settings to be set. + /// Returns true if the setting is successfully changed to the given value. Returns false otherwise. + [System.Obsolete("Obsolete and will be removed in future releases. Use MeshSettings.TrySetMeshComputeSettings() function instead.", true)] + public static bool TrySetMeshComputeSettings(this XRMeshSubsystem subsystem, MeshComputeSettings settings) + { + return InternalMeshSettings.TrySetMeshComputeSettings(settings); + } + } + } + + /// + /// Static entry point for updating the mesh compute settings. + /// + public static class MeshSettings + { + /// + /// Change the settings for future meshes given by the XRMeshSubsystem. + /// + /// The mesh compute settings to be set. + /// Returns true if the setting is successfully changed to the given value. Returns false otherwise. + public static bool TrySetMeshComputeSettings(MeshComputeSettings settings) + { + return InternalMeshSettings.TrySetMeshComputeSettings(settings); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/MeshSettings.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/MeshSettings.cs.meta new file mode 100644 index 0000000..5a5f9ee --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/MeshSettings.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 24c4134a5519f5b498e067fe83af882c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRContext.cs b/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRContext.cs new file mode 100644 index 0000000..b5dacaf --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRContext.cs @@ -0,0 +1,52 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Retrieve the current OpenXR instance, session handles and states. + /// + public class OpenXRContext + { + /// + /// Get the current OpenXR context. + /// + public static OpenXRContext Current { get; } = new OpenXRContext(); + + /// + /// The XrInstance handle, or 0 when instance is not initialized. + /// + public ulong Instance => OpenXRFeaturePluginManager.ActiveFeature != null ? OpenXRFeaturePluginManager.ActiveFeature.Instance : 0; + + /// + /// The XrSystemId, or 0 when system is not available. + /// + public ulong SystemId => OpenXRFeaturePluginManager.ActiveFeature != null ? OpenXRFeaturePluginManager.ActiveFeature.SystemId : 0; + + /// + /// The XrSession handle, or 0 when session is not created. + /// + public ulong Session => OpenXRFeaturePluginManager.ActiveFeature != null ? OpenXRFeaturePluginManager.ActiveFeature.Session : 0; + + /// + /// Whether the current XrSession is running, i.e. when the frame loop is in progress. + /// + public bool IsSessionRunning => OpenXRFeaturePluginManager.ActiveFeature != null && OpenXRFeaturePluginManager.ActiveFeature.IsSessionRunning; + + /// + /// An XrSpace handle to the reference space of the current Unity scene origin, or 0 when not available. + /// It's typically a LOCAL, a STAGE or an UNBOUNDED reference space handle when available. + /// + public ulong SceneOriginSpace => OpenXRFeaturePluginManager.ActiveFeature != null ? OpenXRFeaturePluginManager.ActiveFeature.SceneOriginSpace : 0; + + /// + /// Get the function pointer to PFN_xrGetInstanceProcAddr that includes Unity OpenXR plugin and features overrides. + /// Returns 0 when XR is not loaded in Unity or xrInstance handle above is 0. + /// + public IntPtr PFN_xrGetInstanceProcAddr => OpenXRFeaturePlugin.PFN_xrGetInstanceProcAddr; + + private OpenXRContext() { } // Should use static singleton `Current` property instead. + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRContext.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRContext.cs.meta new file mode 100644 index 0000000..7dd7b48 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRContext.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: eda0de53724343840843c40c809b8f33 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRTime.cs b/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRTime.cs new file mode 100644 index 0000000..58b8e3f --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRTime.cs @@ -0,0 +1,48 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Provides methods to interact with OpenXR time, including retrieving the + /// predicting display times, and converting XR time to Query Performance Counter (QPC) time. + /// + public class OpenXRTime + { + /// + /// Get the current OpenXRTime. + /// + public static OpenXRTime Current => m_current; + + /// + /// Retrieves the predicted display time for the current frame based on the specified frame timing. + /// + /// + /// Will return 0 if called from Unity Editor. + /// + /// The time of a frame in pipelined rendering for which to retrieve the predicted display time. + /// The predicted display time if available, otherwise 0 + public long GetPredictedDisplayTimeInXrTime(FrameTime frameTime) + { + return NativeLib.GetPredictedDisplayTimeInXrTime(frameTime); + } + + /// + /// Converts a time value from XR time to Query Performance Counter (QPC) time. + /// + /// + /// Will return 0 if called from Unity Editor. + /// + /// The time in XR time units to be converted. + /// The equivalent time in QPC units. If the conversion cannot be performed the function returns 0. + public long ConvertXrTimeToQpcTime(long xrTime) + { + return NativeLib.ConvertXrTimeToQpcTime(xrTime); + } + + private OpenXRTime() { }// use static Current property instead. + private static readonly OpenXRTime m_current = new(); + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRTime.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRTime.cs.meta new file mode 100644 index 0000000..07756f1 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/OpenXRTime.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: ad3b3fd233c291e4f939d41c0e9d2857 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/PerceptionInterop.cs b/com.microsoft.mixedreality.openxr/Runtime/API/PerceptionInterop.cs new file mode 100644 index 0000000..19889be --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/PerceptionInterop.cs @@ -0,0 +1,43 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Runtime.InteropServices; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Interop functions for Windows Perception APIs + /// + public static class PerceptionInterop + { + private static MixedRealityFeaturePlugin Feature => OpenXRFeaturePlugin.Feature; + + /// + /// Get a COM wrapper object of a Windows.Perception.Spatial.SpatialCoordinateSystem object + /// located at the given pose in the current Unity scene. + /// If failed, the function returns nullptr. + /// The application should acquire a new one when session origin is changed or tracking mode is changed + /// by listening to XRInputSubsystem.trackingOriginUpdated and monitoring ARSession.currentTrackingMode. + /// + /// The pose of returned coordinate system in the current Unity scene. + /// If input Pose.identity, the returned coordinate system will be at the origin of the current Unity scene. + /// Returns a COM wrapper C# object of type Windows.Perception.Spatial.SpatialCoordinateSystem. + /// Returns null if such coordinate system cannot be found at the moment. + public static object GetSceneCoordinateSystem(Pose poseInScene) + { + if (Feature.IsValidAndEnabled() && OpenXRContext.Current.IsSessionRunning) + { + IntPtr unknown = NativeLib.TryAcquireSceneCoordinateSystem(poseInScene); + if (unknown != IntPtr.Zero) + { + object result = Marshal.GetObjectForIUnknown(unknown); + Marshal.Release(unknown); // Balance the ref count because "feature.TryAcquire" increment it on return. + return result; + } + } + return null; + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/PerceptionInterop.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/PerceptionInterop.cs.meta new file mode 100644 index 0000000..bdf48f1 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/PerceptionInterop.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6d0c25da243e4cf47bfd6fb38bfb4a90 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/SelectKeywordRecognizer.cs b/com.microsoft.mixedreality.openxr/Runtime/API/SelectKeywordRecognizer.cs new file mode 100644 index 0000000..e2b967e --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/SelectKeywordRecognizer.cs @@ -0,0 +1,79 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR || UNITY_STANDALONE || WINDOWS_UWP +using System; +using UnityEngine.Windows.Speech; +using static UnityEngine.Windows.Speech.PhraseRecognizer; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// A keyword recognizer listening to the "select" keyword localized in the system display language of HoloLens 2. + /// + /// + /// This class is only required by HoloLens 2 as the OS filters out the "select" keyword and thus + /// the Unity does not fire events when the word is heard. + /// The API surface is made mostly identical to the Unity for ease of use. + /// Like the Unity , this class is only available under the UNITY_EDITOR || UNITY_STANDALONE || WINDOWS_UWP flags. + /// We recommend checking for those flags in the code using #if before referencing this class, especially when developing a cross-platform application. + /// + public sealed class SelectKeywordRecognizer : IDisposable + { + + /// + /// Create a new SelectKeywordRecognizer. + /// + /// + /// Use to check whether the recognizer is supported by the current platform / Unity version first before calling the constructor. + /// The constructor does the same check and will throw an exception if not supported. + /// + public SelectKeywordRecognizer() + { + m_provider = new SelectKeywordRecognizerProvider(); + } + + /// + /// Check whether the recognizer is supported by the current platform / Unity version + /// + public static bool IsSupported => SelectKeywordRecognizerProvider.IsSupported; + + /// + /// Return whether the recognizer is running + /// + public bool IsRunning => m_provider.IsRunning; + + /// + /// Event to be fired when the "select" keyword is recognized + /// + public event PhraseRecognizedDelegate OnPhraseRecognized + { + add + { + m_provider.OnPhraseRecognized += value; + } + remove + { + m_provider.OnPhraseRecognized -= value; + } + } + + /// + /// Start the SelectKeywordRecognizer to listen for the select keyword + /// + public void Start() => m_provider.Start(); + + /// + /// Stop the SelectKeywordRecognizer from listening for the select keyword + /// + public void Stop() => m_provider.Stop(); + + /// + /// Dispose the resources used by SelectKeywordRecognizer + /// + public void Dispose() => m_provider.Dispose(); + + private SelectKeywordRecognizerProvider m_provider; + } +} +#endif // UNITY_EDITOR || UNITY_STANDALONE || WINDOWS_UWP \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/SelectKeywordRecognizer.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/SelectKeywordRecognizer.cs.meta new file mode 100644 index 0000000..49d4bab --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/SelectKeywordRecognizer.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 2b16eb3983fc26a4f9e9b20bce4fbe19 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/SpatialGraphNode.cs b/com.microsoft.mixedreality.openxr/Runtime/API/SpatialGraphNode.cs new file mode 100644 index 0000000..3b0d437 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/SpatialGraphNode.cs @@ -0,0 +1,132 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// A spatial graph node represents a spatially tracked point provided by the driver. + /// + /// + /// There are two types of spatial graph nodes: static and dynamic. + /// + /// A static spatial graph node tracks the pose of a fixed location in the world. + /// The tracking of static nodes may slowly adjust the pose over time for better accuracy + /// but the pose is relatively stable in the short term, such as between rendering frames. + /// + /// A dynamic spatial graph node tracks the pose of a physical object that moves + /// continuously relative to reference spaces. The pose of a dynamic spatial graph node + /// can be very different within the duration of a rendering frame. + /// + public class SpatialGraphNode + { + /// + /// Creating a SpatialGraphNode with given static node id, or return null upon failure. + /// + /// + /// The application typically obtains the Guid for the static node + /// from other spatial graph driver APIs. For example, a static node id + /// representing the tracking of a QR code can be obtained from HoloLens 2 QR code library. + /// + /// A GUID represents a spatial graph static node. + /// Returns either a valid SpatialGraphNode object if succeeded + /// or null if the given static node id cannot be found at the moment. + public static SpatialGraphNode FromStaticNodeId(System.Guid id) + { + if (OpenXRContext.Current.IsSessionRunning && + NativeLib.TryCreateSpaceFromStaticNodeId(id, out ulong spaceId)) + { + return new SpatialGraphNode() + { + Id = id, + m_spaceId = spaceId, + }; + } + else + { + return null; + } + } + + /// + /// Creating a SpatialGraphNode with given dynamic node id, or return null upon failure. + /// + /// + /// The application typically obtains the Guid for the dynamic node + /// from other spatial graph driver APIs. For example, a dynamic node id + /// representing the tracking of the Photo and Video camera on HoloLens 2 + /// can be obtained from media foundation APIs for the camera. + /// + /// A GUID represents a spatial graph dynamic node. + /// Returns either a valid SpatialGraphNode object if succeeded + /// or null if the given dynamic node id cannot be found at the moment. + public static SpatialGraphNode FromDynamicNodeId(System.Guid id) + { + if (OpenXRContext.Current.IsSessionRunning && + NativeLib.TryCreateSpaceFromDynamicNodeId(id, out ulong spaceId)) + { + return new SpatialGraphNode() + { + Id = id, + m_spaceId = spaceId, + }; + } + else + { + return null; + } + } + + /// + /// Get the Guid of the SpatialGraphNode + /// + public System.Guid Id { get; private set; } = System.Guid.Empty; + + /// + /// Locate the SpatialGraphNode at the given frame time. + /// The returned pose is in the current Unity scene origin space. + /// + /// + /// Return true if the output pose is actively tracked, or return false indicating the node lost tracking. + /// + /// + /// This function is typically used to locate the spatial graph node used in Unity's render pipeline + /// at either OnUpdate or OnBeforeRender callbacks. Providing the correct input frameTime + /// allows the runtime to provide correct motion prediction of the tracked node to the display time + /// of the current rendering frame. + /// + /// Specify the to locate the spatial graph node. + /// Output the pose when the function returns true. Discard the value if the function returns false. + public bool TryLocate(FrameTime frameTime, out Pose pose) + { + pose = Pose.identity; + return OpenXRContext.Current.IsSessionRunning && NativeLib.TryLocateSpatialGraphNodeSpace(m_spaceId, frameTime, out pose); + } + + /// + /// Locate the SpatialGraphNode at the given QPC time. + /// The returned pose is in the current Unity scene origin space. + /// + /// + /// Return true if the output pose is actively tracked, or return false indicating the node lost tracking. + /// + /// + /// This function is typically used to locate the spatial graph node using historical timestamp + /// obtained from other spatial graph APIs, for example the qpcTime of a IMFSample from media + /// foundation APIs representing the time when a Photo and Video camera captured the image. + /// Providing an accurate qpcTime from the camera sensor allows the runtime to locate precisely + /// where the dynamic node was tracked when the image was taken. + /// + /// Specify the QPC (i.e. query performance counter) time to locate the spatial graph node. + /// Output the pose when the function returns true. Discard the value if the function returns false. + public bool TryLocate(long qpcTime, out Pose pose) + { + pose = Pose.identity; + return OpenXRContext.Current.IsSessionRunning && NativeLib.TryLocateSpatialGraphNodeSpace(m_spaceId, qpcTime, out pose); + } + + private SpatialGraphNode() { } + private ulong m_spaceId = 0; + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/SpatialGraphNode.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/SpatialGraphNode.cs.meta new file mode 100644 index 0000000..4f73fa4 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/SpatialGraphNode.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4ed36762912e54144bcb9122243e7cce +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/TrackingMapAPI.cs b/com.microsoft.mixedreality.openxr/Runtime/API/TrackingMapAPI.cs new file mode 100644 index 0000000..998c58c --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/TrackingMapAPI.cs @@ -0,0 +1,172 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. +using System; +using System.Collections.Generic; +using System.Threading.Tasks; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Type of tracking maps + /// + public enum TrackingMapType : uint + { + /// + /// Default tracking map on HoloLens 2, shared between all applications. + /// + Shared = 0, + /// + /// New tracking environment private to application, which can be persisted + /// and restored with some limitations. + /// + ApplicationExclusive = 0x00000001 + } + + /// + /// TrackingMapManager class allows an application to opt into running in an Application-Exclusive tracking mode instead of default shared environment. + /// + /// + /// + /// Activating an Application-Exclusive tracking map creates a brand-new environment for the application, unencumbered by any Device Space + /// tracking inaccuracies as the result of degradation over time. + /// This is equivalent to using the "Remove All Holograms" command from Settings, but only applicable to the running application. + /// Holograms for all other applications (including the HoloLens Shell) remain intact and available as before. + /// Returning to the Shell or activating another application will return the HoloLens to the Device Shared tracking mode automatically. + /// + /// + /// When first entering the Application-Exclusive tracking mode, the calling application will be issued a unique identifier + /// that can be used to resume tracking the app-specific map in future sessions of the application + /// (like if the user switches away from the application and it is terminated in the background due to system resource constraints). + /// However, if the device simply goes to sleep or the user briefly interacts with the Shell, + /// the application will automatically resume in the Application-Exclusive tracking mode once it is reactivated (and all application state will remain available). + /// + /// + /// There are two limitations to be aware of when using the Application-Exclusive tracking mode: + /// + /// + /// + /// Only a single Application-Exclusive tracking map can exist on the HoloLens at one time. If an application requests a new Application-Exclusive tracking mode, + /// then any previous Application-Exclusive tracking data would be erased and all SpatialAnchor objects (and attached holograms) would be lost, + /// even if the data was created by a completely different application using its own Application-Exclusive tracking mode. + /// Therefore, attempting to return to a previous Application-Exclusive map (by specifying the identifier received when this map was created) + /// may result in a return value indicating that the previous map was not found. + /// Applications must be prepared to handle the scenario where a previous Application-Exclusive tracking map is not available. + /// + /// + /// The disk storage available to the Application-Exclusive tracking mode is limited to one third of what is available for the Device Shared tracking mode, + /// although this is unlikely to be an issue for most users. When this limit is reached, HoloLens will begin erasing its least valuable tracking data, + /// which will eventually result in poorer tracking accuracy. + /// The smaller limit is still large enough to maintain good accuracy for house-sized environments and is unlikely to be a concern for most application scenarios. + /// + /// + /// Given these limitations, the target scenario for the Application-Exclusive tracking mode is for applications with high accuracy requirements that are task oriented, + /// where a task may be interrupted by the user returning the HoloLens Shell or the device going to sleep. + /// However, once the user's task is complete, nothing about the task (with respect to the 3D environment) needs to be saved and so can be erased. + /// + /// Examples: + /// + /// + /// High-accuracy alignment of holograms to a real-world object, using QR codes to bootstrap the scenario. + /// + /// + /// Editing a 3D model with high-accuracy requirements when no 3D spatial persistence of the model needs to occur after the session ends. + /// + /// + /// Tracking in places that have a lot of environmental churn (like people moving around), which sometimes results in poorer tracking quality than more static environments. + /// + /// + /// + public class TrackingMapManager + { + private TrackingMapSubsystem m_trackingMapSubsystem; + private TrackingMapManager(TrackingMapSubsystem trackingMapSubsystem) { + m_trackingMapSubsystem = trackingMapSubsystem; + } + + /// + /// Gets an instance of TrackingMapManager. + /// + /// Task<TrackingMapManager> which is completed when the Tracking Map Manager is fully initialized. + /// The Result property of the task is a TrackingMapManager instance. + /// Use IsSupported to check if the HoloLens 2 + /// on which the application is currently executing supports Application-Exclusive maps. + public static Task GetAsync() + { + Task result = Task.Run(() => + { + TrackingMapSubsystem trackingMapSubsystem = TrackingMapSubsystem.TryCreateTrackingMapSubsystem(); + return new TrackingMapManager(trackingMapSubsystem); + }); + return result; + } + + /// + /// Indicates if a tracking map type is supported + /// + /// Tracking map type to check for support + /// True if the tracking map type is supported. + /// + /// TrackingMapType.Shared is always supported. + /// + public bool IsSupported(TrackingMapType trackingMapType) + { + switch (trackingMapType) + { + case TrackingMapType.Shared: + return true; + case TrackingMapType.ApplicationExclusive: + return (m_trackingMapSubsystem != null) && m_trackingMapSubsystem.SupportsApplicationExclusiveMaps(); + default: + return false; + } + } + + /// + /// Indicates the active tracking map type. + /// + public TrackingMapType ActiveTrackingMapType + { + get => (m_trackingMapSubsystem != null) ? m_trackingMapSubsystem.ActiveTrackingMapType : TrackingMapType.Shared; + } + + /// + /// Creates and activates a new application-exclusive map. + /// + /// Optional identifier to try reactivating a previously created map. If null is passed, a new map is created. + /// If a non empty guid map is passed, the tracking manager tries to load the corresponding map. + /// If the map cannot be found or loaded, a new map is created. + /// Task<Guid> which is completed when the system has created and activated an application-exclusive map. + /// The Result property contains the guid of the activated map. If it is equal to existingMapId, it means that the existing map has been reloaded. + /// If not, it means that a new map has been created. + /// Thrown when the TrackingMapManager does not support application-exclusive Maps + /// or if a map change is already in progress. + /// In order to take effect, the application must be the active, immersive 3D application. + /// ActivateApplicationExclusiveMapAsync should only be called once the application has started rendering its 3D user interface (and not, for example, when the application first starts). + /// + public Task ActivateApplicationExclusiveMapAsync(Guid? existingMapId = null) + { + if (m_trackingMapSubsystem == null) throw new InvalidOperationException("Feature not supported."); + Task result = Task.Run(() => + { + return m_trackingMapSubsystem.ActivateApplicationExclusiveMap(existingMapId); + }); + return result; + } + + /// + /// Leaves the current application-exclusive map and returns to the default shared map. + /// If the device was already in the default shared map, this method does nothing. + /// + /// Task which is completed when the system has returned to the default map. + /// Thrown when a map change is already in progress. + public Task ActivateSharedMapAsync() + { + if (m_trackingMapSubsystem == null) throw new InvalidOperationException("Feature not supported."); + Task result = Task.Run(() => + { + m_trackingMapSubsystem.ActivateSharedMapAsync(); + }); + return result; + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/TrackingMapAPI.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/TrackingMapAPI.cs.meta new file mode 100644 index 0000000..f995573 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/TrackingMapAPI.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 2c2d1287ff38483f93fb02a197def246 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ViewConfiguration.cs b/com.microsoft.mixedreality.openxr/Runtime/API/ViewConfiguration.cs new file mode 100644 index 0000000..248d148 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ViewConfiguration.cs @@ -0,0 +1,166 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// A view configuration is a semantically meaningful set of one or more views for which an application can render images. + /// + public enum ViewConfigurationType + { + /// + /// A primary view configuration is a view configuration intended to be presented to the viewer interacting with the XR application. + /// + PrimaryStereo = 2, + + /// + /// This first-person observer view configuration intended for a first-person view of the scene to be composed onto video frames + /// being captured from a camera attached to and moved with the primary display on the form factor, which is generally for viewing + /// on a 2D screen by an external observer. This first-person camera will be facing forward with roughly the same perspective as + /// the primary views, and so the application should render its view to show objects that surround the user and avoid rendering the user's body avatar. + /// + SecondaryMonoFirstPersonObserver = 1000054000 + } + + /// + /// Observe and manage a view configuration of the current XR session. + /// + public class ViewConfiguration + { + private static MixedRealityFeaturePlugin Feature => OpenXRFeaturePlugin.Feature; + internal readonly OpenXRViewConfiguration m_openxrViewConfiguration; + + /// + /// Get all enabled view configurations when the XR session is started. + /// + public static IReadOnlyList EnabledViewConfigurations => + Feature.IsValidAndEnabled() ? Feature.EnabledViewConfigurations : Array.Empty(); + + /// + /// Get the primary view configuration of the XR session. + /// + public static ViewConfiguration Primary => Feature.IsValidAndEnabled() ? Feature.PrimaryViewConfiguration : null; + + internal ViewConfiguration(OpenXRViewConfiguration openxrViewConfiguration) + { + m_openxrViewConfiguration = openxrViewConfiguration; + } + + /// + /// Get the view configuration type + /// + public ViewConfigurationType ViewConfigurationType => m_openxrViewConfiguration.ViewConfigurationType; + + /// + /// Get whether or not this view configuration is active for the current frame. + /// If IsActive is false, the rendering into the view configuration will be ignored and not visible to user. + /// + public bool IsActive => m_openxrViewConfiguration.IsActive; + + /// + /// Adjustment to stereo separation in meters for primary stereo view configuration. + /// The value will be ignored for mono or secondary view configurations. + /// + public float StereoSeparationAdjustment + { + set => m_openxrViewConfiguration.SetStereoSeparationAdjustment(value); + get => m_openxrViewConfiguration.StereoSeparationAdjustment; + } + + /// + /// Get all supported reprojection modes for this view configuration. + /// + public IReadOnlyList SupportedReprojectionModes => m_openxrViewConfiguration.SupportedReprojectionModes; + + /// + /// Set the reprojection settings for the view configuration that will be used for the current frame. + /// + /// + /// The given setting only affects the current frame, and must be set for each frame to maintain the effect. + /// + /// The reprojection settings to be set. + public void SetReprojectionSettings(ReprojectionSettings settings) => m_openxrViewConfiguration.SetReprojectionSettings(settings); + } + + /// + /// The ReprojectionMode describes the reprojection mode of a projection composition layer. + /// + public enum ReprojectionMode + { + /// + /// Indicates the rendering may benefit from per-pixel depth reprojection. + /// This mode is typically used for world-locked content that should remain physically stationary as the user walks around. + /// + Depth = 1, + + /// + /// Indicates the rendering may benefit from planar reprojection and the plane can be calculated from the corresponding depth information. + /// This mode works better when the application knows the content is mostly placed on a plane. + /// + PlanarFromDepth = 2, + + /// + /// Indicates that the rendering may benefit from planar reprojection. + /// The application can customize the plane by ReprojectionSettings. + /// The app can also omit the plane override, indicating the runtime should use the default reprojection plane settings. + /// This mode works better when the application knows the content is mostly placed on a plane, or when it cannot afford to submit depth information. + /// + PlanarManual = 3, + + /// + /// Indicates the layer should be stabilized only for changes to orientation, ignoring positional changes. + /// This mode works better for body-locked content that should follow the user as they walk around, such as 360-degree video. + /// + OrientationOnly = 4, + + /// + /// Indicates the rendering should not be stabilized by the runtime. + /// + NoReprojection = -1 + } + + /// + /// The settings to control the reprojection of current rendering frame, + /// including the reprojection mode and optional stabilization plane override. + /// + public struct ReprojectionSettings + { + /// + /// The reprojection mode to be used with this view configuration. Overrides any reprojection mode + /// set in XRDisplaySubsystem. The default value is ReprojectionMode.Depth. + /// + public ReprojectionMode ReprojectionMode + { + get => m_reprojectionMode ?? ReprojectionMode.Depth; + set => m_reprojectionMode = value; + } + private ReprojectionMode? m_reprojectionMode; + + /// + /// When the application is confident that overriding the reprojection plane can benefit hologram + /// stability, it can provide this override to further help the runtime fine tune the reprojection + /// details. This Vector3 describes the position of the focus plane represented in the Unity scene. + /// + public Vector3? ReprojectionPlaneOverridePosition; + + /// + /// When the application is confident that overriding the reprojection plane can benefit hologram + /// stability, it can provide this override to further help the runtime fine tune the reprojection + /// details. This Vector3 is a unit vector describing the focus plane normal represented in the + /// Unity scene. + /// + public Vector3? ReprojectionPlaneOverrideNormal; + + /// + /// When the application is confident that overriding the reprojection plane can benefit hologram + /// stability, it can provide this override to further help the runtime fine tune the reprojection + /// details. This Vector3 is a velocity of the position in the Unity scene, measured in meters per + /// second. + /// + public Vector3? ReprojectionPlaneOverrideVelocity; + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/ViewConfiguration.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/ViewConfiguration.cs.meta new file mode 100644 index 0000000..7357e91 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/ViewConfiguration.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 86e8851215d071a429136c968f7c66e6 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorStore.cs b/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorStore.cs new file mode 100644 index 0000000..ecba39b --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorStore.cs @@ -0,0 +1,149 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using System.Threading.Tasks; +using UnityEngine.XR.ARFoundation; +using UnityEngine.XR.ARSubsystems; + +namespace Microsoft.MixedReality.OpenXR.ARFoundation +{ + /// + /// Provides extension methods to use an ARAnchorManager to load an XRAnchorStore. + /// + public static class AnchorManagerExtensions + { + /// + /// Use this ARAnchorManager to load an XRAnchorStore. + /// + /// + /// A task which, when completed, will contain a valid XRAnchorStore, or will contain null if the anchor store could not be loaded. + /// + /// The to receive the loaded anchors from the store. + /// + /// The anchor subsystem might not be available if the XR session is not initialized at start up. In this case, the returned anchor store might be null. + /// Make sure to reload the anchor store after the XR session is initialized. + /// + [System.Obsolete("Obsolete and will be removed in future releases. Use XRAnchorStore.LoadAnchorStoreAsync() function instead.", true)] + public static Task LoadAnchorStoreAsync(this ARAnchorManager anchorManager) + { + return XRAnchorStore.LoadAnchorStoreAsync(anchorManager.subsystem); + } + } +} + +namespace Microsoft.MixedReality.OpenXR.ARSubsystems +{ + /// + /// Provides extension methods to use an XRAnchorSubsystem to load an XRAnchorStore. + /// + public static class AnchorSubsystemExtensions + { + /// + /// Use this XRAnchorSubsystem to load an XRAnchorStore. + /// + /// + /// A task which, when completed, will contain a valid XRAnchorStore, or contain null if the anchor store could not be loaded. + /// + /// The to receive the loaded anchors from the store. + /// + /// The anchor subsystem might not be available if the XR session is not initialized at start up. In this case, the returned anchor store might be null. + /// Make sure to reload the anchor store after the XR session is initialized. + /// + [System.Obsolete("Obsolete and will be removed in future releases. Use XRAnchorStore.LoadAnchorStoreAsync() function instead.", true)] + public static Task LoadAnchorStoreAsync(this XRAnchorSubsystem anchorSubsystem) + { + return XRAnchorStore.LoadAnchorStoreAsync(anchorSubsystem); + } + } +} + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Handles persisting anchors from the scene to the anchor store, loading anchors from the anchor store to the scene, + /// and managing anchors persisted in the anchor store. + /// + /// + /// Persisting and unpersisting anchors from the anchor store does not affect their state in the Unity scene. + /// These operations only affect whether anchors will be available to load from the anchor store in the future. + /// + public class XRAnchorStore + { + /// + /// The names of all persisted anchors available in this anchor store. Each of these persisted anchors can be loaded with . + /// + public IReadOnlyList PersistedAnchorNames => m_openxrAnchorStore.PersistedAnchorNames; + + /// + /// Begin loading an anchor from the anchor store into the Unity scene. + /// On a future update of the ARAnchorManager or XRAnchorSubsystem, a new corresponding anchor will be created. + /// + /// If the persisted anchor has already been loaded, the TrackableId for the existing anchor in the scene will be returned. + /// The TrackableId which the anchor will use once it is created. + /// The name of the anchor to be loaded from the store. + public TrackableId LoadAnchor(string name) => m_openxrAnchorStore.LoadAnchor(name); + + /// + /// Persist an anchor in the anchor store, where it can be retrieved using . + /// + /// True if the anchor was successfully persisted, false if an error occurred. + /// The of an anchor to be persisted in the store. + /// A string to identify this anchor if it's successfully persisted in the store. + public bool TryPersistAnchor(TrackableId trackableId, string name) => m_openxrAnchorStore.TryPersistAnchor(name, trackableId); + + /// + /// Unpersist an anchor from the anchor store. + /// + /// After an anchor is unpersisted from the store, it will still be valid and locatable in the current session. + /// The name of the anchor to be unpersist from the store. + public void UnpersistAnchor(string name) => m_openxrAnchorStore.UnpersistAnchor(name); + + /// + /// Clear all persisted anchors from the anchor store. + /// + /// After the anchors are cleared from the anchor store, they will still be valid and locatable in the current session. + public void Clear() => m_openxrAnchorStore.Clear(); + + /// + /// This method will reload the anchor store. Use this if the head tracking map or anchors changed from outside the application. + /// + /// All anchors must be removed from the scene prior to this call. + /// True if the the reload succeeded, False if an error occurred. + public Task TryReloadAnchorStoreAsync() => m_openxrAnchorStore.TryReloadAnchorStoreAsync(); + + /// + /// Deprecated: Uses an existing to load an XRAnchorStore. + /// + /// + /// A task which, when completed, will contain a valid XRAnchorStore, or contain null if the anchor store could not be loaded. + /// + /// + /// The anchor subsystem might not be available if the XR session is not initialized at start up. In this case, the returned anchor store might be null. + /// Make sure to reload the anchor store after the XR session is initialized. + /// + [Obsolete("This method is obsolete. Use the XRAnchorStore.LoadAnchorStoreAsync() function instead.", false)] + public static async Task LoadAsync(XRAnchorSubsystem anchorSubsystem) => await LoadAnchorStoreAsync(anchorSubsystem); + + /// + /// A task which, when completed, will contain a valid XRAnchorStore, or contain null if the anchor store could not be loaded. + /// + /// + /// The anchor subsystem might not be available if the XR session is not initialized at start up. In this case, the returned anchor store might be null. + /// Make sure to reload the anchor store after the XR session is initialized. + /// + public static async Task LoadAnchorStoreAsync(XRAnchorSubsystem anchorSubsystem) + { + OpenXRAnchorStore openxrAnchorStore = await OpenXRAnchorStoreFactory.LoadAnchorStoreAsync(anchorSubsystem); + return openxrAnchorStore == null ? null : new XRAnchorStore(openxrAnchorStore); + } + + internal XRAnchorStore(OpenXRAnchorStore openxrAnchorStore) + { + m_openxrAnchorStore = openxrAnchorStore; + } + + private readonly OpenXRAnchorStore m_openxrAnchorStore; + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorStore.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorStore.cs.meta new file mode 100644 index 0000000..f91ab82 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorStore.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 851b37b663296334ba335f3e10defb82 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorTransferBatch.cs b/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorTransferBatch.cs new file mode 100644 index 0000000..41cf0a5 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorTransferBatch.cs @@ -0,0 +1,110 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System.Collections.Generic; +using System.IO; +using System.Threading.Tasks; +using UnityEngine.XR.ARSubsystems; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Provides the ability to build up a batch of anchors and export them to a binary stream for transfer. + /// Typically on a second device, it then supports importing the transfer stream and loading in the original batch of anchors. + /// + /// Use of this class requires an ARAnchorManager in the scene or some other manual management of an XRAnchorSubsystem. + public class XRAnchorTransferBatch + { + /// + /// Constructor. + /// + public XRAnchorTransferBatch() : this(new AnchorTransferBatch()) { } + + private XRAnchorTransferBatch(AnchorTransferBatch anchorTransferBatch) + { + m_anchorTransferBatch = anchorTransferBatch; + } + + private readonly AnchorTransferBatch m_anchorTransferBatch; + + /// + /// Provides a list of all identifiers currently mapped in this AnchorTransferBatch. + /// + public IReadOnlyList AnchorNames => m_anchorTransferBatch.AnchorNames; + + /// + /// Tries to convert and add an anchor with the corresponding to an export list. + /// + /// Call to get the transferable anchor data. + /// The of an anchor to be exported. + /// A string to identify this anchor upon import to another device. + /// Whether the anchor was successfully converted into a Perception SpatialAnchor and added to the export list. + public bool AddAnchor(TrackableId trackableId, string name) => m_anchorTransferBatch.AddAnchor(trackableId, name); + + /// + /// Removes an anchor from the transfer batch. Doesn't remove the existing Unity anchor, if one is present. + /// + /// After an anchor is removed from the transfer batch, it will still be valid and locatable in the current session. + /// The name of the anchor to be removed from the transfer batch. + public void RemoveAnchor(string name) => m_anchorTransferBatch.RemoveAnchor(name); + + /// + /// Removes all anchors from the transfer batch. Doesn't remove any existing Unity anchors, if present. + /// + /// After the anchors are cleared from the transfer batch, they will still be valid and locatable in the current session. + public void Clear() => m_anchorTransferBatch.Clear(); + + /// + /// Attempts to load a specified anchor from the transfer batch and reports it to Unity as an XRAnchor/ARAnchor. + /// + /// It's then typically recommended to use an ARAnchorManager to access the resulting Unity anchor. + /// The anchor's identifier from the transfer batch. + /// The of the resulting Unity anchor if successfully loaded, or TrackableId.invalidId if the given name is not found. + public TrackableId LoadAnchor(string name) => m_anchorTransferBatch.LoadAnchor(name); + + /// + /// Attempts to load a specified anchor from the transfer batch and replace the specified Unity anchor's tracking data with the new anchor. + /// + /// The anchor's identifier from the transfer batch. + /// The existing Unity anchor to update to track this new spatial anchor. + /// The of the resulting Unity anchor (usually the same as the passed-in parameter) if successfully loaded, + /// or TrackableId.invalidId if the given name is not found. + public TrackableId LoadAndReplaceAnchor(string name, TrackableId trackableId) => m_anchorTransferBatch.LoadAndReplaceAnchor(name, trackableId); + + /// + /// Exports any anchors added via into a Stream for transfer. Use for reading this Stream. + /// + /// The anchor transfer batch instance to export from. This instance should have had anchors added before attempting export. + /// A task which, when completed, will contain the exported array, or null if the export was unsuccessful. + public static async Task ExportAsync(XRAnchorTransferBatch anchorTransferBatch) + { + MemoryStream output = new MemoryStream(); + SerializationCompletionReason reason = await anchorTransferBatch.m_anchorTransferBatch.ExportAsync(output); + + if (reason == SerializationCompletionReason.Succeeded) + { + return output; + } + + return null; + } + + /// + /// Imports the provided Stream into an . + /// + /// The streamed data representing the result of a call to . This stream must be readable. + /// A task which, when completed, will contain the resulting XRAnchorTransferBatch, or null if the import was unsuccessful. + public static async Task ImportAsync(Stream inputStream) + { + AnchorTransferBatch anchorTransfer = new AnchorTransferBatch(); + SerializationCompletionReason reason = await anchorTransfer.ImportAsync(inputStream); + + if (reason == SerializationCompletionReason.Succeeded) + { + return new XRAnchorTransferBatch(anchorTransfer); + } + + return null; + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorTransferBatch.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorTransferBatch.cs.meta new file mode 100644 index 0000000..b00263f --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/API/XRAnchorTransferBatch.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: eecf66f01160b98408ac129343201009 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/AssemblyInfo.cs b/com.microsoft.mixedreality.openxr/Runtime/AssemblyInfo.cs new file mode 100644 index 0000000..e0c789f --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/AssemblyInfo.cs @@ -0,0 +1,12 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System.Reflection; +using System.Runtime.CompilerServices; + +[assembly: InternalsVisibleTo("Microsoft.MixedReality.OpenXR.Editor")] +[assembly: InternalsVisibleTo("Microsoft.MixedReality.OpenXR.Tests")] +[assembly: InternalsVisibleTo("Microsoft.MixedReality.OpenXR.Internal")] +[assembly: InternalsVisibleTo("Microsoft.MixedReality.OpenXR.Internal.Editor")] + +[assembly: AssemblyVersion("1.11.1")] diff --git a/com.microsoft.mixedreality.openxr/Runtime/AssemblyInfo.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/AssemblyInfo.cs.meta new file mode 100644 index 0000000..2996d3d --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/AssemblyInfo.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 3c8378ddbfc33df4aa66c2a3ee3ef81f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins.meta b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins.meta new file mode 100644 index 0000000..8324d97 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 8044fce2b68cec14897fcad810aa21a1 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/AppRemotingPlugin.cs b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/AppRemotingPlugin.cs new file mode 100644 index 0000000..57bfc0b --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/AppRemotingPlugin.cs @@ -0,0 +1,104 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using UnityEngine; + +#if UNITY_EDITOR +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; +#endif + +namespace Microsoft.MixedReality.OpenXR.Remoting +{ +#if UNITY_EDITOR + [OpenXRFeature(UiName = featureName, + BuildTargetGroups = new[] { BuildTargetGroup.Standalone, BuildTargetGroup.WSA }, + Company = "Microsoft", + Desc = "Feature to enable " + featureName + ".", + DocumentationLink = "https://aka.ms/openxr-unity-app-remoting", + OpenxrExtensionStrings = requestedExtensions, + Category = FeatureCategory.Feature, + Required = false, + Priority = -100, // hookup before other plugins so it affects json before GetProcAddr. + FeatureId = featureId, + Version = "1.11.1")] +#endif + [RequiresNativePluginDLLs] + internal class AppRemotingPlugin : OpenXRFeaturePlugin + { + internal const string featureId = "com.microsoft.openxr.feature.appremoting"; + internal const string featureName = "Holographic Remoting remote app"; + private const string requestedExtensions = "XR_MSFT_holographic_remoting XR_MSFT_holographic_remoting_speech"; + + private OpenXRRuntimeRestartHandler m_restartHandler = null; + + protected override IntPtr HookGetInstanceProcAddr(IntPtr func) + { + if (enabled) + { + AppRemotingSubsystem.GetCurrent().TryEnableRemotingOverride(); + } + + return base.HookGetInstanceProcAddr(func); + } + + protected override void OnSubsystemCreate() + { + base.OnSubsystemCreate(); + + if (enabled && m_restartHandler == null) + { + m_restartHandler = new OpenXRRuntimeRestartHandler(this, skipRestart: true, skipQuitApp: true); + } + else if (!enabled && m_restartHandler != null) + { + m_restartHandler.Dispose(); + m_restartHandler = null; + } + } + + protected override void OnInstanceDestroy(ulong instance) + { + if (enabled) + { + AppRemotingSubsystem.GetCurrent().ResetRemotingOverride(); + } + + Debug.Log($"[AppRemotingPlugin] OnInstanceDestroy, remotingState was {AppRemotingSubsystem.AppRemotingState}."); + base.OnInstanceDestroy(instance); + } + + protected override void OnSystemChange(ulong systemId) + { + base.OnSystemChange(systemId); + + if (systemId != 0) + { + Debug.Log($"[AppRemotingPlugin] OnSystemChange, systemId = {systemId}"); + if(enabled) + { + AppRemotingSubsystem.GetCurrent().InitializeRemoting(); + } + } + } + + protected override void OnSessionStateChange(int oldState, int newState) + { + if ((XrSessionState)newState == XrSessionState.LossPending) + { + if(enabled) + { + AppRemotingSubsystem.GetCurrent().OnSessionLossPending(); + } + } + } + +#if UNITY_EDITOR + protected override void GetValidationChecks(System.Collections.Generic.List results, BuildTargetGroup targetGroup) + { + AppRemotingValidator.GetValidationChecks(this, results, targetGroup); + } +#endif + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/AppRemotingPlugin.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/AppRemotingPlugin.cs.meta new file mode 100644 index 0000000..373578d --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/AppRemotingPlugin.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 2d2e2731103cdda44af77955a0b4814c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HPMixedRealityControllerProfile.cs b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HPMixedRealityControllerProfile.cs new file mode 100644 index 0000000..9e0747e --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HPMixedRealityControllerProfile.cs @@ -0,0 +1,548 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEngine.XR.OpenXR.Features; + +#if ENABLE_VR || (UNITY_GAMECORE && INPUT_SYSTEM_1_4_OR_NEWER) +using System.Collections.Generic; +using UnityEngine.InputSystem; +using UnityEngine.InputSystem.Layouts; +using UnityEngine.InputSystem.Controls; +using UnityEngine.InputSystem.XR; +using UnityEngine.Scripting; +using UnityEngine.XR; +using UnityEngine.XR.OpenXR.Input; + +#if USE_INPUT_SYSTEM_POSE_CONTROL +using PoseControl = UnityEngine.InputSystem.XR.PoseControl; +#else +using PoseControl = UnityEngine.XR.OpenXR.Input.PoseControl; +#endif +#endif + +#if UNITY_EDITOR +using UnityEditor; +#endif + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// This enables the use of the HP Mixed Reality Controller interaction profile in OpenXR. + /// +#if UNITY_EDITOR + [UnityEditor.XR.OpenXR.Features.OpenXRFeature(UiName = "HP Reverb G2 Controller Profile", + BuildTargetGroups = new[] { BuildTargetGroup.Standalone, BuildTargetGroup.WSA }, + Company = "Microsoft", + Desc = "Supports the input mapping to the HP Reverb G2 controller.", + DocumentationLink = "https://aka.ms/openxr-unity", + CustomRuntimeLoaderBuildTargets = null, + OpenxrExtensionStrings = requestedExtensions, + Required = false, + Category = UnityEditor.XR.OpenXR.Features.FeatureCategory.Interaction, + FeatureId = featureId, + Version = "1.11.1")] +#endif + internal class HPMixedRealityControllerProfile : OpenXRInteractionFeature + { + internal const string featureId = "com.microsoft.openxr.feature.interaction.hpmixedrealitycontroller"; + private const string requestedExtensions = "XR_EXT_hp_mixed_reality_controller"; + +#if ENABLE_VR || (UNITY_GAMECORE && INPUT_SYSTEM_1_4_OR_NEWER) + private const string kDeviceLocalizedName = "HP Reverb G2 Controller OpenXR"; + private const string kDeviceDisplayName = "HP Reverb G2 Controller (OpenXR)"; + + /// + /// An Input System device based off the hand interaction profile in the HP Reverb G2 Controller. + /// + [Preserve, InputControlLayout(displayName = kDeviceDisplayName, commonUsages = new[] { "LeftHand", "RightHand" })] + public class HPMixedRealityController : XRControllerWithRumble + { + /// + /// A representing the OpenXR binding. + /// + [Preserve, InputControl(aliases = new[] { "Primary2DAxis", "Joystick" })] + public Vector2Control thumbstick { get; private set; } + + /// + /// A representing the OpenXR binding. + /// + [Preserve, InputControl(aliases = new[] { "GripAxis", "squeeze" })] + public AxisControl grip { get; private set; } + + /// + /// A representing the OpenXR binding. + /// + [Preserve, InputControl(aliases = new[] { "GripButton", "squeezeClicked" })] + public ButtonControl gripPressed { get; private set; } + + /// + /// A representing the OpenXR binding. + /// + [Preserve, InputControl(aliases = new[] { "menuButton" })] + public ButtonControl menu { get; private set; } + + /// + /// A representing the OpenXR bindings, depending on handedness. + /// + [Preserve, InputControl(aliases = new[] { "A", "X", "buttonA", "buttonX" })] + public ButtonControl primaryButton { get; private set; } + + /// + /// A representing the OpenXR bindings, depending on handedness. + /// + [Preserve, InputControl(aliases = new[] { "B", "Y", "buttonB", "buttonY" })] + public ButtonControl secondaryButton { get; private set; } + + /// + /// A representing the OpenXR binding. + /// + [Preserve, InputControl] + public AxisControl trigger { get; private set; } + + /// + /// A representing the OpenXR binding. + /// + [Preserve, InputControl(aliases = new[] { "indexButton", "indexTouched", "triggerbutton" })] + public ButtonControl triggerPressed { get; private set; } + + /// + /// A representing the OpenXR binding. + /// + [Preserve, InputControl(aliases = new[] { "joystickOrPadPressed" })] + public ButtonControl thumbstickClicked { get; private set; } + + /// + /// A representing the OpenXR binding. + /// + [Preserve, InputControl(offset = 0, aliases = new[] { "device", "gripPose" })] + public PoseControl devicePose { get; private set; } + + /// + /// A representing the OpenXR binding. + /// + [Preserve, InputControl(offset = 0, aliases = new[] { "aimPose" })] + public PoseControl pointer { get; private set; } + + /// + /// A required for back compatibility with the XRSDK layouts. This represents the overall tracking state of the device. This value is equivalent to mapping devicePose/isTracked. + /// + [Preserve, InputControl(offset = 26)] + new public ButtonControl isTracked { get; private set; } + + /// + /// A required for back compatibility with the XRSDK layouts. This represents the bit flag set indicating what data is valid. This value is equivalent to mapping devicePose/trackingState. + /// + [Preserve, InputControl(offset = 28)] + new public IntegerControl trackingState { get; private set; } + + /// + /// A required for back compatibility with the XRSDK layouts. This is the device position. For the HP mixed reality controller, this is the grip position. This value is equivalent to mapping devicePose/position. + /// + [Preserve, InputControl(offset = 32, aliases = new[] { "gripPosition" })] + new public Vector3Control devicePosition { get; private set; } + + /// + /// A required for back compatibility with the XRSDK layouts. This is the device orientation. For the HP mixed reality controller, this is the grip rotation. This value is equivalent to mapping devicePose/rotation. + /// + [Preserve, InputControl(offset = 44, aliases = new[] { "deviceOrientation", "gripRotation", "gripOrientation" })] + new public QuaternionControl deviceRotation { get; private set; } + + /// + /// A required for backwards compatibility with the XRSDK layouts. This is the pointer position. This value is equivalent to mapping pointerPose/position. + /// + [Preserve, InputControl(offset = 92)] + public Vector3Control pointerPosition { get; private set; } + + /// + /// A required for backwards compatibility with the XRSDK layouts. This is the pointer rotation. This value is equivalent to mapping pointerPose/rotation. + /// + [Preserve, InputControl(offset = 104, aliases = new[] { "pointerOrientation" })] + public QuaternionControl pointerRotation { get; private set; } + + /// + /// A that represents the binding. + /// + [Preserve, InputControl(usage = "Haptic")] + public HapticControl haptic { get; private set; } + + /// + /// Internal call used to assign controls to the correct element. + /// + protected override void FinishSetup() + { + base.FinishSetup(); + + thumbstick = GetChildControl("thumbstick"); + thumbstickClicked = GetChildControl("thumbstickClicked"); + trigger = GetChildControl("trigger"); + triggerPressed = GetChildControl("triggerPressed"); + grip = GetChildControl("grip"); + gripPressed = GetChildControl("gripPressed"); + + menu = GetChildControl("menu"); + primaryButton = GetChildControl("primaryButton"); + secondaryButton = GetChildControl("secondaryButton"); + + devicePose = GetChildControl("devicePose"); + pointer = GetChildControl("pointer"); + + isTracked = GetChildControl("isTracked"); + trackingState = GetChildControl("trackingState"); + devicePosition = GetChildControl("devicePosition"); + deviceRotation = GetChildControl("deviceRotation"); + pointerPosition = GetChildControl("pointerPosition"); + pointerRotation = GetChildControl("pointerRotation"); + + haptic = GetChildControl("haptic"); + } + } +#endif + + /// + /// The interaction profile string used to reference the HP Mixed Reality Controller. + /// + public const string profile = "/interaction_profiles/hp/mixed_reality_controller"; + + // Available Bindings + // Left Hand Only + /// + /// Constant for a interaction binding '.../input/x/click' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. This binding is only available for the user path. + /// + public const string buttonX = "/input/x/click"; + /// + /// Constant for a interaction binding '.../input/y/click' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. This binding is only available for the user path. + /// + public const string buttonY = "/input/y/click"; + + // Right Hand Only + /// + /// Constant for a interaction binding '.../input/a/click' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. This binding is only available for the user path. + /// + public const string buttonA = "/input/a/click"; + /// + /// Constant for a interaction binding '..."/input/b/click' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. This binding is only available for the user path. + /// + public const string buttonB = "/input/b/click"; + + // Both Hands + /// + /// Constant for a interaction binding '.../input/menu/click' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. + /// + public const string menu = "/input/menu/click"; + /// + /// Constant for a interaction binding '.../input/squeeze/value' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. + /// + public const string squeeze = "/input/squeeze/value"; + /// + /// Constant for a interaction binding '.../input/trigger/value' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. + /// + public const string trigger = "/input/trigger/value"; + /// + /// Constant for a interaction binding '.../input/thumbstick' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. + /// + public const string thumbstick = "/input/thumbstick"; + /// + /// Constant for a interaction binding '.../input/thumbstick/click' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. + /// + public const string thumbstickClick = "/input/thumbstick/click"; + /// + /// Constant for a interaction binding '.../input/grip/pose' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. + /// + public const string grip = "/input/grip/pose"; + /// + /// Constant for a interaction binding '.../input/aim/pose' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. + /// + public const string aim = "/input/aim/pose"; + /// + /// Constant for a interaction binding '.../output/haptic' OpenXR Input Binding. Used by the input subsystem to bind actions to physical inputs. + /// + public const string haptic = "/output/haptic"; + +#if ENABLE_VR || (UNITY_GAMECORE && INPUT_SYSTEM_1_4_OR_NEWER) + /// + protected override void RegisterDeviceLayout() + { + InputSystem.RegisterLayout(typeof(HPMixedRealityController), + matches: new InputDeviceMatcher() + .WithInterface(XRUtilities.InterfaceMatchAnyVersion) + .WithProduct(kDeviceLocalizedName)); + } + + /// + protected override void UnregisterDeviceLayout() + { + InputSystem.RemoveLayout(typeof(HPMixedRealityController).Name); + } + + /// + protected override void RegisterActionMapsWithRuntime() + { + ActionMapConfig actionMap = new ActionMapConfig() + { + name = "hpmixedrealitycontroller", + localizedName = kDeviceLocalizedName, + desiredInteractionProfile = profile, + manufacturer = "HP", + serialNumber = "", + deviceInfos = new List() + { + new DeviceConfig() + { + characteristics = InputDeviceCharacteristics.HeldInHand | InputDeviceCharacteristics.TrackedDevice | InputDeviceCharacteristics.Controller | InputDeviceCharacteristics.Left, + userPath = UserPaths.leftHand + }, + new DeviceConfig() + { + characteristics = InputDeviceCharacteristics.HeldInHand | InputDeviceCharacteristics.TrackedDevice | InputDeviceCharacteristics.Controller | InputDeviceCharacteristics.Right, + userPath = UserPaths.rightHand + } + }, + actions = new List() + { + // Joystick + new ActionConfig() + { + name = "thumbstick", + localizedName = "Thumbstick", + type = ActionType.Axis2D, + usages = new List() + { + "Primary2DAxis" + }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = thumbstick, + interactionProfileName = profile, + } + } + }, + // A / X Press + new ActionConfig() + { + name = "primarybutton", + localizedName = "Primary Button", + type = ActionType.Binary, + usages = new List() + { + "PrimaryButton" + }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = buttonX, + interactionProfileName = profile, + userPaths = new List() { UserPaths.leftHand } + }, + new ActionBinding() + { + interactionPath = buttonA, + interactionProfileName = profile, + userPaths = new List() { UserPaths.rightHand } + }, + } + }, + // B / Y Press + new ActionConfig() + { + name = "secondarybutton", + localizedName = "Secondary Button", + type = ActionType.Binary, + usages = new List() + { + "SecondaryButton" + }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = buttonY, + interactionProfileName = profile, + userPaths = new List() { UserPaths.leftHand } + }, + new ActionBinding() + { + interactionPath = buttonB, + interactionProfileName = profile, + userPaths = new List() { UserPaths.rightHand } + }, + } + }, + // Menu + new ActionConfig() + { + name = "menu", + localizedName = "Menu", + type = ActionType.Binary, + usages = new List() + { + "MenuButton" + }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = menu, + interactionProfileName = profile, + } + } + }, + // Grip + new ActionConfig() + { + name = "grip", + localizedName = "Grip", + type = ActionType.Axis1D, + usages = new List() + { + "Grip" + }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = squeeze, + interactionProfileName = profile, + } + } + }, + // Grip Pressed + new ActionConfig() + { + name = "grippressed", + localizedName = "Grip Pressed", + type = ActionType.Binary, + usages = new List() + { + "GripButton" + }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = squeeze, + interactionProfileName = profile, + } + } + }, + // Trigger + new ActionConfig() + { + name = "trigger", + localizedName = "Trigger", + type = ActionType.Axis1D, + usages = new List() + { + "Trigger" + }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = trigger, + interactionProfileName = profile, + } + } + }, + // Trigger Pressed + new ActionConfig() + { + name = "triggerpressed", + localizedName = "Trigger Pressed", + type = ActionType.Binary, + usages = new List() + { + "TriggerButton" + }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = trigger, + interactionProfileName = profile, + } + } + }, + // Thumbstick Clicked + new ActionConfig() + { + name = "thumbstickclicked", + localizedName = "Thumbstick Clicked", + type = ActionType.Binary, + usages = new List() + { + "Primary2DAxisClick" + }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = thumbstickClick, + interactionProfileName = profile, + } + } + }, + // Device Pose + new ActionConfig() + { + name = "devicepose", + localizedName = "Device Pose", + type = ActionType.Pose, + usages = new List() + { + "Device" + }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = grip, + interactionProfileName = profile, + } + } + }, + // Pointer Pose + new ActionConfig() + { + name = "pointer", + localizedName = "Pointer Pose", + type = ActionType.Pose, + usages = new List() + { + "Pointer" + }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = aim, + interactionProfileName = profile, + } + } + }, + // Haptics + new ActionConfig() + { + name = "haptic", + localizedName = "Haptic Output", + type = ActionType.Vibrate, + usages = new List() { "Haptic" }, + bindings = new List() + { + new ActionBinding() + { + interactionPath = haptic, + interactionProfileName = profile, + } + } + } + } + }; + + AddActionMap(actionMap); + } +#endif + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HPMixedRealityControllerProfile.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HPMixedRealityControllerProfile.cs.meta new file mode 100644 index 0000000..56bfaaf --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HPMixedRealityControllerProfile.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 3482401f887b8864183e401715462f46 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HandTrackingFeaturePlugin.cs b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HandTrackingFeaturePlugin.cs new file mode 100644 index 0000000..59a5c9c --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HandTrackingFeaturePlugin.cs @@ -0,0 +1,159 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEngine; + +#if ENABLE_VR || (UNITY_GAMECORE && INPUT_SYSTEM_1_4_OR_NEWER) +using UnityEngine.InputSystem; +using UnityEngine.InputSystem.Controls; +using UnityEngine.InputSystem.Layouts; +using UnityEngine.InputSystem.XR; +using UnityEngine.XR.OpenXR.Input; +#if UNITY_EDITOR +using System.Linq; +using UnityEngine.XR.OpenXR; +using UnityEngine.XR.OpenXR.Features; +#endif +#endif + +#if UNITY_EDITOR +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; +#endif + +namespace Microsoft.MixedReality.OpenXR +{ +#if UNITY_EDITOR + [OpenXRFeature(UiName = featureName, + BuildTargetGroups = new[] { BuildTargetGroup.Standalone, BuildTargetGroup.WSA, BuildTargetGroup.Android}, + Company = "Microsoft", + Desc = "Supports articulated hand tracking with 26 hand joints.", + DocumentationLink = "https://aka.ms/openxr-unity", + CustomRuntimeLoaderBuildTargets = null, + OpenxrExtensionStrings = requestedExtensions, + Required = false, + Category = FeatureCategory.Feature, + FeatureId = featureId, + Version = "1.11.1")] +#endif + [RequiresNativePluginDLLs] + internal class HandTrackingFeaturePlugin : OpenXRFeaturePlugin + { + internal const string featureId = "com.microsoft.openxr.feature.handtracking"; + internal const string featureName = "Hand Tracking"; + private const string requestedExtensions = + "XR_EXT_hand_tracking " + + "XR_EXT_hand_joints_motion_range " + + "XR_EXT_hand_tracking_data_source " + + "XR_MSFT_hand_tracking_mesh"; + + [SerializeField] + private HandTrackingOptions leftHandTrackingOptions = default; + + [SerializeField] + private HandTrackingOptions rightHandTrackingOptions = default; + + internal enum QuestHandTracking + { + v1, + v2, + } + +#if UNITY_EDITOR + [EditorDrawerVisibleToBuildTarget(BuildTargetGroup.Android)] +#endif + [SerializeField, + Tooltip("Allows for toggling specific versions of the Quest hand tracking runtime."), + LabelWidth(200f), + DocURL("https://developer.oculus.com/blog/presence-platforms-hand-tracking-api-gets-an-upgrade/")] + private QuestHandTracking questHandTrackingMode = QuestHandTracking.v2; + + internal QuestHandTracking QuestHandTrackingMode => questHandTrackingMode; + + private HandTrackingSubsystemController m_handTrackingSubsystemController; + + HandTrackingFeaturePlugin() + { + AddSubsystemController(m_handTrackingSubsystemController = new HandTrackingSubsystemController(this)); + } + + protected override void OnSubsystemStart() + { + base.OnSubsystemStart(); + NativeLib.SetHandJointsMotionRange(Handedness.Left, leftHandTrackingOptions.MotionRange); + NativeLib.SetHandJointsMotionRange(Handedness.Right, rightHandTrackingOptions.MotionRange); + } + +#if ENABLE_VR || (UNITY_GAMECORE && INPUT_SYSTEM_1_4_OR_NEWER) + /// + protected override bool OnInstanceCreate(ulong instance) + { + RegisterDeviceLayout(); + return base.OnInstanceCreate(instance); + } + + [UnityEngine.Scripting.Preserve, InputControlLayout(displayName = featureName + " (OpenXR)", commonUsages = new[] { "LeftHand", "RightHand" }, isGenericTypeOfDevice = true)] + public class OpenXRHandTracking : OpenXRDevice + { + [InputControl] + public ButtonControl isTracked { get; private set; } + + [InputControl] + public IntegerControl trackingState { get; private set; } + + protected override void FinishSetup() + { + base.FinishSetup(); + + isTracked = GetChildControl("isTracked"); + trackingState = GetChildControl("trackingState"); + } + } + + private static void RegisterDeviceLayout() + { + InputSystem.RegisterLayout(typeof(OpenXRHandTracking), + matches: new InputDeviceMatcher() + .WithInterface(XRUtilities.InterfaceMatchAnyVersion) + .WithProduct("OpenXR (Right|Left) Hand")); + } + + private static void UnregisterDeviceLayout() + { + InputSystem.RemoveLayout(nameof(OpenXRHandTracking)); + } + +#if UNITY_EDITOR + /// + protected override void OnEnabledChange() + { + base.OnEnabledChange(); + CheckRegistration(); + } + + /// + /// In the editor, we need to make sure the device layout gets registered + /// even if the user doesn't navigate to the project settings. + /// + [InitializeOnLoadMethod] + private static void CheckRegistration() + { + // Keep the layouts registered in the editor as long as at least one of the build target + // groups has the feature enabled. + EditorBuildSettings.TryGetConfigObject(Constants.k_SettingsKey, out UnityEngine.Object obj); + if (obj is IPackageSettings packageSettings) + { + if (packageSettings != null && packageSettings.GetFeatures().Any(f => f.feature.enabled)) + { + RegisterDeviceLayout(); + } + else + { + UnregisterDeviceLayout(); + } + } + } +#endif // UNITY_EDITOR +#endif // ENABLE_VR || (UNITY_GAMECORE && INPUT_SYSTEM_1_4_OR_NEWER) + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HandTrackingFeaturePlugin.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HandTrackingFeaturePlugin.cs.meta new file mode 100644 index 0000000..fb1fbb9 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/HandTrackingFeaturePlugin.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c79c911b38743a649b1c1eddb5097202 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MixedRealityFeaturePlugin.cs b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MixedRealityFeaturePlugin.cs new file mode 100644 index 0000000..4abed87 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MixedRealityFeaturePlugin.cs @@ -0,0 +1,144 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using UnityEngine; +using UnityEngine.XR.OpenXR.Features; +using Microsoft.MixedReality.OpenXR.ARSubsystems; + +#if UNITY_EDITOR +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; +#endif + +namespace Microsoft.MixedReality.OpenXR +{ + +#if UNITY_EDITOR + [OpenXRFeature(UiName = featureName, + BuildTargetGroups = new[] { BuildTargetGroup.Standalone, BuildTargetGroup.WSA }, + Company = "Microsoft", + Desc = "Supports features on HoloLens 2 and Mixed Reality headsets.", + DocumentationLink = "https://aka.ms/openxr-unity", + CustomRuntimeLoaderBuildTargets = null, + OpenxrExtensionStrings = requestedExtensions, + Required = true, + Category = FeatureCategory.Feature, + FeatureId = featureId, + Version = "1.11.1")] +#endif + [RequiresNativePluginDLLs] + internal class MixedRealityFeaturePlugin : OpenXRFeaturePlugin + { + internal enum ValidationRuleTargetPlatform + { + None = 0, + [InspectorName("HoloLens 2")] + HoloLens2 = 1 + } + + internal const string featureId = "com.microsoft.openxr.feature.hololens"; + internal const string featureName = "Mixed Reality Features"; + internal const string mixedRealityExtensions = "" + + "XR_MSFT_unbounded_reference_space " + + "XR_MSFT_spatial_anchor " + + "XR_MSFT_secondary_view_configuration " + + "XR_MSFT_first_person_observer " + + "XR_MSFT_spatial_graph_bridge " + + "XR_MSFT_perception_anchor_interop " + + "XR_MSFT_spatial_anchor_persistence " + + "XR_MSFT_scene_understanding " + + "XR_MSFT_scene_understanding_serialization " + + "XR_MSFT_scene_marker " + + "XR_MSFT_spatial_anchor_export_preview " + + "XR_MSFT_composition_layer_reprojection"; // Do not add space at the end + + internal const string requestedExtensions = "" + + "XR_MSFT_holographic_window_attachment " + + "XR_KHR_win32_convert_performance_counter_time " + + mixedRealityExtensions; + + [Header("Mixed Reality Plugin Settings")] + + [SerializeField, LabelWidth(250f), + Tooltip("Using first person observer, Mixed Reality Capture (MRC) will render from the perspective of the PV camera with an extra rendering pass. " + + "This provides better hologram quality and alignment to the physical world but may use more rendering resources. " + + "When this flag is set, MRC will render from one of the eyes without an extra rendering pass, reducing the " + + "rendering cost for MRC but potentially introducing visual disparity, especially on hand tracking visuals.")] + private bool disableFirstPersonObserver = false; + + [SerializeField, LabelWidth(250f), + Tooltip("Using the before-render pose update allows the app to use lower latency action poses in the render scripts." + + "However, this pose update adds cost in the pre-render phase. The before-render pose update is disabled by default " + + "so that action poses are updated once a frame in the update phase which has 2 frames of latency.")] + private bool enablePoseUpdateOnBeforeRender = false; + + +#if UNITY_EDITOR + [EditorDrawerVisibleToBuildTarget(BuildTargetGroup.WSA)] +#endif + [SerializeField, HideInInspector] + internal ValidationRuleTargetPlatform validationRuleTarget; + + private OpenXRViewConfigurationSettings m_viewConfigurationSettings; + + internal MixedRealityFeaturePlugin() + { + AddSubsystemController(new SessionSubsystemController(this)); + AddSubsystemController(new AnchorSubsystemController(this)); + AddSubsystemController(new PlaneSubsystemController(this)); + AddSubsystemController(new RaycastSubsystemController(this)); + AddSubsystemController(new MeshSubsystemController(this)); + AddSubsystemController(m_viewConfigurationSettings = new OpenXRViewConfigurationSettings(this)); + AddSubsystemController(new MarkerSubsystemController(this)); + } + +#if UNITY_EDITOR + // UnityEditor.SessionState will store these across assembly reloading, but will be cleared when the Unity editor is closed. + internal static string VersionInstalledOnLaunch + { + get => UnityEditor.SessionState.GetString(versionInstalledOnLaunchKey, defaultValue: string.Empty); + set => UnityEditor.SessionState.SetString(versionInstalledOnLaunchKey, value); + } + private const string versionInstalledOnLaunchKey = "MixedRealityFeaturePlugin_VersionInstalledOnLaunch"; + + protected override void OnEnable() + { + base.OnEnable(); + + // Only cache the version here on first load. + if (string.IsNullOrWhiteSpace(VersionInstalledOnLaunch)) + { + VersionInstalledOnLaunch = UnityEditor.PackageManager.PackageInfo.FindForAssembly(typeof(MixedRealityFeaturePlugin).Assembly)?.version; + } + } +#endif // UNITY_EDITOR + + protected override bool OnInstanceCreate(ulong instance) + { + bool returnValue = base.OnInstanceCreate(instance); + + NativeLib.SetMixedRealityPluginOptions(new MixedRealityPluginOptions() + { + DisableFirstPersonObserver = disableFirstPersonObserver, + EnablePoseUpdateOnBeforeRender = enablePoseUpdateOnBeforeRender, + }); + + return returnValue; + } + + internal IReadOnlyList EnabledViewConfigurations + => m_viewConfigurationSettings.EnabledViewConfigurations; + + internal ViewConfiguration PrimaryViewConfiguration => m_viewConfigurationSettings.PrimaryViewConfiguration; + + +#if UNITY_EDITOR + protected override void GetValidationChecks(List results, BuildTargetGroup targetGroup) + { + MixedRealityFeatureValidator.GetValidationChecks(this, results, targetGroup); + } +#endif + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MixedRealityFeaturePlugin.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MixedRealityFeaturePlugin.cs.meta new file mode 100644 index 0000000..bed4898 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MixedRealityFeaturePlugin.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 3f8ec2975f18d5e479159feb34b4dc86 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MotionControllerFeaturePlugin.cs b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MotionControllerFeaturePlugin.cs new file mode 100644 index 0000000..ed7d00a --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MotionControllerFeaturePlugin.cs @@ -0,0 +1,32 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; + +#if UNITY_EDITOR +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; +#endif + +namespace Microsoft.MixedReality.OpenXR +{ +#if UNITY_EDITOR + [OpenXRFeature(UiName = "Motion Controller Model", + BuildTargetGroups = new[] { BuildTargetGroup.Standalone, BuildTargetGroup.WSA, BuildTargetGroup.Android }, + Company = "Microsoft", + Desc = "Supports loading a glTF model for controllers.", + DocumentationLink = "https://aka.ms/openxr-unity", + CustomRuntimeLoaderBuildTargets = null, + OpenxrExtensionStrings = requestedExtensions, + Required = false, + Category = FeatureCategory.Feature, + FeatureId = featureId, + Version = "1.11.1")] +#endif + [RequiresNativePluginDLLs] + internal class MotionControllerFeaturePlugin : OpenXRFeaturePlugin + { + internal const string featureId = "com.microsoft.openxr.feature.controller"; + private const string requestedExtensions = "XR_MSFT_controller_model XR_FB_render_model"; + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MotionControllerFeaturePlugin.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MotionControllerFeaturePlugin.cs.meta new file mode 100644 index 0000000..2de9ca8 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/MotionControllerFeaturePlugin.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0c8f1ce8139888c4ab621f6b3c8bb558 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePlugin.cs b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePlugin.cs new file mode 100644 index 0000000..27158a3 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePlugin.cs @@ -0,0 +1,262 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using UnityEngine; +using UnityEngine.XR.OpenXR; +using UnityEngine.XR.OpenXR.Features; +using UnityEngine.XR.OpenXR.NativeTypes; + +namespace Microsoft.MixedReality.OpenXR +{ + internal abstract class OpenXRFeaturePlugin + : OpenXRFeaturePlugin where TPlugin : OpenXRFeaturePlugin + { + private static OpenXRFeature m_feature; + + internal static TPlugin Feature => (TPlugin)m_feature; + + protected override void OnEnable() + { + // Important notes about this Feature reference initialization: + // - Awake and OnEnable are called sequentially when this ScriptableObject is created, far earlier than when values such as the Feature singleton should be used. + // - ScriptableObject::OnEnable always runs when the feature is supported, independent of whether or not the enabled value defined in OpenXRFeature is true. + // - The reference must be refreshed in OnEnable, as Awake is not called on domain refresh (e.g. editing scripts while the editor is open). + // - References to OpenXR Project Settings are avoided, as they may not be available (e.g. while regenerating the project settings asset). + + var featureInCurrentPlatform = OpenXRSettings.Instance.GetFeature(); + if (featureInCurrentPlatform == this) + { + // C# scriptable object might be enabled for project settings in editor + // that's meant for other platforms, and won't be used when running the app. + // Only initialize those scriptable objects in the current platform. + // This field will be refreshed upon app domain refresh. + m_feature = this; + } + + base.OnEnable(); + } + } + + internal abstract class OpenXRFeaturePlugin + : OpenXRFeature, IOpenXRContext, ISubsystemPlugin + { + private List m_subsystemControllers = new List(); + + public ulong Instance { get; private set; } = 0; + public ulong SystemId { get; private set; } = 0; + public ulong Session { get; private set; } = 0; + public bool IsSessionRunning { get; private set; } = false; + public XrSessionState SessionState { get; private set; } = XrSessionState.Unknown; + public ulong SceneOriginSpace { get; private set; } = 0; + + public event OpenXRContextEvent InstanceCreated; // after instance is created + public event OpenXRContextEvent InstanceDestroying; // before instance is destroyed + public event OpenXRContextEvent SessionCreated; // after session is created + public event OpenXRContextEvent SessionDestroying; // before session is destroyed + public event OpenXRContextEvent SessionBegun; // after session is begun + public event OpenXRContextEvent SessionEnding; // before session is ended + + // Convert protected OpenXRFeature.xrGetInstanceProcAddr to internal visibility + internal static IntPtr PFN_xrGetInstanceProcAddr => OpenXRFeature.xrGetInstanceProcAddr; + + protected override void OnEnable() + { + base.OnEnable(); + OpenXRFeaturePluginManager.OnFeaturePluginInitializing(this); + + if (OpenXRFeaturePluginManager.NativeLibAvailable) + { + PluginEnvironmentSubsystem.InitializePlugin(); + } + } + + protected override IntPtr HookGetInstanceProcAddr(IntPtr func) + { + OpenXRFeaturePluginManager.InitializeOpenXRFeatureList(); + + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + func = NativeLib.HookGetInstanceProcAddr(func); + } + return func; + } + + protected void AddSubsystemController(SubsystemController subsystemController) + { + m_subsystemControllers.Add(subsystemController); + } + + protected override void OnSubsystemCreate() + { + m_subsystemControllers.ForEach(controller => controller.OnSubsystemCreate(this)); + } + + protected override void OnSubsystemStart() + { + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.OnSubsystemsStarting(); + } + m_subsystemControllers.ForEach(controller => controller.OnSubsystemStart(this)); + } + + protected override void OnSubsystemStop() + { + m_subsystemControllers.ForEach(controller => controller.OnSubsystemStop(this)); + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.OnSubsystemsStopped(); + } + } + + protected override void OnSubsystemDestroy() + { + m_subsystemControllers.ForEach(controller => controller.OnSubsystemDestroy(this)); + } + + protected override bool OnInstanceCreate(ulong instance) + { + if (Instance != 0) + { + Debug.LogWarning("New instance was created without properly destroying the previous one."); + } + + Instance = instance; + + string[] enabledExtensionNames = OpenXRRuntime.GetEnabledExtensions(); + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.OnInstanceCreated(instance, PFN_xrGetInstanceProcAddr, enabledExtensionNames, enabledExtensionNames.Length); + } + + InstanceCreated?.Invoke(this, EventArgs.Empty); + return true; + } + + protected override void OnInstanceDestroy(ulong instance) + { + if (Instance == 0) + { + // Unity might call destroy when instance handle was not successfully created + // Ignore such cases since there's no resources associated with instance of 0. + return; + } + + if (SystemId != 0) + { + // Unity's OnSystemChange event won't trigger when destroying instance. + // Reset resources associated with system before destroying the instance. + SystemId = 0; + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.SetXrSystemId(0); + } + } + + InstanceDestroying?.Invoke(this, EventArgs.Empty); + Instance = 0; + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.OnInstanceDestroyed(); + } + } + + protected override void OnSystemChange(ulong systemId) + { + SystemId = systemId; + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.SetXrSystemId(systemId); + } + } + + protected override void OnSessionCreate(ulong session) + { + Session = session; + PluginEnvironmentSubsystem.OnSessionCreated(); + + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.SetXrSession(session); + } + SessionCreated?.Invoke(this, EventArgs.Empty); + } + + protected override void OnSessionBegin(ulong session) + { + // This virtual function is called right after xrSessionBegin returns, + // All C# scripts should observe that the session is running. + IsSessionRunning = true; + + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.SetXrSessionRunning(true); + } + SessionBegun?.Invoke(this, EventArgs.Empty); + } + + protected override void OnSessionStateChange(int oldState, int newState) + { + SessionState = (XrSessionState)newState; + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.SetSessionState((uint)newState); + } + } + + protected override void OnSessionEnd(ulong session) + { + SessionEnding?.Invoke(this, EventArgs.Empty); + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.SetXrSessionRunning(false); + } + + // This virtual function is called right before xrSessionEnd is called. + // All C# scripts should still observe that the session is running until this point. + IsSessionRunning = false; + } + + protected override void OnSessionDestroy(ulong session) + { + SessionDestroying?.Invoke(this, EventArgs.Empty); + Session = 0; + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.SetXrSession(0); + } + } + + protected override void OnAppSpaceChange(ulong sceneOriginSpace) + { + SceneOriginSpace = sceneOriginSpace; + if (OpenXRFeaturePluginManager.IsResponsibleForNativeLib(this)) + { + NativeLib.SetSceneOriginSpace(sceneOriginSpace); + } + } + + // Convert protected function to internal + internal static new void SetEnvironmentBlendMode(XrEnvironmentBlendMode environmentBlendMode) + { + OpenXRFeature.SetEnvironmentBlendMode(environmentBlendMode); + } + + // Convert protected function to internal + internal static new XrEnvironmentBlendMode GetEnvironmentBlendMode() + { + return OpenXRFeature.GetEnvironmentBlendMode(); + } + + void ISubsystemPlugin.CreateSubsystem(List descriptors, string id) => + base.CreateSubsystem(descriptors, id); + + void ISubsystemPlugin.StartSubsystem() => base.StartSubsystem(); + + void ISubsystemPlugin.StopSubsystem() => base.StopSubsystem(); + + void ISubsystemPlugin.DestroySubsystem() => base.DestroySubsystem(); + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePlugin.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePlugin.cs.meta new file mode 100644 index 0000000..927811b --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePlugin.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4c0f56cf108f4ddabdb3436471ec9aff +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePluginManager.cs b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePluginManager.cs new file mode 100644 index 0000000..485b69d --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePluginManager.cs @@ -0,0 +1,72 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using UnityEngine.XR.OpenXR; + +namespace Microsoft.MixedReality.OpenXR +{ + [AttributeUsage(AttributeTargets.Class)] + public class RequiresNativePluginDLLsAttribute : Attribute { } + + // A utility for managing connections between Mixed Reality OpenXR feature plugins and the native plugin DLL. + internal static class OpenXRFeaturePluginManager + { + // If this is true, the NativeLib is guaranteed to be available. + // If this is false, the NativeLib may or may not be available. + internal static bool NativeLibAvailable { get => m_nativeLibAvailable; } + private static bool m_nativeLibAvailable = false; + + // Cache the result of all enabled OpenXRSettings.Instance.GetFeatures() in our package. + // The first element in the list will serve as the active feature for routing states to native library. + // This list is updated at query for PFN_xrGetInstanceProcAddr, which is openxr_loader first load the runtime. + private static List m_enabledPluginFeatures = new List(); + + // ActiveFeature is one of the enabled OpenXRFeature, when not null, it's in sync with OpenXR runtime status. + internal static OpenXRFeaturePlugin ActiveFeature => m_enabledPluginFeatures.Count > 0 ? m_enabledPluginFeatures[0] : null; + + // Because Unity's OpenXRFeature notifies child classes about OpenXR state changes + // always in a loop. reference: OpenXRFeature.cs, ReceiveNativeEvent() function. + // To update our nativeLib, only one of such sync function is enough. + // Choose the 1st enabled feature related to nativeLib as the responsible child class. + internal static bool IsResponsibleForNativeLib(OpenXRFeaturePlugin plugin) + { + return m_enabledPluginFeatures.Count > 0 ? m_enabledPluginFeatures[0] == plugin : false; + } + + internal static void OnFeaturePluginInitializing(OpenXRFeaturePlugin feature) + { + if (!m_nativeLibAvailable) + { + TryInitializeNativeLibAvailable(feature.GetType()); + } + } + + internal static void TryInitializeNativeLibAvailable(Type featureType) + { + if (!m_nativeLibAvailable) + { + var feature = OpenXRSettings.Instance.GetFeature(featureType); + if (feature != null && feature.enabled && Attribute.IsDefined(featureType, typeof(RequiresNativePluginDLLsAttribute))) + { + m_nativeLibAvailable = true; + } + } + } + + internal static void InitializeOpenXRFeatureList() + { + m_enabledPluginFeatures.Clear(); + var features = OpenXRSettings.Instance.GetFeatures(typeof(OpenXRFeaturePlugin)); + foreach (var pluginFeature in features) + { + if (pluginFeature != null && pluginFeature.enabled) + { + m_enabledPluginFeatures.Add((OpenXRFeaturePlugin)pluginFeature); + NativeLib.InitializePluginProviders(pluginFeature.GetType().Name); + } + } + } + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePluginManager.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePluginManager.cs.meta new file mode 100644 index 0000000..3cd9260 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/OpenXRFeaturePluginManager.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 88000497eee0bfa41a83d09e89862559 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/PlayModeRemotingPlugin.cs b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/PlayModeRemotingPlugin.cs new file mode 100644 index 0000000..4fa9d42 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/PlayModeRemotingPlugin.cs @@ -0,0 +1,206 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Globalization; +using System.IO; +using UnityEngine; + +#if UNITY_EDITOR +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; +using UnityEngine.XR.Management; +using UnityEngine.XR.OpenXR; +using UnityEngine.XR.OpenXR.Features; +#endif + +namespace Microsoft.MixedReality.OpenXR.Remoting +{ +#if UNITY_EDITOR + [OpenXRFeature(UiName = featureName, + BuildTargetGroups = new[] { BuildTargetGroup.Standalone }, + Company = "Microsoft", + Desc = featureName + " in Unity editor.", + DocumentationLink = "https://aka.ms/openxr-unity-editor-remoting", + OpenxrExtensionStrings = requestedExtensions, + Category = FeatureCategory.Feature, + Required = false, + Priority = -100, // hookup before other plugins so it affects json before GetProcAddr. + FeatureId = featureId, + Version = "1.11.1")] +#endif + internal class PlayModeRemotingPlugin : OpenXRFeaturePlugin +#if UNITY_EDITOR + , ISerializationCallbackReceiver +#endif + { + internal const string featureId = "com.microsoft.openxr.feature.playmoderemoting"; + internal const string featureName = "Holographic Remoting for Play Mode"; + private const string requestedExtensions = "XR_MSFT_holographic_remoting XR_MSFT_holographic_remoting_speech"; + private const string SettingsFileName = "MixedRealityOpenXRRemotingSettings.asset"; + private static string UserSettingsFolder => Path.Combine(Application.dataPath, "..", "UserSettings"); + private static string SettingsAssetPath => Path.Combine(UserSettingsFolder, SettingsFileName); + + [SerializeField, Tooltip("The host name or IP address of the player running in network server mode to connect to."), Obsolete("Use the remotingSettings values instead")] + private string m_remoteHostName = string.Empty; + + [SerializeField, Tooltip("The port number of the server's handshake port."), Obsolete("Use the remotingSettings values instead")] + private ushort m_remoteHostPort = 8265; + + [SerializeField, Tooltip("The max bitrate in Kbps to use for the connection."), Obsolete("Use the remotingSettings values instead")] + private uint m_maxBitrate = 20000; + + [SerializeField, Tooltip("The video codec to use for the connection."), Obsolete("Use the remotingSettings values instead")] + private RemotingVideoCodec m_videoCodec = RemotingVideoCodec.Auto; + + [SerializeField, Tooltip("Enable/disable audio remoting."), Obsolete("Use the remotingSettings values instead")] + private bool m_enableAudio = false; + + private readonly bool m_playModeRemotingIsActive = +#if UNITY_EDITOR + true; +#else + false; +#endif + private RemotingSettings m_remotingSettings; + + protected override IntPtr HookGetInstanceProcAddr(IntPtr func) + { + if (enabled && m_playModeRemotingIsActive) + { + AppRemotingSubsystem.GetCurrent().TryEnableRemotingOverride(); + } + return base.HookGetInstanceProcAddr(func); + } + + protected override void OnInstanceDestroy(ulong instance) + { + if (enabled && m_playModeRemotingIsActive) + { + AppRemotingSubsystem.GetCurrent().ResetRemotingOverride(); + } + base.OnInstanceDestroy(instance); + } + + protected override void OnSystemChange(ulong systemId) + { + base.OnSystemChange(systemId); + + if (systemId != 0 && m_playModeRemotingIsActive) + { + RemotingSettings remotingSettings = GetOrLoadRemotingSettings(); + AppRemotingSubsystem.GetCurrent().InitializePlayModeRemoting(new RemotingConnectConfiguration + { + RemoteHostName = remotingSettings.RemoteHostName, + RemotePort = remotingSettings.RemoteHostPort, + MaxBitrateKbps = remotingSettings.MaxBitrate, + VideoCodec = remotingSettings.VideoCodec, + EnableAudio = remotingSettings.EnableAudio, + AudioCaptureMode = remotingSettings.AudioCaptureMode, + secureConnectConfiguration = null + }); + } + } + + protected override void OnSessionStateChange(int oldState, int newState) + { + if (m_playModeRemotingIsActive && (XrSessionState)newState == XrSessionState.LossPending) + { + AppRemotingSubsystem.GetCurrent().OnSessionLossPending(); +#if UNITY_EDITOR + EditorApplication.ExitPlaymode(); +#endif + } + } + + internal RemotingSettings GetOrLoadRemotingSettings() + { + if (m_remotingSettings == null) + { + // If this file doesn't yet exist, create it and port from the old values. + m_remotingSettings = CreateInstance(); + + if (File.Exists(SettingsAssetPath)) + { + using (StreamReader settingsReader = new StreamReader(SettingsAssetPath)) + { + JsonUtility.FromJsonOverwrite(settingsReader.ReadToEnd(), m_remotingSettings); + } + } + else + { +#pragma warning disable CS0618 // to use the obsolete fields to port to the new asset file + m_remotingSettings.RemoteHostName = m_remoteHostName; + m_remotingSettings.RemoteHostPort = m_remoteHostPort; + m_remotingSettings.MaxBitrate = m_maxBitrate; + m_remotingSettings.VideoCodec = m_videoCodec; + m_remotingSettings.EnableAudio = m_enableAudio; +#pragma warning restore CS0618 + } + } + + return m_remotingSettings; + } + +#if UNITY_EDITOR + internal bool HasValidSettings() => !string.IsNullOrEmpty(GetOrLoadRemotingSettings().RemoteHostName); + + private void SaveSettings() + { + // Don't try to load the settings here. If this is null, then there's + // no need to do extra work to load and save the same file. + // When remoting is used, this is guaranteed to be non-null. + if (m_remotingSettings == null) + { + return; + } + + if (!Directory.Exists(UserSettingsFolder)) + { + Directory.CreateDirectory(UserSettingsFolder); + } + + using (StreamWriter settingsWriter = new StreamWriter(SettingsAssetPath)) + { + settingsWriter.Write(JsonUtility.ToJson(m_remotingSettings, true)); + } + } + + protected override void GetValidationChecks(System.Collections.Generic.List results, BuildTargetGroup targetGroup) + { + PlayModeRemotingValidator.GetValidationChecks(this, results); + } + + void ISerializationCallbackReceiver.OnBeforeSerialize() => SaveSettings(); + + void ISerializationCallbackReceiver.OnAfterDeserialize() { } // Can't call EnsureSettingsLoaded() here, since Application.dataPath can't be accessed during deserialization + + [InitializeOnEnterPlayMode] + private static void OnEnterPlaymodeInEditor(EnterPlayModeOptions options) + { + StartOrStopXRHelper.OnEnterPlaymodeInEditor(); + } +#endif + } + + internal class RemotingSettings : ScriptableObject + { + [field: SerializeField, Tooltip("The host name or IP address of the player running in network server mode to connect to.")] + public string RemoteHostName { get; set; } = string.Empty; + + [field: SerializeField, Tooltip("The port number of the server's handshake port.")] + public ushort RemoteHostPort { get; set; } = 8265; + + [field: SerializeField, Tooltip("The max bitrate in Kbps to use for the connection.")] + public uint MaxBitrate { get; set; } = 20000; + + [field: SerializeField, Tooltip("The video codec to use for the connection.")] + public RemotingVideoCodec VideoCodec { get; set; } = RemotingVideoCodec.Auto; + + [field: SerializeField, Tooltip("Enable/disable audio remoting.")] + public bool EnableAudio { get; set; } = false; + + [field: SerializeField, Tooltip("The audio capture mode to use for the connection.")] + public RemotingAudioCaptureMode AudioCaptureMode {get; set;} = RemotingAudioCaptureMode.SystemWideCapture; + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/PlayModeRemotingPlugin.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/PlayModeRemotingPlugin.cs.meta new file mode 100644 index 0000000..54c018d --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeaturePlugins/PlayModeRemotingPlugin.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 9f34c86d1a130cc45a438373e1e8a4fc +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators.meta b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators.meta new file mode 100644 index 0000000..2ba04c7 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 0ac685d9d590e2a479a9c24585822c91 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/AppRemotingValidator.cs b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/AppRemotingValidator.cs new file mode 100644 index 0000000..44e29dd --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/AppRemotingValidator.cs @@ -0,0 +1,89 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR + +using Microsoft.MixedReality.OpenXR.Remoting; +using System.Collections.Generic; +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; +using UnityEngine.XR.Management; +using UnityEngine.XR.OpenXR; +using UnityEngine.XR.OpenXR.Features; +using static UnityEngine.XR.OpenXR.Features.OpenXRFeature; + +namespace Microsoft.MixedReality.OpenXR +{ + internal class AppRemotingValidator + { + internal static void GetValidationChecks(OpenXRFeature feature, List results, BuildTargetGroup targetGroup) + { + results.Add(new ValidationRule(feature) + { + message = $"\"{AppRemotingPlugin.featureName}\" and \"Initialize XR on Startup\" are both enabled. XR initialization should be delayed until a specific IP address is entered.", + error = true, + checkPredicate = () => + { + // This validation rule is a fallback, in case no validation ruleset is selected. + // If a validation rule is selected, rules in PlatformValidation.cs will manage this setting. + if(ValidationSettings.CurrentRuleset != ValidationRuleset.None) + { + return true; + } + + XRGeneralSettings settings = XRSettingsHelpers.GetOrCreateXRGeneralSettings(targetGroup); + return settings != null && !settings.InitManagerOnStart; + }, + fixIt = () => + { + XRGeneralSettings settings = XRSettingsHelpers.GetOrCreateXRGeneralSettings(targetGroup); + if (settings != null) + { + settings.InitManagerOnStart = false; + } + } + }); + + if (targetGroup == BuildTargetGroup.WSA) + { + results.Add(new ValidationRule(feature) + { + message = "Required InternetClient capabilty in Unity PlayerSettings is not enabled for Holographic Application Remoting to work properly", + error = true, + checkPredicate = () => PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.InternetClient), + fixIt = () => + { + PlayerSettings.WSA.SetCapability(PlayerSettings.WSACapability.InternetClient, true); + } + }); + + results.Add(new ValidationRule(feature) + { + message = "Required InternetClientServer, PrivateNetworkClientServer capabilties in Unity PlayerSettings are not enabled for Holographic Application Remoting to work properly", + error = false, + checkPredicate = () => PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.InternetClientServer) && + PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.PrivateNetworkClientServer), + fixIt = () => + { + PlayerSettings.WSA.SetCapability(PlayerSettings.WSACapability.InternetClientServer, true); + PlayerSettings.WSA.SetCapability(PlayerSettings.WSACapability.PrivateNetworkClientServer, true); + } + }); + + results.Add(new ValidationRule(feature) + { + message = "Consider enabling Microphone capabilty in Unity PlayerSettings for Holographic Application Remoting Speech recognition to work properly", + error = false, + checkPredicate = () => PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.Microphone), + fixIt = () => + { + PlayerSettings.WSA.SetCapability(PlayerSettings.WSACapability.Microphone, true); + } + }); + } + + } + } +} + +#endif diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/AppRemotingValidator.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/AppRemotingValidator.cs.meta new file mode 100644 index 0000000..1568ed0 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/AppRemotingValidator.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 9f51b4ffc7bc20d48bab3923b1b0161c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/MixedRealityFeatureValidator.cs b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/MixedRealityFeatureValidator.cs new file mode 100644 index 0000000..0314687 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/MixedRealityFeatureValidator.cs @@ -0,0 +1,88 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR + +using System.Collections.Generic; +using System.Reflection; +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; +using UnityEngine.XR.OpenXR; +using UnityEngine.XR.OpenXR.Features; +using static UnityEngine.XR.OpenXR.Features.OpenXRFeature; + +namespace Microsoft.MixedReality.OpenXR +{ + internal class MixedRealityFeatureValidator + { + internal static void GetValidationChecks(OpenXRFeature feature, List results, BuildTargetGroup targetGroup) + { + if (targetGroup == BuildTargetGroup.WSA) + { + results.Add(new ValidationRule(feature) + { + message = "Windows Mixed Reality support may need the WebCam capability for the locatable camera feature.", + error = false, + checkPredicate = () => PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.WebCam), + fixIt = () => PlayerSettings.WSA.SetCapability(PlayerSettings.WSACapability.WebCam, true) + }); + results.Add(new ValidationRule(feature) + { + message = "Windows Mixed Reality support may need the SpatialPerception capability for plane detection.", + error = false, + checkPredicate = () => PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.SpatialPerception), + fixIt = () => PlayerSettings.WSA.SetCapability(PlayerSettings.WSACapability.SpatialPerception, true) + }); + } + + results.Add(new ValidationRule(feature) + { + message = "The Mixed Reality OpenXR package has been updated, and Unity must be restarted to complete the update.", + error = true, + errorEnteringPlaymode = true, + checkPredicate = () => + { + // MixedRealityFeaturePlugin caches its version when it's loaded into the editor. This validation rule checks to make + // sure the currently installed package matches the version we've cached. If they don't match, we can assume the + // Mixed Reality OpenXR Plugin package has been updated during this editor session, which can lead to stale native + // DLL references and a crash on the next time the play button is pressed. Triggering a restart here prevents the crash. + UnityEditor.PackageManager.PackageInfo packageInfo = UnityEditor.PackageManager.PackageInfo.FindForAssembly(typeof(MixedRealityFeaturePlugin).Assembly); + return packageInfo != null && MixedRealityFeaturePlugin.VersionInstalledOnLaunch == packageInfo.version; + }, + fixIt = RequireRestart + }); + +#if (UNITY_2020 && !UNITY_2020_3_43_OR_NEWER) || (UNITY_2021 && !UNITY_2021_3_16_OR_NEWER) || (UNITY_2022 && !UNITY_2022_2_1_OR_NEWER) || (UNITY_2023 && !UNITY_2023_1_0A23_OR_NEWER) + bool shouldRunInBackground = true; +#else + bool shouldRunInBackground = false; +#endif + results.Add(new ValidationRule(feature) + { + message = shouldRunInBackground + ? "\"Run in Background\" is necessary in this version of Unity for XR Unity apps to continue rendering when they have lost keyboard focus." + : "\"Run in Background\" is not necessary in this version of Unity and can add unwanted performance costs for XR Unity apps.", + error = false, + checkPredicate = () => PlayerSettings.runInBackground == shouldRunInBackground, // Note: The settings for "run in background" are connected for both standalone and UWP. + fixIt = () => PlayerSettings.runInBackground = shouldRunInBackground + }); + } + + private static void RequireRestart() + { + if (!EditorUtility.DisplayDialog("Unity editor restart required", "The Unity editor must be restarted for this change to take effect.", "Apply", "Cancel")) + { + return; + } + + RestartEditorAndRecompileScripts(); + } + + internal static void RestartEditorAndRecompileScripts() + { + typeof(EditorApplication).GetMethod("RestartEditorAndRecompileScripts", BindingFlags.NonPublic | BindingFlags.Static)?.Invoke(null, null); + } + } +} + +#endif diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/MixedRealityFeatureValidator.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/MixedRealityFeatureValidator.cs.meta new file mode 100644 index 0000000..1e522f5 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/MixedRealityFeatureValidator.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 380323a95b9e3f348b4f8a9a6a3213fb +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/PlayModeRemotingValidator.cs b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/PlayModeRemotingValidator.cs new file mode 100644 index 0000000..bc196cb --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/PlayModeRemotingValidator.cs @@ -0,0 +1,164 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR + +using Microsoft.MixedReality.OpenXR.Remoting; +using System.Collections.Generic; +using System.Linq; +using UnityEditor; +using UnityEditor.XR.Management.Metadata; +using UnityEditor.XR.OpenXR.Features; +using UnityEngine; +using UnityEngine.XR.Management; +using UnityEngine.XR.OpenXR; +using UnityEngine.XR.OpenXR.Features; +using UnityEngine.XR.OpenXR.Features.Interactions; +using static UnityEngine.XR.OpenXR.Features.OpenXRFeature; + +namespace Microsoft.MixedReality.OpenXR +{ + internal class PlayModeRemotingValidator + { + internal const string RemotingNotConfigured = "Using Holographic Remoting requires the Remote Host Name in settings " + + "to match the IP address displayed in the Holographic Remoting Player running on your HoloLens 2 device."; + + internal static readonly string DependenciesNotEnabled = "Using Holographic Remoting requires the following HoloLens features " + + "to be enabled in the `PC, Mac & Linux Standalone settings` tab, because the Unity editor runs as a standalone XR application. " + + "\n - Eye Gaze Interaction Profile" + + $"\n - {HandTrackingFeaturePlugin.featureName}" + + $"\n - {MixedRealityFeaturePlugin.featureName}" + + "\n - Microsoft Hand Interaction Profile"; + + internal const string OpenXRLoaderNotAssigned = "Using Holographic Remoting requires the OpenXR loader " + + "to be enabled in the `PC, Mac & Linux Standalone settings` tab, because the Unity editor runs as a standalone XR application."; + + internal const string PlayModeRemotingMenuPath = "Mixed Reality/Remoting/" + PlayModeRemotingPlugin.featureName; + internal const string PlayModeRemotingMenuPath2 = "Window/XR/" + PlayModeRemotingPlugin.featureName; + + internal const string CannotAutoConfigureRemoting = "Could not automatically apply recommended settings to enable " + PlayModeRemotingPlugin.featureName + + ". Please see https://aka.ms/openxr-unity-editor-remoting for manual set up instructions."; + + internal static void GetValidationChecks(OpenXRFeature feature, List results) + { + results.Add(new ValidationRule(feature) + { + message = DependenciesNotEnabled, + error = true, + checkPredicate = () => + { + return AreDependenciesEnabled(); + }, + fixIt = () => + { + EnableDependencies(); + } + }); + + results.Add(new ValidationRule(feature) + { + message = OpenXRLoaderNotAssigned, + error = true, + checkPredicate = () => + { + return IsLoaderAssigned(); + }, + fixIt = () => + { + AssignLoader(); + } + }); + + results.Add(new ValidationRule(feature) + { + message = RemotingNotConfigured, + error = true, + fixItAutomatic = false, + helpText = $"To open this feature's settings, click the \"Edit\" button here or click the settings icon to the right of the \"{PlayModeRemotingPlugin.featureName}\" feature in the XR Plug-in Management settings.", + checkPredicate = () => + { + FeatureHelpers.RefreshFeatures(BuildTargetGroup.Standalone); + PlayModeRemotingPlugin remotingFeature = OpenXRFeaturePlugin.Feature; + return remotingFeature != null && remotingFeature.HasValidSettings(); + }, + fixIt = () => + { + EditorApplication.ExecuteMenuItem(PlayModeRemotingMenuPath); + } + }); + } + + internal static bool IsLoaderAssigned() + { + XRManagerSettings standaloneManagerSettings = XRSettingsHelpers.GetOrCreateXRManagerSettings(BuildTargetGroup.Standalone); + return standaloneManagerSettings != null && + standaloneManagerSettings.activeLoaders.Any(l => l.GetType().Equals(typeof(OpenXRLoader))); + } + + internal static bool AreDependenciesEnabled() + { + FeatureHelpers.RefreshFeatures(BuildTargetGroup.Standalone); + OpenXRSettings openxrSettings = OpenXRSettings.Instance; + return openxrSettings != null && + IsFeatureEnabled(openxrSettings) && + IsFeatureEnabled(openxrSettings) && + IsFeatureEnabled(openxrSettings) && + IsFeatureEnabled(openxrSettings); + } + + internal static void AssignLoader() + { + // Workaround: when the XR Plug-in Management window is open, we cannot assign the loader properly + SettingsService.OpenProjectSettings("Project/Editor"); + XRManagerSettings standaloneManagerSettings = XRSettingsHelpers.GetOrCreateXRManagerSettings(BuildTargetGroup.Standalone); + if (standaloneManagerSettings == null || + !XRPackageMetadataStore.AssignLoader(standaloneManagerSettings, typeof(OpenXRLoader).Name, BuildTargetGroup.Standalone)) + { + SettingsService.OpenProjectSettings("Project/XR Plug-in Management"); + Debug.LogError(CannotAutoConfigureRemoting); + } + SettingsService.OpenProjectSettings("Project/XR Plug-in Management"); + } + + internal static void EnableDependencies() + { + var buildTarget = BuildTargetGroup.Standalone; + FeatureHelpers.RefreshFeatures(buildTarget); + OpenXRSettings openxrSettings = OpenXRSettings.Instance; + if (openxrSettings != null) + { + var featureSetId = "com.microsoft.openxr.featureset.wmr"; // must be same as in WMRFeatureSet.cs + var featureSet = OpenXRFeatureSetManager.GetFeatureSetWithId(buildTarget, featureSetId); + + featureSet.isEnabled = true; + OpenXRFeatureSetManager.SetFeaturesFromEnabledFeatureSets(buildTarget); + + EnableFeature(openxrSettings); + EnableFeature(openxrSettings); + EnableFeature(openxrSettings); + EnableFeature(openxrSettings); + } + else + { + Debug.LogError(CannotAutoConfigureRemoting); + } + } + + private static bool IsFeatureEnabled(OpenXRSettings openxrSettings) where T : OpenXRFeature + { + var feature = openxrSettings.GetFeature(); + return feature != null && feature.enabled; + } + + private static void EnableFeature(OpenXRSettings openxrSettings) where T : OpenXRFeature + { + var feature = openxrSettings.GetFeature(); + if (feature != null) + { + feature.enabled = true; + } + } + } +} + +#endif diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/PlayModeRemotingValidator.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/PlayModeRemotingValidator.cs.meta new file mode 100644 index 0000000..05570bf --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/PlayModeRemotingValidator.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6c6ab773bbe20c544853c8a9b8b3ed01 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/ValidationRuleset.cs b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/ValidationRuleset.cs new file mode 100644 index 0000000..3c2fd1b --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/ValidationRuleset.cs @@ -0,0 +1,188 @@ + +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR + +using Microsoft.MixedReality.OpenXR.Editor; +using Microsoft.MixedReality.OpenXR.Remoting; +using System; +using System.IO; +using UnityEditor; +using UnityEditor.XR.OpenXR.Features; +using UnityEngine; +using UnityEngine.XR.OpenXR.Features.Interactions; + +namespace Microsoft.MixedReality.OpenXR +{ + internal enum ValidationRuleset { + None, + Win32Standalone, + HoloLens2, + Win32AppRemoting, + UWPAppRemoting + } + + internal static class ValidationRulesetExtensions + { + // Example usage: $"This validation is for building {scenario name} apps" + internal static string GetScenarioName(this ValidationRuleset validationRuleset) + { + switch(validationRuleset) + { + case ValidationRuleset.Win32Standalone: return "Standalone Win32 XR"; + case ValidationRuleset.HoloLens2: return "HoloLens 2"; + case ValidationRuleset.Win32AppRemoting: return "Holographic Remoting Win32 Remote"; + case ValidationRuleset.UWPAppRemoting: return "Holographic Remoting UWP Remote"; + } + Debug.LogError($"ScenarioName of ValidationRuleset \"{validationRuleset}\" is not defined."); + return ""; + } + + // Example usage: $"loader must be enabled for {platformShortName}" + internal static string GetPlatformShortName(this ValidationRuleset validationRuleset) + { + switch(validationRuleset) + { + case ValidationRuleset.Win32Standalone: return "Standalone"; + case ValidationRuleset.HoloLens2: return "UWP"; + case ValidationRuleset.Win32AppRemoting: return "Standalone"; + case ValidationRuleset.UWPAppRemoting: return "UWP"; + } + Debug.LogError($"PlatformShortName of ValidationRuleset \"{validationRuleset}\" is not defined."); + return ""; + } + + internal static BuildTargetGroup GetBuildTargetGroup(this ValidationRuleset validationRuleset) + { + switch(validationRuleset) + { + case ValidationRuleset.Win32Standalone: return BuildTargetGroup.Standalone; + case ValidationRuleset.HoloLens2: return BuildTargetGroup.WSA; + case ValidationRuleset.Win32AppRemoting: return BuildTargetGroup.Standalone; + case ValidationRuleset.UWPAppRemoting: return BuildTargetGroup.WSA; + } + Debug.LogError($"BuildTargetGroup of ValidationRuleset \"{validationRuleset}\" is not defined."); + return BuildTargetGroup.Unknown; + } + + internal static BuildTarget GetBuildTarget(this ValidationRuleset validationRuleset) + { + switch(validationRuleset) + { + case ValidationRuleset.Win32Standalone: return BuildTarget.StandaloneWindows64; + case ValidationRuleset.HoloLens2: return BuildTarget.WSAPlayer; + case ValidationRuleset.Win32AppRemoting: return BuildTarget.StandaloneWindows64; + case ValidationRuleset.UWPAppRemoting: return BuildTarget.WSAPlayer; + } + Debug.LogError($"BuildTarget of ValidationRuleset \"{validationRuleset}\" is not defined."); + return BuildTarget.StandaloneWindows64; + } + + internal static bool GetRemotingEnabled(this ValidationRuleset validationRuleset) + { + switch (validationRuleset) + { + case ValidationRuleset.Win32Standalone: return false; + case ValidationRuleset.HoloLens2: return false; + case ValidationRuleset.Win32AppRemoting: return true; + case ValidationRuleset.UWPAppRemoting: return true; + } + Debug.LogError($"RemotingEnabled of ValidationRuleset \"{validationRuleset}\" is not defined."); + return false; + } + } + + internal static class ValidationSettings + { + private const string SettingsFileName = "MixedRealityOpenXRValidationSettings.asset"; + private static string XRSettingsFolder => Path.Combine(Application.dataPath, "..", "ProjectSettings"); + private static string SettingsAssetPath => Path.Combine(XRSettingsFolder, SettingsFileName); + + private class SerializableSettings : ScriptableObject { + [field: SerializeField] + internal ValidationRuleset Ruleset { get; set; } = ValidationRuleset.None; + } + + private static SerializableSettings settingsInstance; + + private static SerializableSettings CurrentSettings + { + get + { + if (settingsInstance == null) + { + LoadSettings(); + } + return settingsInstance; + } + } + + internal static ValidationRuleset CurrentRuleset + { + get => CurrentSettings.Ruleset; + set + { + if(CurrentSettings.Ruleset != value) + { + CurrentSettings.Ruleset = value; + SaveSettings(); + } + } + } + + private static void LoadSettings() + { + settingsInstance = ScriptableObject.CreateInstance(); + + if (File.Exists(SettingsAssetPath)) + { + using (StreamReader settingsReader = new StreamReader(SettingsAssetPath)) + { + JsonUtility.FromJsonOverwrite(settingsReader.ReadToEnd(), settingsInstance); + } + } + + // If this file doesn't yet exist, port the old value from MixedRealityFeaturePlugin. + else + { + MixedRealityFeaturePlugin plugin = BuildProcessorHelpers.GetOpenXRFeature(BuildTargetGroup.WSA, false); + if (plugin == null) + { + FeatureHelpers.RefreshFeatures(BuildTargetGroup.WSA); + plugin = BuildProcessorHelpers.GetOpenXRFeature(BuildTargetGroup.WSA, false); + } + + if (plugin != null && plugin.validationRuleTarget == MixedRealityFeaturePlugin.ValidationRuleTargetPlatform.HoloLens2) + { + settingsInstance.Ruleset = ValidationRuleset.HoloLens2; + } + else + { + settingsInstance.Ruleset = ValidationRuleset.None; + } + SaveSettings(); + } + } + + private static void SaveSettings() + { + if (settingsInstance == null) + { + return; + } + + if (!Directory.Exists(XRSettingsFolder)) + { + Directory.CreateDirectory(XRSettingsFolder); + } + + using (StreamWriter settingsWriter = new StreamWriter(SettingsAssetPath)) + { + settingsWriter.Write(JsonUtility.ToJson(settingsInstance, true)); + } + } + } +} + +#endif // UNITY_EDITOR \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/ValidationRuleset.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/ValidationRuleset.cs.meta new file mode 100644 index 0000000..53e4b88 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/FeatureValidators/ValidationRuleset.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 525a1aaacd724ef42a03b63295a615d0 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Microsoft.MixedReality.OpenXR.asmdef b/com.microsoft.mixedreality.openxr/Runtime/Microsoft.MixedReality.OpenXR.asmdef new file mode 100644 index 0000000..943bcac --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Microsoft.MixedReality.OpenXR.asmdef @@ -0,0 +1,69 @@ +{ + "name": "Microsoft.MixedReality.OpenXR", + "rootNamespace": "Microsoft.MixedReality.OpenXR", + "references": [ + "Unity.InputSystem", + "Unity.XR.ARFoundation", + "Unity.XR.ARSubsystems", + "Unity.XR.CoreUtils", + "Unity.XR.Management", + "Unity.XR.Management.Editor", + "Unity.XR.OpenXR", + "Unity.XR.OpenXR.Editor" + ], + "includePlatforms": [ + "Android", + "Editor", + "WSA", + "WindowsStandalone64" + ], + "excludePlatforms": [], + "allowUnsafeCode": true, + "overrideReferences": false, + "precompiledReferences": [], + "autoReferenced": true, + "defineConstraints": [], + "versionDefines": [ + { + "name": "com.unity.inputsystem", + "expression": "1.4.0", + "define": "INPUT_SYSTEM_1_4_OR_NEWER" + }, + { + "name": "com.unity.xr.arfoundation", + "expression": "5.0.0", + "define": "USE_ARFOUNDATION_5_OR_NEWER" + }, + { + "name": "Unity", + "expression": "2020.3.43f1", + "define": "UNITY_2020_3_43_OR_NEWER" + }, + { + "name": "Unity", + "expression": "2021.3.16f1", + "define": "UNITY_2021_3_16_OR_NEWER" + }, + { + "name": "Unity", + "expression": "2022.2.1f1", + "define": "UNITY_2022_2_1_OR_NEWER" + }, + { + "name": "Unity", + "expression": "2023.1.0a23", + "define": "UNITY_2023_1_0A23_OR_NEWER" + }, + { + "name": "Unity", + "expression": "2021.3.11f1", + "define": "UNITY_2021_3_11_OR_NEWER" + }, + { + "name": "Unity", + "expression": "2021.3.18f1", + "define": "UNITY_2021_3_18_OR_NEWER" + } + ], + "noEngineReferences": false +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/Microsoft.MixedReality.OpenXR.asmdef.meta b/com.microsoft.mixedreality.openxr/Runtime/Microsoft.MixedReality.OpenXR.asmdef.meta new file mode 100644 index 0000000..adf51a8 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Microsoft.MixedReality.OpenXR.asmdef.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: 3b76ab8242cc35d479edf0a949d56d59 +AssemblyDefinitionImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/NativeLib.cs b/com.microsoft.mixedreality.openxr/Runtime/NativeLib.cs new file mode 100644 index 0000000..d6ef23f --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/NativeLib.cs @@ -0,0 +1,458 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Runtime.InteropServices; +using System.Text; +using Microsoft.MixedReality.OpenXR.ARSubsystems; +using UnityEngine; +using UnityEngine.XR.ARSubsystems; + +namespace Microsoft.MixedReality.OpenXR +{ + [Flags] + internal enum NativeSpaceLocationFlags : uint + { + OrientationValid = 1, + PositionValid = 2, + OrientationTracked = 4, + PositionTracked = 8, + All = 15 + } + + [Serializable, StructLayout(LayoutKind.Sequential, Pack = 8)] + internal struct MixedRealityPluginOptions + { + private byte m_disableFirstPersonObserver; + private byte m_enablePoseUpdateOnBeforeRender; + + public bool DisableFirstPersonObserver + { + get { return m_disableFirstPersonObserver != 0; } + set { m_disableFirstPersonObserver = (byte)(value ? 1 : 0); } + } + public bool EnablePoseUpdateOnBeforeRender + { + get { return m_enablePoseUpdateOnBeforeRender != 0; } + set { m_enablePoseUpdateOnBeforeRender = (byte)(value ? 1 : 0); } + } + } + + [Serializable, StructLayout(LayoutKind.Sequential, Pack = 8)] + internal struct HandTrackingOptions + { + [SerializeField, Tooltip("The requested motion range for this hand.")] + private HandJointsMotionRange motionRange; + + public HandJointsMotionRange MotionRange + { + get => motionRange; + set => motionRange = value; + } + } + + // IL2CPP does not support marshaling delegates that do not have this attribute. + internal class MonoPInvokeCallbackAttribute : Attribute { public MonoPInvokeCallbackAttribute() { } } + + internal class NativeLib + { + internal const string DllName = "MicrosoftOpenXRPlugin"; + + // Configure Unity's IL2CPP compiler to process C# string (always UTF16) interop to C++ "const char*". + // Unity by default is converting to UTF8 when compiling IL2CPP code for pinvoke. + // Using the [MarshalAs(UnmanagedUTF8Type)] string to make this conversion more explicitly. + // UnmanagedType.LPUTF8Str is only defined in Net40, so for compatibility on NET20, use 48 instead. + internal const short UnmanagedUTF8Type = 48; // UnmanagedType.LPUTF8Str + + [DllImport(DllName, EntryPoint = "openxr_plugin_InitializePlugin")] + internal static extern void InitializePlugin(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_InitializePluginProviders")] + internal static extern void InitializePluginProviders([MarshalAs(UnmanagedUTF8Type)] string featureName); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetPluginEnvironment")] + internal static extern void SetPluginEnvironment(PluginEnvironment pluginEnvironment, [MarshalAs(UnmanagedUTF8Type)] string pluginInfo); + + [DllImport(DllName, EntryPoint = "openxr_plugin_OnSubsystemsStarting")] + internal static extern void OnSubsystemsStarting(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_OnSubsystemsStopped")] + internal static extern void OnSubsystemsStopped(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_HookGetInstanceProcAddr")] + internal static extern IntPtr HookGetInstanceProcAddr(IntPtr func); + + [DllImport(DllName, EntryPoint = "openxr_plugin_OnInstanceCreated")] + internal static extern void OnInstanceCreated(ulong instance, IntPtr xrGetInstanceProcAddr, string[] enabledExtensionNames, int enabledExtensionNamesCount); + + [DllImport(DllName, EntryPoint = "openxr_plugin_OnInstanceDestroyed")] + internal static extern void OnInstanceDestroyed(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetXrSystemId")] + internal static extern void SetXrSystemId(ulong systemId); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetXrSession")] + internal static extern void SetXrSession(ulong session); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetXrSessionRunning")] + internal static extern void SetXrSessionRunning([MarshalAs(UnmanagedType.U1)] bool running); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetXrSessionState")] + internal static extern void SetSessionState(uint sessionState); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetViewTrackingFlags")] + internal static extern NativeSpaceLocationFlags GetViewTrackingFlags(ViewConfigurationType viewConfigurationType); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetMixedRealityPluginOptions")] + internal static extern void SetMixedRealityPluginOptions(MixedRealityPluginOptions mixedRealityPluginOptions); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetSceneOriginSpace")] + internal static extern void SetSceneOriginSpace(ulong sceneOriginSpace); + + [DllImport(DllName, EntryPoint = "openxr_plugin_IsSelectKeywordFiltered")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool IsSelectKeywordFiltered(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetEnabledViewConfigurationTypesCount")] + internal static extern uint GetEnabledViewConfigurationTypesCount(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetEnabledViewConfigurationTypes")] + internal static extern void GetEnabledViewConfigurationTypes(ViewConfigurationType[] viewConfigurationTypes, uint viewConfigurationTypesCapacity); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetViewConfigurationIsActive")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool GetViewConfigurationIsActive(ViewConfigurationType viewConfigurationType); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetViewConfigurationIsPrimary")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool GetViewConfigurationIsPrimary(ViewConfigurationType viewConfigurationType); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetStereoSeparationAdjustment")] + internal static extern void SetStereoSeparationAdjustment(float stereoSeparationAdjustment); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetStereoSeparationAdjustment")] + internal static extern float GetStereoSeparationAdjustment(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetSupportedReprojectionModesCount")] + internal static extern uint GetSupportedReprojectionModesCount(ViewConfigurationType viewConfigurationType); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetSupportedReprojectionModes")] + internal static extern void GetSupportedReprojectionModes(ViewConfigurationType viewConfigurationType, ReprojectionMode[] reprojectionModes, uint reprojectionModesCapacity); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetReprojectionSettings")] + internal static extern void SetReprojectionSettings(ViewConfigurationType viewConfigurationType, NativeReprojectionSettings nativeReprojectionSettings); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetPrimaryViewTrackingState")] + internal static extern NativeSpaceLocationFlags GetPrimaryViewTrackingState(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_CreatePlaneProvider")] + internal static extern void CreatePlaneProvider(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_StartPlaneSubsystem")] + internal static extern void StartPlaneSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_StopPlaneSubsystem")] + internal static extern void StopPlaneSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_DestroyPlaneSubsystem")] + internal static extern void DestroyPlaneSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetPlaneDetectionMode")] + internal static extern void SetPlaneDetectionMode(PlaneDetectionMode planeDetectionMode); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetNumPlaneChanges")] + internal static extern void GetNumPlaneChanges(FrameTime frameTime, ref uint numAddedPlanes, ref uint numUpdatedPlanes, ref uint numRemovedPlanes); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetPlaneChanges")] + unsafe internal static extern void GetPlaneChanges(uint addedPlanesSize, void* addedPlanes, uint updatedPlanesSize, void* updatedPlanes, uint removedPlanesSize, void* removedPlanes); + + [DllImport(DllName, EntryPoint = "openxr_plugin_StartAnchorSubsystem")] + internal static extern void StartAnchorSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_StopAnchorSubsystem")] + internal static extern void StopAnchorSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_DestroyAnchorSubsystemPending")] + internal static extern void DestroyAnchorSubsystemPending(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_DestroyAnchorSubsystem")] + internal static extern void DestroyAnchorSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_RemoveAllAnchors")] + internal static extern void RemoveAllAnchors(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryAddAnchor")] + [return: MarshalAs(UnmanagedType.U1)] + unsafe internal static extern bool TryAddAnchor(FrameTime frameTime, Quaternion rotation, Vector3 position, void* anchorPtr); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryRemoveAnchor")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryRemoveAnchor(Guid anchorId); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetNumAnchorChanges")] + internal static extern void GetNumAnchorChanges(FrameTime frameTime, ref uint numAddedAnchors, ref uint numUpdatedAnchors, ref uint numRemovedAnchors); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetAnchorChanges")] + unsafe internal static extern void GetAnchorChanges(uint addedAnchorsSize, void* addedAnchors, uint updatedAnchorsSize, void* updatedAnchors, uint removedAnchorsSize, void* removedAnchors); + + [DllImport(DllName, EntryPoint = "openxr_plugin_LoadAnchorStore")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool LoadAnchorStore(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_ResetAnchorStore")] + internal static extern void ResetAnchorStore(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetAnchorCount")] + internal static extern uint GetAnchorCount(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetNumPersistedAnchorNames")] + internal static extern uint GetNumPersistedAnchorNames(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetPersistedAnchorName")] + internal static extern void GetPersistedAnchorName(uint idx, [MarshalAs(UnmanagedUTF8Type)] StringBuilder nameOut, uint capacity); + + [DllImport(DllName, EntryPoint = "openxr_plugin_LoadPersistedAnchor")] + internal static extern Guid LoadPersistedAnchor([MarshalAs(UnmanagedUTF8Type)] string name); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryPersistAnchor")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryPersistAnchor([MarshalAs(UnmanagedUTF8Type)] string name, Guid anchorId); + + [DllImport(DllName, EntryPoint = "openxr_plugin_UnpersistAnchor")] + internal static extern void UnpersistAnchor([MarshalAs(UnmanagedUTF8Type)] string name); + + [DllImport(DllName, EntryPoint = "openxr_plugin_ClearPersistedAnchors")] + internal static extern void ClearPersistedAnchors(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetHandJointsMotionRange")] + internal static extern HandJointsMotionRange GetHandJointsMotionRange(Handedness handedness); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetHandJointsMotionRange")] + internal static extern void SetHandJointsMotionRange(Handedness handedness, HandJointsMotionRange handTrackingOptions); + +#pragma warning disable CS0618 + [DllImport(DllName, EntryPoint = "openxr_plugin_TryGetHandJointData")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryGetHandJointData(Handedness handedness, FrameTime frameTime, + [MarshalAs(UnmanagedType.LPArray, SizeConst = HandTracker.JointCount)] HandJointLocation[] handJoints); +#pragma warning restore CS0618 + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryLocateHandMesh")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryLocateHandMesh(Handedness handedness, FrameTime frameTime, HandPoseType handPoseType, out Pose pose); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryGetHandMesh")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryGetHandMesh(Handedness handedness, FrameTime frameTime, HandPoseType handPoseType, + ref ulong vertexBufferKey, out uint vertexCount, Vector3[] vertexPositions, Vector3[] vertexNormals, + ref uint indexBufferKey, out uint indexCount, int[] indices); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryGetHandMeshBufferSizes")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryGetHandMeshBufferSizes(out uint maxVertexCount, out uint maxIndexCount); + + [DllImport(DllName, EntryPoint = "openxr_plugin_IsControllerModelSupported")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool IsControllerModelSupported(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryGetControllerModelKey")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryGetControllerModelKey(Handedness handedness, out ulong modelKey); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryGetControllerModel")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryGetControllerModel(ulong modelKey, uint bufferCapacityInput, out uint bufferCountOutput, byte[] modelBuffer = null); + +#pragma warning disable CS0618 + [DllImport(DllName, EntryPoint = "openxr_plugin_TryGetControllerModelProperties")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryGetControllerModelProperties(ulong key, uint nodeCapacityInput, out uint nodeCountOutput, [Out] ControllerModel.ControllerModelNodeProperties[] properties = null); +#pragma warning restore CS0618 + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryGetControllerModelState")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryGetControllerModelState(ulong key, uint nodeCapacityInput, out uint nodeCountOutput, Pose[] poses = null); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryEnableRemotingOverride")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryEnableRemotingOverride(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_ResetRemotingOverride")] + internal static extern void ResetRemotingOverride(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_ConnectRemoting")] + internal static extern void ConnectRemoting(Remoting.InternalRemotingConnectConfiguration configuration, [MarshalAs(UnmanagedType.U1)] bool secureConnect, + [MarshalAs(UnmanagedUTF8Type)] string authenticationToken, [MarshalAs(UnmanagedType.U1)] bool performSystemValidation, + [MarshalAs(UnmanagedType.FunctionPtr)] Remoting.InternalValidateServerCertificateDelegate validateServerCertificateCallback); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetNumCertificates")] + internal static extern uint GetNumCertificates(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetCertificate")] + internal static extern IntPtr GetCertificate(uint certIndex, out int size); + + [DllImport(DllName, EntryPoint = "openxr_plugin_ListenRemoting")] + unsafe internal static extern void ListenRemoting( + Remoting.InternalRemotingListenConfiguration listenConfiguration, + [MarshalAs(UnmanagedType.U1)] bool secureListen, + void* certificate, + uint certificateByteCount, + [MarshalAs(UnmanagedUTF8Type)] string subjectName, + [MarshalAs(UnmanagedUTF8Type)] string keyPassPhrase, + [MarshalAs(UnmanagedType.FunctionPtr)] Remoting.SecureRemotingValidateAuthenticationTokenDelegate validateAuthenticationTokenCallback); + + [DllImport(DllName, EntryPoint = "openxr_plugin_DisconnectRemoting")] + internal static extern void DisconnectRemoting(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryGetRemotingConnectionState")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryGetRemotingConnectionState(out Remoting.ConnectionState connectionState, out Remoting.DisconnectReason disconnectReason); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetRemoteSpeechCulture")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool SetRemoteSpeechCulture([MarshalAs(UnmanagedUTF8Type)] string cultureName); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryLocateUserReferenceSpace")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryLocateUserReferenceSpace(FrameTime frameTime, out Pose pose); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryConvertToRemoteTime")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryConvertToRemoteTime(long playerPerformanceCount, out long remotePerformanceCount); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryConvertToPlayerTime")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryConvertToPlayerTime(long remotePerformanceCount, out long playerPerformanceCount); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryCreateSpaceFromStaticNodeId")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryCreateSpaceFromStaticNodeId(Guid id, out ulong spaceId); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryCreateSpaceFromDynamicNodeId")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryCreateSpaceFromDynamicNodeId(Guid id, out ulong spaceId); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryLocateSpatialGraphNodeSpace")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryLocateSpatialGraphNodeSpace(ulong spaceId, FrameTime frameTime, out Pose pose); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryLocateSpatialGraphNodeSpaceWithQpcTime")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryLocateSpatialGraphNodeSpace(ulong spaceId, long qpcTime, out Pose pose); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryLocateViewSpace")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryLocateViewSpace(FrameTime frameTime, out Pose pose, out long time); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryAcquireSceneCoordinateSystem")] + internal static extern IntPtr TryAcquireSceneCoordinateSystem(Pose poseInScene); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryAcquirePerceptionSpatialAnchorByHandle")] + internal static extern IntPtr TryAcquirePerceptionSpatialAnchor(ulong anchorHandle); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryAcquirePerceptionSpatialAnchorById")] + internal static extern IntPtr TryAcquirePerceptionSpatialAnchor(Guid trackableId); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryCreateARAnchorFromOpenXRHandle")] + internal static extern Guid TryCreateARAnchorFromOpenXRHandle(ulong openxrAnchor); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryCreateARAnchorFromPerceptionAnchor")] + internal static extern Guid TryCreateARAnchorFromPerceptionAnchor([MarshalAs(UnmanagedType.IUnknown)] object perceptionAnchor); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryAcquireAndReplaceXrSpatialAnchor")] + internal static extern Guid TryAcquireAndReplaceXrSpatialAnchor([MarshalAs(UnmanagedType.IUnknown)] object perceptionAnchor, Guid id); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryGetPerceptionDeviceFactory")] + internal static extern IntPtr TryGetPerceptionDeviceFactory(IntPtr pfnGetInstanceProcAddr); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetMeshComputeSettings")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool SetMeshComputeSettings(MeshComputeSettings settings); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryCreateGestureRecognizer")] + internal static extern ulong TryCreateGestureRecognizer(GestureSettings settings); + + [DllImport(DllName, EntryPoint = "openxr_plugin_DestroyGestureRecognizer")] + internal static extern void DestroyGestureRecognizer(ulong gestureRecognizer); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TrySetGestureSettings")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TrySetGestureSettings(ulong gestureRecognizer, GestureSettings settings); + + [DllImport(DllName, EntryPoint = "openxr_plugin_CancelPendingGesture")] + internal static extern void CancelPendingGesture(ulong gestureRecognizer); + + [DllImport(DllName, EntryPoint = "openxr_plugin_TryGetNextEventData")] + [return: MarshalAs(UnmanagedType.U1)] + internal static extern bool TryGetNextEventData(ulong gestureRecognizer, ref GestureEventData eventData); + + [DllImport(DllName, EntryPoint = "openxr_plugin_StartGestureRecognizer")] + internal static extern void StartGestureRecognizer(ulong gestureRecognizer); + + [DllImport(DllName, EntryPoint = "openxr_plugin_StopGestureRecognizer")] + internal static extern void StopGestureRecognizer(ulong gestureRecognizer); + + [DllImport(DllName, EntryPoint = "openxr_plugin_StartMarkerSubsystem")] + internal static extern void StartMarkerSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_StopMarkerSubsystem")] + internal static extern void StopMarkerSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_DestroyMarkerSubsystem")] + internal static extern void DestroyMarkerSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_SetEnabledMarkerTypes")] + internal static extern void SetEnabledMarkerTypes(XrSceneMarkerTypeMSFT[] enabledMarkerTypes, int numMarkerTypes); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetNumMarkerChanges")] + internal static extern void GetNumMarkerChanges(FrameTime frameTime, ref uint numAddedMarkers, ref uint numUpdatedMarkers, ref uint numRemovedMarkers); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetMarkerChanges")] + unsafe internal static extern void GetMarkerChanges(uint addedMarkersSize, void* addedMarkers, uint updatedMarkersSize, void* updatedMarkers, uint removedMarkersSize, void* removedMarkers); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetMarkerRawDataSize")] + internal static extern uint GetMarkerRawDataSize(Guid markerId); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetMarkerRawData")] + unsafe internal static extern void GetMarkerRawData(Guid markerId, void* rawDataOut, uint capacity); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetMarkerDecodedStringLength")] + internal static extern uint GetMarkerDecodedStringLength(Guid markerId); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetMarkerDecodedString")] + internal static extern void GetMarkerDecodedString(Guid markerId, [MarshalAs(UnmanagedUTF8Type)] StringBuilder decodedStringOut, uint capacity); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetMarkerQRCodeProperties")] + unsafe internal static extern void GetMarkerQRCodeProperties(Guid markerId, NativeQRCodeProperties* qrCodePropertiesOut, uint capacity); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetCurrentQpcTimeAsXrTime")] + internal static extern long GetCurrentQpcTimeAsXrTime(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_ConvertXrTimeToQpcTime")] + internal static extern long ConvertXrTimeToQpcTime(long xrTime); + + [DllImport(DllName, EntryPoint = "openxr_plugin_GetPredictedDisplayTimeInXrTime")] + internal static extern long GetPredictedDisplayTimeInXrTime(FrameTime frameTime); + + [DllImport(DllName, EntryPoint = "openxr_plugin_StartMapTrackingSubsystem")] + internal static extern void StartMapTrackingSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_StopMapTrackingSubsystem")] + internal static extern void StopMapTrackingSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_DestroyMapTrackingSubsystem")] + internal static extern void DestroyMapTrackingSubsystem(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_MapTrackingManagerSupportsApplicationExclusiveMaps")] + internal static extern bool MapTrackingManagerSupportsApplicationExclusiveMaps(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_MapTrackingManagerGetActiveTrackingMapType")] + internal static extern int MapTrackingManagerGetActiveTrackingMapType(); + + [DllImport(DllName, EntryPoint = "openxr_plugin_MapTrackingManagerActivateExclusiveMap")] + internal static extern bool MapTrackingManagerActivateExclusiveMap(ref Guid mapId); + + [DllImport(DllName, EntryPoint = "openxr_plugin_MapTrackingManagerActivateSharedMap")] + internal static extern bool MapTrackingManagerActivateSharedMap(); + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/NativeLib.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/NativeLib.cs.meta new file mode 100644 index 0000000..8bc2471 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/NativeLib.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e0098eb2468121a4b81b97a5707b145f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems.meta new file mode 100644 index 0000000..d4bcffd --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: c8c99e6140f249542a3772c57b4559a4 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker.meta new file mode 100644 index 0000000..961564d --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: ea8ba54b83bd2b94f88550d78e3866a2 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker/MarkerSubsystem.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker/MarkerSubsystem.cs new file mode 100644 index 0000000..c20ead9 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker/MarkerSubsystem.cs @@ -0,0 +1,428 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Runtime.InteropServices; +using System.Text; +using Unity.Collections; +using Unity.Collections.LowLevel.Unsafe; +using UnityEngine; +using UnityEngine.XR.OpenXR; +using UnityEngine.XR.ARSubsystems; + +namespace Microsoft.MixedReality.OpenXR.ARSubsystems +{ + // Mapped to native XrSceneMarkerTypeMSFT + internal enum XrSceneMarkerTypeMSFT + { + XR_SCENE_MARKER_TYPE_QR_CODE_MSFT = 1 + } + + // Mapped to native XrSceneMarkerQRCodeSymbolTypeMSFT + internal enum XrSceneMarkerQRCodeSymbolTypeMSFT + { + XR_SCENE_MARKER_QR_CODE_SYMBOL_TYPE_QR_CODE_MSFT = 1, + XR_SCENE_MARKER_QR_CODE_SYMBOL_TYPE_MICRO_QR_CODE_MSFT = 2 + } + + [StructLayout(LayoutKind.Sequential, Pack = 8)] + internal struct NativeMarker + { + public Guid id; + public Vector3 position; + public Quaternion rotation; + public TrackingState trackingState; + public Vector2 center; + public Vector2 size; + public Int64 lastSeenTime; + public XrSceneMarkerTypeMSFT type; + } + + [StructLayout(LayoutKind.Sequential, Pack = 8)] + internal struct NativeQRCodeProperties + { + public XrSceneMarkerQRCodeSymbolTypeMSFT type; + public uint version; + } + + internal struct TimeOffsetInfo + { + public float lastOffsetCalculationTime; + public float offset; + } + + internal class MarkerSubsystem : XRMarkerSubsystem + { + public const string Id = "OpenXR marker tracking"; + + private class OpenXRProvider : Provider + { + private ARMarkerType[] m_enabledMarkerTypes = { ARMarkerType.QRCode }; + + private TransformMode m_defaultTransformMode = TransformMode.MostStable; + + private Dictionary m_Markers = new Dictionary(); + + private Dictionary m_PendingTransforms = new Dictionary(); + + private TimeOffsetInfo m_TimeOffsetInfo = new TimeOffsetInfo(); + + public OpenXRProvider() + { + } + + public override void Destroy() + { + NativeLib.DestroyMarkerSubsystem(); + } + + internal override ARMarkerType[] EnabledMarkerTypes + { + get => m_enabledMarkerTypes; + set + { + m_enabledMarkerTypes = value; + NativeLib.SetEnabledMarkerTypes(ToXrSceneMarkerTypeMSFT(m_enabledMarkerTypes), m_enabledMarkerTypes.Length); + } + } + + internal override TransformMode DefaultTransformMode + { + get => m_defaultTransformMode; + set => m_defaultTransformMode = value; + } + + public unsafe override TrackableChanges GetChanges(XRMarker defaultMarker, Allocator allocator) + { + float realTimeSinceStartup = Time.realtimeSinceStartup; + + // Fetching current QPC time if over a second has passed since it was fetched last + if (realTimeSinceStartup - m_TimeOffsetInfo.lastOffsetCalculationTime > 1) + { + m_TimeOffsetInfo.lastOffsetCalculationTime = realTimeSinceStartup; + long xrTime = NativeLib.GetCurrentQpcTimeAsXrTime(); + m_TimeOffsetInfo.offset = realTimeSinceStartup - (xrTime / (float)1e9); + } + + uint numAddedMarkers = 0; + uint numUpdatedMarkers = 0; + uint numRemovedMarkers = 0; + NativeLib.GetNumMarkerChanges(FrameTime.OnUpdate, ref numAddedMarkers, ref numUpdatedMarkers, ref numRemovedMarkers); + + using (var addedNativeMarkers = new NativeArray((int)numAddedMarkers, allocator, NativeArrayOptions.UninitializedMemory)) + using (var updatedNativeMarkers = new NativeArray((int)numUpdatedMarkers, allocator, NativeArrayOptions.UninitializedMemory)) + using (var removedNativeMarkers = new NativeArray((int)numRemovedMarkers, allocator, NativeArrayOptions.UninitializedMemory)) + { + if (numAddedMarkers + numUpdatedMarkers + numRemovedMarkers > 0) + { + NativeLib.GetMarkerChanges( + (uint)(numAddedMarkers * sizeof(NativeMarker)), + NativeArrayUnsafeUtility.GetUnsafePtr(addedNativeMarkers), + (uint)(numUpdatedMarkers * sizeof(NativeMarker)), + NativeArrayUnsafeUtility.GetUnsafePtr(updatedNativeMarkers), + (uint)(numRemovedMarkers * sizeof(Guid)), + NativeArrayUnsafeUtility.GetUnsafePtr(removedNativeMarkers)); + } + + var addedMarkers = HandleAddedMarkers(addedNativeMarkers); + var updatedMarkers = HandleUpdatedMarkers(updatedNativeMarkers); + var removedMarkers = HandleRemovedMarkers(removedNativeMarkers); + + // Handling transforms for markers that weren't added, updated or removed + if (m_PendingTransforms.Count > 0) + { + foreach (var trackableId in m_PendingTransforms.Keys.ToList()) + { + XRMarker xrMarker = m_Markers[trackableId]; + xrMarker.transformMode = m_PendingTransforms[trackableId]; + xrMarker = ApplyTransform(xrMarker); + + // Adding the marker to the updated list + updatedMarkers.Add(xrMarker); + m_Markers[trackableId] = xrMarker; + } + m_PendingTransforms.Clear(); + } + + // Handling tracking state for markers that were updated by the runtime but their last seen time is too old. + // These markers are already part of updatedMarkers list and so we need to go through them and change the + // tracking state in the list. + HashSet handledMarkers = new HashSet(); + for (int i = 0; i < updatedMarkers.Count; ++i) + { + handledMarkers.Add(updatedMarkers[i].trackableId); + if (IsLastSeenTimeTooOld(updatedMarkers[i])) + { + XRMarker xrMarker = updatedMarkers[i]; + xrMarker.trackingState = TrackingState.Limited; + updatedMarkers[i] = xrMarker; + m_Markers[updatedMarkers[i].trackableId] = xrMarker; + } + } + + // Handling tracking state for markers that were not updated by the runtime and their last seen time is too old. + // We ensure that the markers already part of the updatedMarkers list are not considered again. + foreach (var trackableId in m_Markers.Keys.ToList()) + { + if (!handledMarkers.Contains(trackableId)) + { + XRMarker xrMarker = m_Markers[trackableId]; + if (IsLastSeenTimeTooOld(xrMarker)) + { + xrMarker.trackingState = TrackingState.Limited; + updatedMarkers.Add(xrMarker); + m_Markers[trackableId] = xrMarker; + } + } + } + + return TrackableChanges.CopyFrom( + new NativeArray(addedMarkers.ToArray(), allocator), + new NativeArray(updatedMarkers.ToArray(), allocator), + new NativeArray(removedMarkers, allocator), + allocator); + } + } + + public override void SetTransformMode(TrackableId trackableId, TransformMode transformMode) + { + if (m_Markers.ContainsKey(trackableId) && m_Markers[trackableId].transformMode != transformMode) + { + // Adding transform as pending + m_PendingTransforms.Add(trackableId, transformMode); + } + } + + public unsafe override NativeArray GetRawData(TrackableId trackableId, Allocator allocator) + { + if (m_Markers.ContainsKey(trackableId)) + { + Guid guid = FeatureUtils.ToGuid(trackableId); + int rawDataSize = (int)NativeLib.GetMarkerRawDataSize(guid); + if (rawDataSize > 0) + { + NativeArray rawData = new NativeArray(rawDataSize, allocator, NativeArrayOptions.UninitializedMemory); + NativeLib.GetMarkerRawData(guid, NativeArrayUnsafeUtility.GetUnsafePtr(rawData), (uint)rawDataSize); + return rawData; + } + } + return new NativeArray(0, allocator, NativeArrayOptions.UninitializedMemory); + } + + public override string GetDecodedString(TrackableId trackableId) + { + if (m_Markers.ContainsKey(trackableId)) + { + Guid guid = FeatureUtils.ToGuid(trackableId); + int decodedStringLength = (int)NativeLib.GetMarkerDecodedStringLength(guid); + if (decodedStringLength > 0) + { + StringBuilder stringBuilder = new StringBuilder(decodedStringLength); + NativeLib.GetMarkerDecodedString(guid, stringBuilder, (uint)stringBuilder.Capacity); + return stringBuilder.ToString(); + } + } + return null; + } + + public override unsafe QRCodeProperties GetQRCodeProperties(TrackableId trackableId) + { + Guid guid = FeatureUtils.ToGuid(trackableId); + NativeQRCodeProperties nativeQRCodeProperties = new NativeQRCodeProperties(); + QRCodeProperties qrCodeProperties = new QRCodeProperties(); + NativeLib.GetMarkerQRCodeProperties(guid, &nativeQRCodeProperties, (uint)sizeof(NativeQRCodeProperties)); + qrCodeProperties.version = nativeQRCodeProperties.version; + qrCodeProperties.type = (QRCodeType)nativeQRCodeProperties.type; + return qrCodeProperties; + } + + public override void Start() + { + NativeLib.StartMarkerSubsystem(); + } + + public override void Stop() + { + NativeLib.StopMarkerSubsystem(); + } + + private List HandleAddedMarkers(NativeArray addedNativeMarkers) + { + var addedMarkers = new List(); + for (int i = 0; i < addedNativeMarkers.Length; ++i) + { + XRMarker xrMarker = ToXRMarker(addedNativeMarkers[i]); + if (xrMarker.transformMode == TransformMode.Center) + { + // If the default transform mode is center, we apply the transform here + xrMarker = ApplyCenterTransform(xrMarker); + } + m_Markers.Add(xrMarker.trackableId, xrMarker); + addedMarkers.Add(xrMarker); + } + return addedMarkers; + } + + private List HandleUpdatedMarkers(NativeArray updatedNativeMarkers) + { + var updatedMarkers = new List(); + for (int i = 0; i < updatedNativeMarkers.Length; ++i) + { + TrackableId updatedId = FeatureUtils.ToTrackableId(updatedNativeMarkers[i].id); + if (m_Markers.ContainsKey(updatedId)) + { + XRMarker xrMarker = m_Markers[updatedId]; + + Pose xrMarkerPose = xrMarker.pose; + xrMarkerPose.position = updatedNativeMarkers[i].position; + xrMarkerPose.rotation = updatedNativeMarkers[i].rotation; + xrMarker.pose = xrMarkerPose; + + xrMarker.center = updatedNativeMarkers[i].center; + xrMarker.size = updatedNativeMarkers[i].size; + xrMarker.lastSeenTime = GetLastSeenTimeAsRealTimeSinceStartup(updatedNativeMarkers[i].lastSeenTime); + xrMarker.trackingState = updatedNativeMarkers[i].trackingState; + + if (m_PendingTransforms.ContainsKey(updatedId)) + { + // Change transform mode if there is a pending transform + xrMarker.transformMode = m_PendingTransforms[updatedId]; + m_PendingTransforms.Remove(updatedId); + } + if (xrMarker.transformMode == TransformMode.Center) + { + // If the marker is supposed to be centered, we apply the transform here + xrMarker = ApplyCenterTransform(xrMarker); + } + + m_Markers[updatedId] = xrMarker; + updatedMarkers.Add(m_Markers[updatedId]); + } + } + return updatedMarkers; + } + + private TrackableId[] HandleRemovedMarkers(NativeArray removedNativeMarkers) + { + var removedMarkers = new TrackableId[removedNativeMarkers.Length]; + for (int i = 0; i < removedNativeMarkers.Length; ++i) + { + TrackableId removedId = FeatureUtils.ToTrackableId(removedNativeMarkers[i]); + if (m_Markers.ContainsKey(removedId)) + { + m_Markers.Remove(removedId); + } + if (m_PendingTransforms.ContainsKey(removedId)) + { + m_PendingTransforms.Remove(removedId); + } + removedMarkers[i] = removedId; + } + return removedMarkers; + } + + private XRMarker ApplyTransform(XRMarker xrMarker) + { + if (xrMarker.transformMode == TransformMode.Center) + { + return ApplyCenterTransform(xrMarker); + } + + return ApplyStableTransform(xrMarker); + } + + private XRMarker ApplyCenterTransform(XRMarker xrMarker) + { + if (xrMarker.transformMode == TransformMode.Center) + { + Pose newPose = xrMarker.pose; + newPose.position += xrMarker.center.x * newPose.right + xrMarker.center.y * newPose.up; + xrMarker.pose = newPose; + } + return xrMarker; + } + + private XRMarker ApplyStableTransform(XRMarker xrMarker) + { + if (xrMarker.transformMode == TransformMode.MostStable) + { + Pose newPose = xrMarker.pose; + newPose.position -= xrMarker.center.x * newPose.right + xrMarker.center.y * newPose.up; + xrMarker.pose = newPose; + } + return xrMarker; + } + + private XRMarker ToXRMarker(NativeMarker nativeMarker) + { + return new XRMarker( + FeatureUtils.ToTrackableId(nativeMarker.id), + new Pose(nativeMarker.position, nativeMarker.rotation), + nativeMarker.trackingState, + nativeMarker.center, + nativeMarker.size, + GetLastSeenTimeAsRealTimeSinceStartup(nativeMarker.lastSeenTime), + m_defaultTransformMode, + (ARMarkerType)nativeMarker.type, + IntPtr.Zero); + } + + private XrSceneMarkerTypeMSFT[] ToXrSceneMarkerTypeMSFT(ARMarkerType[] markerTypes) + { + var xrSceneMarkerTypeMSFTs = new XrSceneMarkerTypeMSFT[markerTypes.Length]; + for (int i = 0; i < markerTypes.Length; ++i) + { + xrSceneMarkerTypeMSFTs[i] = (XrSceneMarkerTypeMSFT)markerTypes[i]; + } + return xrSceneMarkerTypeMSFTs; + } + + private float GetLastSeenTimeAsRealTimeSinceStartup(long lastSeenTime) + { + return lastSeenTime / (float)1e9 + m_TimeOffsetInfo.offset; + } + + // We consider a marker to be too old if it hasn't been seen for more than 2 seconds. + // We choose the threshold based on a 99th percentile calculation of last seen times. + private bool IsLastSeenTimeTooOld(XRMarker xrMarker) + { + return (Time.realtimeSinceStartup - xrMarker.lastSeenTime) > 2 && xrMarker.trackingState == TrackingState.Tracking; + } + } + + [RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.SubsystemRegistration)] + static void RegisterDescriptor() + { + XRMarkerSubsystemDescriptor.Create(new XRMarkerSubsystemDescriptor.Cinfo + { + id = Id, + providerType = typeof(MarkerSubsystem.OpenXRProvider), + subsystemTypeOverride = typeof(MarkerSubsystem), + }); + } + }; + + internal class MarkerSubsystemController : SubsystemController + { + private static List s_MarkerDescriptors = new List(); + + public MarkerSubsystemController(IOpenXRContext context) : base(context) + { + } + + public override void OnSubsystemCreate(ISubsystemPlugin plugin) + { + if (OpenXRRuntime.IsExtensionEnabled("XR_MSFT_scene_marker")) + { + plugin.CreateSubsystem(s_MarkerDescriptors, MarkerSubsystem.Id); + } + } + + public override void OnSubsystemDestroy(ISubsystemPlugin plugin) + { + plugin.DestroySubsystem(); + } + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker/MarkerSubsystem.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker/MarkerSubsystem.cs.meta new file mode 100644 index 0000000..678a6be --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ARMarker/MarkerSubsystem.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 837a83ce181af7145973b36c082e70bf +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorStore.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorStore.cs new file mode 100644 index 0000000..42fd211 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorStore.cs @@ -0,0 +1,159 @@ +// 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.Feature; + private static Task m_anchorStoreLoadTask = null; + + private static ulong m_currentOpenxrSession; + public static Task LoadAnchorStoreAsync(XRAnchorSubsystem anchorSubsystem) + { + if (!(anchorSubsystem is AnchorSubsystem)) + { + Debug.LogWarning($"LoadAnchorStoreAsync: subsystem is not of type Microsoft.MixedReality.AnchorSubsystem. type: {anchorSubsystem.GetType()}"); + return Task.FromResult(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(null); + } + + if (OpenXRContext.Current.Session == 0) + { + Debug.LogWarning("LoadAnchorStoreAsync: Cannot load anchor store without a valid XR session."); + return Task.FromResult(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.Feature; + private List m_persistedAnchorNamesCache; + private bool m_persistedAnchorNamesCacheDirty = true; + private readonly object m_persistedAnchorNamesCacheLock = new object(); + + public IReadOnlyList 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(); + 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 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 diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorStore.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorStore.cs.meta new file mode 100644 index 0000000..672d212 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorStore.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 77e495bc41dbd2149a57eda2544bf4f3 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorSubsystem.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorSubsystem.cs new file mode 100644 index 0000000..4869acb --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorSubsystem.cs @@ -0,0 +1,176 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using System.Runtime.InteropServices; +using Unity.Collections; +using Unity.Collections.LowLevel.Unsafe; +using UnityEngine; +using UnityEngine.XR.ARSubsystems; +using UnityEngine.XR.OpenXR; + +namespace Microsoft.MixedReality.OpenXR +{ + [StructLayout(LayoutKind.Sequential, Pack = 8)] + internal struct NativeAnchorData + { + public uint version; // == 1 + public ulong anchorHandle; // OpenXR XrSpatialAnchor handle + } + + [StructLayout(LayoutKind.Sequential, Pack = 8)] + internal struct NativeAnchor + { + public Guid id; + public Pose pose; + public TrackingState trackingState; + public IntPtr nativePtr; // pointer to NativeAnchorData + } + + internal class AnchorSubsystem : XRAnchorSubsystem + { + public const string Id = "OpenXR Anchors Subsystem"; + + private class OpenXRProvider : Provider + { + public OpenXRProvider() + { + } + + public override void Start() + { + NativeLib.StartAnchorSubsystem(); + } + + public override void Stop() + { + NativeLib.StopAnchorSubsystem(); + } + + public override void Destroy() + { + // If the anchor subsystem is destroyed, transient anchor data will be cleared, so the next time the + // subsystem is created, it will have a fresh new set of anchors. To preserve anchors, the app must use + // anchor persistence through the XRAnchorStore, or keep this subsystem alive. + NativeLib.DestroyAnchorSubsystem(); + } + + public unsafe override TrackableChanges GetChanges(XRAnchor defaultAnchor, Allocator allocator) + { + uint numAddedAnchors = 0; + uint numUpdatedAnchors = 0; + uint numRemovedAnchors = 0; + NativeLib.GetNumAnchorChanges(FrameTime.OnUpdate, ref numAddedAnchors, ref numUpdatedAnchors, ref numRemovedAnchors); + + using (var addedNativeAnchors = new NativeArray((int)numAddedAnchors, allocator, NativeArrayOptions.UninitializedMemory)) + using (var updatedNativeAnchors = new NativeArray((int)numUpdatedAnchors, allocator, NativeArrayOptions.UninitializedMemory)) + using (var removedNativeAnchors = new NativeArray((int)numRemovedAnchors, allocator, NativeArrayOptions.UninitializedMemory)) + { + if (numAddedAnchors + numUpdatedAnchors + numRemovedAnchors > 0) + { + NativeLib.GetAnchorChanges( + (uint)(numAddedAnchors * sizeof(NativeAnchor)), + NativeArrayUnsafeUtility.GetUnsafePtr(addedNativeAnchors), + (uint)(numUpdatedAnchors * sizeof(NativeAnchor)), + NativeArrayUnsafeUtility.GetUnsafePtr(updatedNativeAnchors), + (uint)(numRemovedAnchors * sizeof(Guid)), + NativeArrayUnsafeUtility.GetUnsafePtr(removedNativeAnchors)); + } + + // Added Anchors + var addedAnchors = Array.Empty(); + if (numAddedAnchors > 0) + { + addedAnchors = new XRAnchor[numAddedAnchors]; + for (int i = 0; i < numAddedAnchors; ++i) + addedAnchors[i] = ToXRAnchor(addedNativeAnchors[i]); + } + + // Updated Anchors + var updatedAnchors = Array.Empty(); + if (numUpdatedAnchors > 0) + { + updatedAnchors = new XRAnchor[numUpdatedAnchors]; + for (int i = 0; i < numUpdatedAnchors; ++i) + updatedAnchors[i] = ToXRAnchor(updatedNativeAnchors[i]); + } + + // Removed Anchors + var removedAnchors = Array.Empty(); + if (numRemovedAnchors > 0) + { + removedAnchors = new TrackableId[numRemovedAnchors]; + for (int i = 0; i < numRemovedAnchors; ++i) + removedAnchors[i] = FeatureUtils.ToTrackableId(removedNativeAnchors[i]); + } + + TrackableChanges trackableChanges = TrackableChanges.CopyFrom( + new NativeArray(addedAnchors, allocator), + new NativeArray(updatedAnchors, allocator), + new NativeArray(removedAnchors, allocator), + allocator); + return trackableChanges; + } + } + + private XRAnchor ToXRAnchor(NativeAnchor nativeAnchor) + { + var anchorId = FeatureUtils.ToTrackableId(nativeAnchor.id); + return new XRAnchor(anchorId, nativeAnchor.pose, nativeAnchor.trackingState, nativeAnchor.nativePtr); + } + + unsafe public override bool TryAddAnchor(Pose pose, out XRAnchor anchor) + { + NativeAnchor nativeAnchor = new NativeAnchor(); + bool succeeded = NativeLib.TryAddAnchor(FrameTime.OnUpdate, pose.rotation, pose.position, UnsafeUtility.AddressOf(ref nativeAnchor)); + anchor = ToXRAnchor(nativeAnchor); + return succeeded; + } + + public override bool TryAttachAnchor(TrackableId trackableToAffix, Pose pose, out XRAnchor anchor) + { + return TryAddAnchor(pose, out anchor); + } + + public override bool TryRemoveAnchor(TrackableId anchorId) + { + return NativeLib.TryRemoveAnchor(FeatureUtils.ToGuid(anchorId)); + } + } + + [RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.SubsystemRegistration)] + private static void RegisterDescriptor() + { + XRAnchorSubsystemDescriptor.Create(new XRAnchorSubsystemDescriptor.Cinfo + { + id = Id, + providerType = typeof(AnchorSubsystem.OpenXRProvider), + subsystemTypeOverride = typeof(AnchorSubsystem), + supportsTrackableAttachments = false + }); + } + }; + + internal class AnchorSubsystemController : SubsystemController + { + private static List s_AnchorDescriptors = new List(); + + public AnchorSubsystemController(IOpenXRContext context) : base(context) + { + } + + public override void OnSubsystemCreate(ISubsystemPlugin plugin) + { + if (OpenXRRuntime.IsExtensionEnabled("XR_MSFT_spatial_anchor")) + { + plugin.CreateSubsystem(s_AnchorDescriptors, AnchorSubsystem.Id); + } + } + + public override void OnSubsystemDestroy(ISubsystemPlugin plugin) + { + plugin.DestroySubsystem(); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorSubsystem.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorSubsystem.cs.meta new file mode 100644 index 0000000..af2a7a7 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorSubsystem.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: ded64ca71988f5243a4649034a87ae6a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorTransferBatch.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorTransferBatch.cs new file mode 100644 index 0000000..1c93003 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorTransferBatch.cs @@ -0,0 +1,171 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using System.IO; +using System.Threading.Tasks; +using UnityEngine.XR.ARSubsystems; + +#if WINDOWS_UWP +using UnityEngine; +using Windows.Perception.Spatial; +#endif // WINDOWS_UWP + +namespace Microsoft.MixedReality.OpenXR +{ + internal enum SerializationCompletionReason + { + Succeeded = 0, + NotSupported = 1, + AccessDenied = 2, + UnknownError = 3 + } + + internal class AnchorTransferBatch + { +#if WINDOWS_UWP + private readonly List m_anchorIds = new List(); + private Dictionary m_spatialAnchors = null; +#endif // WINDOWS_UWP + + public IReadOnlyList AnchorNames => +#if WINDOWS_UWP + m_anchorIds; +#else + Array.Empty(); +#endif // WINDOWS_UWP + + public bool AddAnchor(TrackableId trackableId, string name) + { +#if WINDOWS_UWP + m_spatialAnchors ??= new Dictionary(); + + if (!m_anchorIds.Contains(name) + && AnchorConverter.ToPerceptionSpatialAnchor(trackableId) is SpatialAnchor spatialAnchor + && spatialAnchor != null) + { + m_anchorIds.Add(name); + m_spatialAnchors.Add(name, spatialAnchor); + return true; + } +#endif // WINDOWS_UWP + + return false; + } + + public void RemoveAnchor(string name) + { +#if WINDOWS_UWP + m_anchorIds?.Remove(name); + m_spatialAnchors?.Remove(name); +#endif // WINDOWS_UWP + } + + public void Clear() + { +#if WINDOWS_UWP + m_anchorIds?.Clear(); + m_spatialAnchors?.Clear(); +#endif // WINDOWS_UWP + } + + public TrackableId LoadAnchor(string name) + { +#if WINDOWS_UWP + if (m_spatialAnchors?.Count == 0) + { + Debug.LogWarning($"No anchors have been imported yet. Call {nameof(ImportAsync)} before calling {nameof(LoadAnchor)}"); + return TrackableId.invalidId; + } + + if (m_spatialAnchors.TryGetValue(name, out SpatialAnchor spatialAnchor)) + { + return AnchorConverter.CreateFromPerceptionSpatialAnchor(spatialAnchor); + } +#endif // WINDOWS_UWP + + return TrackableId.invalidId; + } + + public TrackableId LoadAndReplaceAnchor(string name, TrackableId trackableId) + { +#if WINDOWS_UWP + if (m_spatialAnchors?.Count == 0) + { + Debug.LogWarning($"No anchors have been imported yet. Call {nameof(ImportAsync)} before calling {nameof(LoadAndReplaceAnchor)}"); + return TrackableId.invalidId; + } + + if (m_spatialAnchors.TryGetValue(name, out SpatialAnchor spatialAnchor)) + { + return AnchorConverter.ReplaceSpatialAnchor(spatialAnchor, trackableId); + } +#endif // WINDOWS_UWP + + return TrackableId.invalidId; + } + + public async Task ExportAsync(Stream output) + { +#if WINDOWS_UWP +#pragma warning disable CS0618 // Turn this off, so we can use the deprecated SpatialAnchorTransferManager + SpatialPerceptionAccessStatus access = await SpatialAnchorTransferManager.RequestAccessAsync(); + if (access != SpatialPerceptionAccessStatus.Allowed) + { + Debug.LogError($"{nameof(SpatialAnchorTransferManager)} access not granted: {access}"); + return SerializationCompletionReason.AccessDenied; + } + + if (m_spatialAnchors?.Count == 0) + { + Debug.LogError("No anchors to export!"); + return SerializationCompletionReason.UnknownError; + } + else + { + bool success = await SpatialAnchorTransferManager.TryExportAnchorsAsync(m_spatialAnchors, output.AsOutputStream()); + return success ? SerializationCompletionReason.Succeeded : SerializationCompletionReason.UnknownError; + } +#pragma warning restore CS0618 +#else + await Task.CompletedTask; + return SerializationCompletionReason.NotSupported; +#endif // WINDOWS_UWP + } + + public async Task ImportAsync(Stream input) + { +#if WINDOWS_UWP +#pragma warning disable CS0618 // Turn this off, so we can use the deprecated SpatialAnchorTransferManager + SpatialPerceptionAccessStatus access = await SpatialAnchorTransferManager.RequestAccessAsync(); + if (access != SpatialPerceptionAccessStatus.Allowed) + { + Debug.LogError($"{nameof(SpatialAnchorTransferManager)} access not granted: {access}"); + return SerializationCompletionReason.AccessDenied; + } + + IReadOnlyDictionary importedAnchors = await SpatialAnchorTransferManager.TryImportAnchorsAsync(input.AsInputStream()); + if (importedAnchors != null) + { + m_spatialAnchors = new Dictionary(importedAnchors.Count); + foreach (KeyValuePair anchor in importedAnchors) + { + m_spatialAnchors.Add(anchor.Key, anchor.Value); + } + + m_anchorIds.AddRange(m_spatialAnchors.Keys); + return SerializationCompletionReason.Succeeded; + } + else + { + return SerializationCompletionReason.UnknownError; + } +#pragma warning restore CS0618 +#else + await Task.CompletedTask; + return SerializationCompletionReason.NotSupported; +#endif // WINDOWS_UWP + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorTransferBatch.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorTransferBatch.cs.meta new file mode 100644 index 0000000..fd298e8 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AnchorTransferBatch.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: dc494c55ea2d0394ca8ed6f6e121099b +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingCoroutineRunner.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingCoroutineRunner.cs new file mode 100644 index 0000000..3b577b4 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingCoroutineRunner.cs @@ -0,0 +1,43 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR.Remoting +{ + internal class AppRemotingCoroutineRunner : MonoBehaviour + { + private static AppRemotingCoroutineRunner m_instance; + protected static AppRemotingCoroutineRunner Instance + { + get + { + if (m_instance == null) + { + SetupInstance(); + } + return m_instance; + } + } + + private static void SetupInstance() + { + GameObject gameObject = new GameObject("AppRemotingCoroutineRunner", typeof(AppRemotingCoroutineRunner)) + { + hideFlags = HideFlags.HideAndDontSave + }; + DontDestroyOnLoad(gameObject); + m_instance = gameObject.GetComponent(); + } + + // Starts a coroutine on this hidden, persistent GameObject, then returns a Coroutine which + // can be used to observe the progress of the internal routine, if desired. The internal + // routine passed into this method will always run to completion. + internal static Coroutine Start(IEnumerator internalRoutine) + { + return Instance.StartCoroutine(internalRoutine); + } + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingCoroutineRunner.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingCoroutineRunner.cs.meta new file mode 100644 index 0000000..a5b8f5a --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingCoroutineRunner.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c3270d43063ff474db769c99cefa531d +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingSubsystem.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingSubsystem.cs new file mode 100644 index 0000000..ab87c42 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingSubsystem.cs @@ -0,0 +1,593 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Globalization; +using System.Runtime.InteropServices; +using System.Security.Cryptography.X509Certificates; +using UnityEngine; +using UnityEngine.XR.Management; + +#if UNITY_EDITOR +using UnityEditor; +#endif + +namespace Microsoft.MixedReality.OpenXR.Remoting +{ + internal class AppRemotingSubsystem + { + private static AppRemotingSubsystem m_instance = new AppRemotingSubsystem(); + private bool m_runtimeOverrideAttempted = false; + private static RemotingState s_remotingState; + internal static RemotingState AppRemotingState + { + get { return s_remotingState; } + } + + private RemotingConnectConfiguration m_remotingConnectConfiguration; + private static SecureRemotingConnectConfiguration s_secureRemotingConnectConfiguration = default; + private static InternalValidateServerCertificateDelegate s_internalValidateServerCertificateCallback = null; + private DisconnectReason m_disconnectReasonOnLossPending = DisconnectReason.None; + + private static ListenMode s_listenMode; + private RemotingListenConfiguration m_remotingListenConfiguration; + private static SecureRemotingListenConfiguration s_secureRemotingListenConfiguration = default; + private static SecureRemotingValidateAuthenticationTokenDelegate s_validateAuthenticationTokenCallback = null; + + internal static AppRemotingSubsystem GetCurrent() + { + return m_instance; + } + + internal static bool UseSystemRuntime { get; set; } = false; + + internal bool IsAppRemotingEnabled() + { + return OpenXRFeaturePlugin.Feature.IsValidAndEnabled(); + } + + internal bool InPlayModeRemoting() + { +#if UNITY_EDITOR + return (OpenXRFeaturePlugin.Feature.IsValidAndEnabled() && EditorApplication.isPlaying); +#else + return false; +#endif + } + + internal bool IsReadyToStart() + { + return IsAppRemotingEnabled() && !InPlayModeRemoting() && s_remotingState == RemotingState.Idle; + } + + internal bool TryGetConnectionState(out ConnectionState connectionState, out DisconnectReason disconnectReason) + { + return NativeLib.TryGetRemotingConnectionState(out connectionState, out disconnectReason); + } + + internal bool TryLocateUserReferenceSpace(FrameTime frameTime, out Pose pose) + { + return NativeLib.TryLocateUserReferenceSpace(frameTime, out pose); + } + + internal bool TryConvertToRemoteTime(long playerPerformanceCount, out long remotePerformanceCount) + { + return NativeLib.TryConvertToRemoteTime(playerPerformanceCount, out remotePerformanceCount); + } + + internal bool TryConvertToPlayerTime(long remotePerformanceCount, out long playerPerformanceCount) + { + return NativeLib.TryConvertToPlayerTime(remotePerformanceCount, out playerPerformanceCount); + } + + internal bool TryEnableRemotingOverride() + { + if (!m_runtimeOverrideAttempted && !UseSystemRuntime) + { + m_runtimeOverrideAttempted = true; + if (NativeLib.TryEnableRemotingOverride()) + { + return true; + } + } + return false; + } + + internal void ResetRemotingOverride() + { + if (m_runtimeOverrideAttempted) + { + m_runtimeOverrideAttempted = false; + NativeLib.ResetRemotingOverride(); + } + } + + internal unsafe void InitializeRemoting() + { + bool secureConnect = false, secureListen = false; + NativeLib.SetRemoteSpeechCulture(CultureInfo.CurrentCulture.Name); + + if (s_remotingState == RemotingState.Connect) + { + Debug.Log($"[AppRemotingSubsystem] Initializing Remoting Connect"); + if (m_remotingConnectConfiguration.secureConnectConfiguration != null) + { + secureConnect = true; + s_secureRemotingConnectConfiguration = m_remotingConnectConfiguration.secureConnectConfiguration.Value; + if (s_secureRemotingConnectConfiguration.ValidateServerCertificateCallback != null) + { + s_internalValidateServerCertificateCallback = new InternalValidateServerCertificateDelegate(ImplementValidateServerCertificate); + } + } + + InternalRemotingConnectConfiguration remotingConnectConfiguration; + remotingConnectConfiguration.RemoteHostName = m_remotingConnectConfiguration.RemoteHostName; + remotingConnectConfiguration.RemotePort = m_remotingConnectConfiguration.RemotePort; + remotingConnectConfiguration.MaxBitrateKbps = m_remotingConnectConfiguration.MaxBitrateKbps; + remotingConnectConfiguration.VideoCodec = m_remotingConnectConfiguration.VideoCodec; + remotingConnectConfiguration.EnableAudio = m_remotingConnectConfiguration.EnableAudio; + remotingConnectConfiguration.AudioCaptureMode = m_remotingConnectConfiguration.AudioCaptureMode; + + // The following method is used for both secure and non-secure Connect in native layer. + // The secure mode parameters are used in native layer only when secureConnect + // is set to true, otherwise they are disregarded. + NativeLib.ConnectRemoting( + remotingConnectConfiguration, + secureConnect, + s_secureRemotingConnectConfiguration.AuthenticationToken, + s_secureRemotingConnectConfiguration.PerformSystemValidation, + s_internalValidateServerCertificateCallback); + } + else if (s_remotingState == RemotingState.Listen) + { + Debug.Log($"[AppRemotingSubsystem] Initializing Remoting Listen"); + if (m_remotingListenConfiguration.secureListenConfiguration != null) + { + secureListen = true; + s_secureRemotingListenConfiguration = m_remotingListenConfiguration.secureListenConfiguration.Value; + if (s_secureRemotingListenConfiguration.ValidateAuthenticationTokenCallback != null) + { + s_validateAuthenticationTokenCallback = new SecureRemotingValidateAuthenticationTokenDelegate(ImplementValidateAuthenticationToken); + } + } + + InternalRemotingListenConfiguration remotingListenConfiguration; + remotingListenConfiguration.ListenInterface = m_remotingListenConfiguration.ListenInterface; + remotingListenConfiguration.HandshakeListenPort = m_remotingListenConfiguration.HandshakeListenPort; + remotingListenConfiguration.TransportListenPort = m_remotingListenConfiguration.TransportListenPort; + remotingListenConfiguration.MaxBitrateKbps = m_remotingListenConfiguration.MaxBitrateKbps; + remotingListenConfiguration.VideoCodec = m_remotingListenConfiguration.VideoCodec; + remotingListenConfiguration.EnableAudio = m_remotingListenConfiguration.EnableAudio; + remotingListenConfiguration.AudioCaptureMode = m_remotingListenConfiguration.AudioCaptureMode; + + if (secureListen) + { + NativeLib.ListenRemoting( + remotingListenConfiguration, + true, + Unity.Collections.LowLevel.Unsafe.NativeArrayUnsafeUtility.GetUnsafePtr(s_secureRemotingListenConfiguration.Certificate), + (uint)(s_secureRemotingListenConfiguration.Certificate.Length), + s_secureRemotingListenConfiguration.SubjectName, + s_secureRemotingListenConfiguration.KeyPassphrase, + s_validateAuthenticationTokenCallback); + + } + else + { + NativeLib.ListenRemoting( + remotingListenConfiguration, + false, + null, + 0, + string.Empty, + string.Empty, + null); + } + } + } + + internal void InitializePlayModeRemoting(RemotingConnectConfiguration playModeConfiguration) + { + m_remotingConnectConfiguration = playModeConfiguration; + s_remotingState = RemotingState.Connect; + InitializeRemoting(); + } + + internal void OnSessionLossPending() + { + if (s_remotingState == RemotingState.Connect) + { + _ = TryGetConnectionState(out ConnectionState connectionState, out m_disconnectReasonOnLossPending); + + if (m_disconnectReasonOnLossPending == DisconnectReason.RemotingVersionMismatch) + { + Debug.LogError($"The Holographic Remoting Player app has a mismatched version " + + $"on the remote host {m_remotingConnectConfiguration.RemoteHostName}:{m_remotingConnectConfiguration.RemotePort}. " + + $"Please update the Player app on your headset and try again."); + } + else + { + Debug.LogError($"[AppRemotingSubsystem] Cannot establish a connection to Holographic Remoting Player " + + $"on the target with IP Address {m_remotingConnectConfiguration.RemoteHostName}:{m_remotingConnectConfiguration.RemotePort}." + + $"Disconnect Reason:{m_disconnectReasonOnLossPending}"); + } + + } + else if (s_remotingState == RemotingState.Listen) + { + Debug.Log("[AppRemotingSubsystem] Listening to incoming Holographic Remoting connection is interrupted."); + } + } + + private System.Collections.IEnumerator ConnectRoutine(RemotingConnectConfiguration connectConfiguration) + { + var defaultWait = new WaitForSeconds(0.5f); + if (s_remotingState == RemotingState.Idle) + { + m_remotingConnectConfiguration = connectConfiguration; + m_remotingListenConfiguration = default; + s_remotingState = RemotingState.Connect; + + ConnectionState previousConnectionState = ConnectionState.Disconnected; + yield return new GameObject("StartOrStopXRHelper", typeof(StartOrStopXRHelper)) + { + hideFlags = HideFlags.HideAndDontSave + }; + + while (true) + { + if (!TryGetConnectionState(out ConnectionState connectionState, out DisconnectReason disconnectReason)) + { + connectionState = ConnectionState.Disconnected; + // TryGetConnectionState() cannot retreive correct disconnectReason after the context gets invalid, + // which happens immediately after session loss pending. Use the prevviously stored disconnectReason + // on session loss pending and a valid context below. + if(m_disconnectReasonOnLossPending != DisconnectReason.None) + { + disconnectReason = m_disconnectReasonOnLossPending; + m_disconnectReasonOnLossPending = DisconnectReason.None; + } + } + + if (connectionState != previousConnectionState) + { + previousConnectionState = connectionState; + + if (connectionState == ConnectionState.Connected) + { + Connected?.Invoke(); + } + else if (connectionState == ConnectionState.Disconnected) + { + Disconnecting?.Invoke(disconnectReason); + } + } + + if (XRGeneralSettings.Instance.Manager.activeLoader == null) + { + break; + } + yield return defaultWait; + } + } + else + { + Debug.LogError("Cannot connect when previous connection is still in progress"); + } + } + + internal void StartConnecting(RemotingConnectConfiguration connectConfiguration) + { + AppRemotingCoroutineRunner.Start(ConnectRoutine(connectConfiguration)); + } + +#pragma warning disable CS0618 // to use the obsolete fields to connect + internal System.Collections.IEnumerator ConnectLegacy(RemotingConfiguration configuration) + { + RemotingConnectConfiguration connectConfiguration; + connectConfiguration.RemoteHostName = configuration.RemoteHostName; + connectConfiguration.RemotePort = configuration.RemotePort; + connectConfiguration.MaxBitrateKbps = configuration.MaxBitrateKbps; + connectConfiguration.VideoCodec = configuration.VideoCodec; + connectConfiguration.EnableAudio = configuration.EnableAudio; + connectConfiguration.AudioCaptureMode = RemotingAudioCaptureMode.SystemWideCapture; + connectConfiguration.secureConnectConfiguration = null; + yield return ConnectRoutine(connectConfiguration); + } +#pragma warning restore CS0618 + + private System.Collections.IEnumerator ListenRoutine(RemotingListenConfiguration listenConfiguration, ListenMode listenMode, Action onRemotingListenCompleted) + { + var defaultWait = new WaitForSeconds(0.5f); + s_listenMode = listenMode; + + if (s_remotingState == RemotingState.Idle) + { + m_remotingListenConfiguration = listenConfiguration; + m_remotingConnectConfiguration = default; + s_remotingState = RemotingState.Listen; + + while (s_remotingState == RemotingState.Listen) + { + ConnectionState previousConnectionState = ConnectionState.Disconnected; + yield return new GameObject("StartOrStopXRHelper", typeof(StartOrStopXRHelper)) + { + hideFlags = HideFlags.HideAndDontSave + }; + + while (true) + { + if (!TryGetConnectionState(out ConnectionState connectionState, out DisconnectReason disconnectReason)) + { + connectionState = ConnectionState.Disconnected; + } + + if (connectionState != previousConnectionState) + { + previousConnectionState = connectionState; + + if (connectionState == ConnectionState.Connected) + { + Connected?.Invoke(); + } + else if (connectionState == ConnectionState.Disconnected) + { + Debug.Log("[AppRemotingSubsystem] Listen, After disconnection, Stop XR Loader."); + Disconnecting?.Invoke(disconnectReason); + StartOrStopXRHelper.StopXrLoader(); + break; // If disconnected, stop XR session and try to restart. + } + } + + if (XRGeneralSettings.Instance.Manager.activeLoader == null) + { + break; // if XR loader is already stopped, try to restart. + } + yield return defaultWait; + } + yield return defaultWait; + } + } + else + { + Debug.LogError("[AppRemotingSubsystem] Cannot listen when previous connection is still in progress"); + } + + if (onRemotingListenCompleted != null && s_listenMode == ListenMode.LegacyListen) + { + onRemotingListenCompleted.Invoke(); + } + } + + internal void StartListening(RemotingListenConfiguration listenConfiguration, ListenMode listenMode, Action onRemotingListenCompleted = null) + { + AppRemotingCoroutineRunner.Start(ListenRoutine(listenConfiguration, listenMode, onRemotingListenCompleted)); + } + + internal System.Collections.IEnumerator ListenLegacy(RemotingListenConfiguration listenConfiguration, ListenMode listenMode, Action onRemotingListenCompleted = null) + { + yield return ListenRoutine(listenConfiguration, listenMode, onRemotingListenCompleted); + } + + // IL2CPP does not support marshaling delegates that point to instance methods to native code. + // Using a static method that handles the callback and redirect accordingly. Note that + // certificate handling is also done in the following method and hence the signature is a little different than ValidateServerCertificateCallback. + [MonoPInvokeCallback] + private static SecureRemotingCertificateValidationResult ImplementValidateServerCertificate(string hostName, SecureRemotingCertificateValidationResult systemValidationResult) + { + X509Certificate2Collection certChain = GetCertificateChain(); + SecureRemotingCertificateValidationResult? systemValidationResultPassed = s_secureRemotingConnectConfiguration.PerformSystemValidation ? systemValidationResult : (SecureRemotingCertificateValidationResult?)null; + return s_secureRemotingConnectConfiguration.ValidateServerCertificateCallback(hostName, certChain, systemValidationResultPassed); + } + + // Intended to use only as part of secure connect + private static X509Certificate2Collection GetCertificateChain() + { + X509Certificate2Collection certChain = new X509Certificate2Collection(); + uint certChainLength = NativeLib.GetNumCertificates(); + for (uint certIndex = 0; certIndex < certChainLength; certIndex++) + { + IntPtr certificate = NativeLib.GetCertificate(certIndex, out int size); + byte[] certByteArray = new byte[size]; + Marshal.Copy(certificate, certByteArray, 0, size); + X509Certificate2 cert = new X509Certificate2(certByteArray); + certChain.Add(cert); + } + return certChain; + } + + // IL2CPP does not support marshaling delegates that point to instance methods to native code. + // Using a static method that handles the callback and redirect accordingly. + [MonoPInvokeCallback] + private static bool ImplementValidateAuthenticationToken(string authenticationTokenToCheck) + { + return s_secureRemotingListenConfiguration.ValidateAuthenticationTokenCallback(authenticationTokenToCheck); + } + + private System.Collections.IEnumerator DisconnectAndStopXR() + { + if (OpenXRContext.Current.Instance != 0) + { + // Notify the AR Foundation subsystems before the subsystem destroy and + // allow some time for cleaning up + NativeLib.DestroyAnchorSubsystemPending(); + + // wait for one frame to make sure the Anchor changes are notified to Unity on GetAnchorChanges() callback + yield return null; + + NativeLib.RemoveAllAnchors(); + + // wait for one frame to make sure removed anchors are notified + yield return null; + + NativeLib.DisconnectRemoting(); + } + + StartOrStopXRHelper.StopXrLoader(); + } + + internal void Disconnect(bool invokedFromStopListening = false) + { + if (s_remotingState != RemotingState.Connect && s_remotingState != RemotingState.Listen) + { + Debug.LogError("[AppRemotingSubsystem] Cannot disconnect when the remoting connection is not in progress."); + } + Disconnecting?.Invoke(DisconnectReason.DisconnectRequest); + if (s_remotingState != RemotingState.Disconnecting) + { + RemotingState previousRemotingState = s_remotingState; + s_remotingState = RemotingState.Disconnecting; + AppRemotingCoroutineRunner.Start(DisconnectAndStopXR()); + if (previousRemotingState == RemotingState.Listen && s_listenMode == ListenMode.Listen && !invokedFromStopListening) + { + // Return if stopListening is not invoked and continue listening. + Debug.Log("[AppRemotingSubsystem] Disconnect, Try restart XR session"); + s_remotingState = RemotingState.Listen; + return; + } + else + { + s_remotingState = RemotingState.Idle; + if (IsAppRemotingEnabled() && !InPlayModeRemoting()) + { + ReadyToStart?.Invoke(); + } + } + } + } + + internal void StopListening() + { + if (s_remotingState != RemotingState.Listen) + { + Debug.LogError("[AppRemotingSubsystem] Cannot stop listening when remoting listen is not in progress"); + return; + } + else if (s_listenMode == ListenMode.LegacyListen) + { + Debug.LogError("[AppRemotingSubsystem] StopListening is not supported with `Listen` coroutine, use `Disconnect` instead"); + return; + } + else + { + Disconnect(invokedFromStopListening: true); + } + } + + internal event ReadyToStartDelegate ReadyToStart; + internal event DisconnectingDelegate Disconnecting; + internal event ConnectedDelegate Connected; + } + + internal class StartOrStopXRHelper : MonoBehaviour + { + private void Start() + { + // Please make sure to enable "Microphone" capability in Unity Player settings for speech recognition to work + // in UWP app remoting. Although it does not use the microphone on remote PC, Unity needs this. + StartCoroutine(EnsureInitialization()); + } + + public static System.Collections.IEnumerator EnsureInitialization() + { + if (XRGeneralSettings.Instance.Manager.activeLoader == null) + { + Debug.Log("[AppRemotingSubsystem] InitializeLoader"); + yield return XRGeneralSettings.Instance.Manager.InitializeLoader(); + } + + if (XRGeneralSettings.Instance.Manager.activeLoader != null) + { + Debug.Log("[AppRemotingSubsystem] StartSubsystems"); + XRGeneralSettings.Instance.Manager.StartSubsystems(); + } + } + + public static void StopXrLoader() + { + if (XRGeneralSettings.Instance.Manager.activeLoader != null) + { + XRGeneralSettings.Instance.Manager.StopSubsystems(); + Debug.Log("[AppRemotingSubsystem] StopSubsystems"); + + if (XRGeneralSettings.Instance.Manager.isInitializationComplete) + { + XRGeneralSettings.Instance.Manager.DeinitializeLoader(); + Debug.Log("[AppRemotingSubsystem] DeinitializeLoader"); + } + } + } + +#if UNITY_EDITOR + public static void OnEnterPlaymodeInEditor() + { + EditorApplication.playModeStateChanged += OnPlayModeStateChanged; + } + + private static void OnPlayModeStateChanged(PlayModeStateChange state) + { + // If PlayModeRemotingPlugin isn't enabled or InitManagerOnStart is enabled, we don't need the helper. + XRGeneralSettings standaloneGeneralSettings = XRSettingsHelpers.GetOrCreateXRGeneralSettings(BuildTargetGroup.Standalone); + if (!OpenXRFeaturePlugin.Feature.IsValidAndEnabled() || standaloneGeneralSettings == null || standaloneGeneralSettings.InitManagerOnStart) + { + EditorApplication.playModeStateChanged -= OnPlayModeStateChanged; + return; + } + + if (state == PlayModeStateChange.EnteredPlayMode) + { + _ = new GameObject("StartOrStopXRHelper", typeof(StartOrStopXRHelper)) + { + hideFlags = HideFlags.HideAndDontSave + }; + } + else if (state == PlayModeStateChange.ExitingPlayMode) + { + StopXrLoader(); + EditorApplication.playModeStateChanged -= OnPlayModeStateChanged; + } + } +#endif + } + + internal delegate SecureRemotingCertificateValidationResult InternalValidateServerCertificateDelegate(string hostName, SecureRemotingCertificateValidationResult systemValidationResult); + + internal enum ListenMode + { + Listen = 0, + LegacyListen = 1 + }; + + internal enum RemotingState + { + Idle = 0, + Connect = 1, + Listen = 2, + Disconnecting = 3 + } + + // This internal struct is same as "RemotingConnectConfiguration" without "SecureRemotingConnectConfiguration" + // used for native marshalling purposes. + internal struct InternalRemotingConnectConfiguration + { + public string RemoteHostName; + public ushort RemotePort; + public uint MaxBitrateKbps; + public RemotingVideoCodec VideoCodec; + public bool EnableAudio; + public RemotingAudioCaptureMode AudioCaptureMode; + } + + // This internal struct is same as "RemotingListenConfiguration" without "SecureRemotingListenConfiguration" + // used for native marshalling purposes. + internal struct InternalRemotingListenConfiguration + { + public string ListenInterface; + public ushort HandshakeListenPort; + public ushort TransportListenPort; + public uint MaxBitrateKbps; + public RemotingVideoCodec VideoCodec; + public bool EnableAudio; + public RemotingAudioCaptureMode AudioCaptureMode; + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingSubsystem.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingSubsystem.cs.meta new file mode 100644 index 0000000..e4628ab --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/AppRemotingSubsystem.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c44c8b3904d989447bf81120a910b29e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/GestureSubsystem.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/GestureSubsystem.cs new file mode 100644 index 0000000..0b8538e --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/GestureSubsystem.cs @@ -0,0 +1,190 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Runtime.CompilerServices; +using System.Runtime.InteropServices; +using UnityEngine; +using UnityEngine.XR.OpenXR; + +namespace Microsoft.MixedReality.OpenXR +{ + [Flags] + internal enum NativeDirectionFlags + { + X = 1, + Y = 2, + Z = 4, + } + + [StructLayout(LayoutKind.Sequential, Pack = 8)] + internal struct NativeGesturePoseData + { + public ulong gestureTime; + public Pose headPose; + public NativeSpaceLocationFlags headPoseFlags; + public Pose eyeGazePose; + public NativeSpaceLocationFlags eyeGazePoseFlags; + public Pose handAimPose; + public NativeSpaceLocationFlags handAimPoseFlags; + public Pose handGripPose; + public NativeSpaceLocationFlags handGripPoseFlags; + } + + [StructLayout(LayoutKind.Sequential, Pack = 8)] + internal struct NativeGestureEventData + { + public GestureEventType eventType; + public GestureHandedness handedness; + public NativeGesturePoseData poseData; + public TappedEventData tappedData; + public ManipulationEventData manipulationData; + public NavigationEventData navigationData; + } + + internal static class GestureSubsystemExtensions + { + [MethodImpl(MethodImplOptions.AggressiveInlining)] + public static bool IsValid(this NativeSpaceLocationFlags flags) + { + return flags.HasFlag(NativeSpaceLocationFlags.OrientationValid) && + flags.HasFlag(NativeSpaceLocationFlags.PositionValid); + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + public static bool IsTracked(this NativeSpaceLocationFlags flags) + { + return flags.HasFlag(NativeSpaceLocationFlags.OrientationTracked) && + flags.HasFlag(NativeSpaceLocationFlags.PositionTracked); + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + public static bool IsTappedEvent(this NativeGestureEventData eventData) + { + return eventData.eventType.HasFlag(GestureEventType.Tapped); + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + public static bool IsManipulationEvent(this NativeGestureEventData eventData) + { + var eventType = eventData.eventType; + return eventType.HasFlag(GestureEventType.ManipulationStarted) || + eventType.HasFlag(GestureEventType.ManipulationUpdated) || + eventType.HasFlag(GestureEventType.ManipulationCompleted); + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + public static bool IsNavigationEvent(this NativeGestureEventData eventData) + { + var eventType = eventData.eventType; + return eventType.HasFlag(GestureEventType.NavigationStarted) || + eventType.HasFlag(GestureEventType.NavigationUpdated) || + eventType.HasFlag(GestureEventType.NavigationCompleted); + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + public static T? Get(this NativeGestureEventData eventData, T value, bool hasValue) where T : struct + { + if (hasValue) + { + return value; + } + return null; + } + } + internal class GestureSubsystem : Disposable + { + private static MixedRealityFeaturePlugin Feature => OpenXRFeaturePlugin.Feature; + private readonly ulong m_gestureRecognizerHandle = 0; + private GestureSettings m_gestureSettings = GestureSettings.None; + private bool m_running = false; + private readonly object m_runningLock = new object(); + + internal static GestureSubsystem TryCreateGestureSubsystem(GestureSettings settings) + { + if (!Feature.IsValidAndEnabled()) + { + Debug.LogWarning($"{MixedRealityFeaturePlugin.featureName} is not enabled."); + return null; + } + + ulong handle = NativeLib.TryCreateGestureRecognizer(settings); + if (handle == 0) + { + Debug.LogWarning($"GestureSubsystem is not supported with settings: {settings}."); + return null; + } + + return new GestureSubsystem(settings, handle); + } + + private GestureSubsystem(GestureSettings settings, ulong handle) + { + m_gestureRecognizerHandle = handle; + m_gestureSettings = settings; + } + + internal GestureSettings GestureSettings + { + get { return m_gestureSettings; } + set + { + if (m_gestureSettings != value) + { + if (NativeLib.TrySetGestureSettings(m_gestureRecognizerHandle, value)) + { + m_gestureSettings = value; + } + else + { + Debug.LogWarning($"Cannot set gesture setting to {value}"); + } + } + } + } + + internal bool TryGetNextEvent(ref GestureEventData eventData) + { + return NativeLib.TryGetNextEventData(m_gestureRecognizerHandle, ref eventData); + } + + internal void CancelPendingGestures() + { + NativeLib.CancelPendingGesture(m_gestureRecognizerHandle); + } + + protected override void DisposeNativeResources() + { + base.DisposeNativeResources(); + NativeLib.DestroyGestureRecognizer(m_gestureRecognizerHandle); + } + + internal void Start() + { + lock (m_runningLock) + { + if (m_running) + { + Debug.LogError($"GestureSubsystem is already started."); + return; + } + NativeLib.StartGestureRecognizer(m_gestureRecognizerHandle); + m_running = true; + } + } + + internal void Stop() + { + lock (m_runningLock) + { + if (!m_running) + { + Debug.LogError($"GestureSubsystem cannot be stopped before started."); + return; + } + m_running = false; + NativeLib.StopGestureRecognizer(m_gestureRecognizerHandle); + } + } + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/GestureSubsystem.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/GestureSubsystem.cs.meta new file mode 100644 index 0000000..1a77ac9 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/GestureSubsystem.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d431827c2b472b74197336eeffefbcce +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/HandTrackingSubsystem.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/HandTrackingSubsystem.cs new file mode 100644 index 0000000..82df883 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/HandTrackingSubsystem.cs @@ -0,0 +1,53 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System.Collections.Generic; +using UnityEngine; +using UnityEngine.XR; + +namespace Microsoft.MixedReality.OpenXR +{ + internal class HandTrackingSubsystemController : SubsystemController + { + // Must be the same as inputs.id in UnitySubsystemsManifest.json + // and the same for RegisterLifecycleProvider in InputProvider.cpp + public const string Id = "OpenXR Input Extension"; + + private XRInputSubsystem m_inputExtensionSubsystem = null; + + public HandTrackingSubsystemController(IOpenXRContext context) : base(context) { } + + public override void OnSubsystemCreate(ISubsystemPlugin plugin) + { + var descriptors = new List(); + SubsystemManager.GetSubsystemDescriptors(descriptors); + foreach (var descriptor in descriptors) + { + if (string.Compare(descriptor.id, Id, true) == 0) + { + m_inputExtensionSubsystem = descriptor.Create(); + if (m_inputExtensionSubsystem != null) + { + break; + } + } + } + } + + public override void OnSubsystemStart(ISubsystemPlugin plugin) + { + m_inputExtensionSubsystem?.Start(); + } + + public override void OnSubsystemStop(ISubsystemPlugin plugin) + { + m_inputExtensionSubsystem?.Stop(); + } + + public override void OnSubsystemDestroy(ISubsystemPlugin plugin) + { + m_inputExtensionSubsystem?.Destroy(); + m_inputExtensionSubsystem = null; + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/HandTrackingSubsystem.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/HandTrackingSubsystem.cs.meta new file mode 100644 index 0000000..09b53c9 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/HandTrackingSubsystem.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: a99a166e90ef7df4b946c73e766fff0a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/InternalMeshSettings.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/InternalMeshSettings.cs new file mode 100644 index 0000000..2ca58cb --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/InternalMeshSettings.cs @@ -0,0 +1,16 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +namespace Microsoft.MixedReality.OpenXR +{ + internal static class InternalMeshSettings + { + /// + /// Change the settings for future meshes given by the OpenXR XRMeshSubsystem. + /// + public static bool TrySetMeshComputeSettings(MeshComputeSettings settings) + { + return NativeLib.SetMeshComputeSettings(settings); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/InternalMeshSettings.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/InternalMeshSettings.cs.meta new file mode 100644 index 0000000..b420f45 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/InternalMeshSettings.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: ce89d7099a44b344191d40c50169ff75 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/MeshSubsystem.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/MeshSubsystem.cs new file mode 100644 index 0000000..51aa4a4 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/MeshSubsystem.cs @@ -0,0 +1,31 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System.Collections.Generic; +using UnityEngine.XR; + +namespace Microsoft.MixedReality.OpenXR +{ + internal class MeshSubsystemController : SubsystemController + { + // Must be the same as meshings.id in UnitySubsystemsManifest.json + // and the same for RegisterLifecycleProvider in InputProvider.cpp + public const string Id = "OpenXR Mesh Extension"; + + private static List s_MeshDescriptors = new List(); + + public MeshSubsystemController(IOpenXRContext context) : base(context) + { + } + + public override void OnSubsystemCreate(ISubsystemPlugin plugin) + { + plugin.CreateSubsystem(s_MeshDescriptors, Id); + } + + public override void OnSubsystemDestroy(ISubsystemPlugin plugin) + { + plugin.DestroySubsystem(); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/MeshSubsystem.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/MeshSubsystem.cs.meta new file mode 100644 index 0000000..855e1dc --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/MeshSubsystem.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0d1fbb86d3168624fbbe8e45490f221a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/OpenXRRuntimeRestartHandler.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/OpenXRRuntimeRestartHandler.cs new file mode 100644 index 0000000..e8cbc2b --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/OpenXRRuntimeRestartHandler.cs @@ -0,0 +1,64 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using UnityEngine; +using UnityEngine.XR.OpenXR; +using UnityEngine.XR.OpenXR.Features; + +namespace Microsoft.MixedReality.OpenXR +{ + internal class OpenXRRuntimeRestartHandler : IDisposable + { + private readonly OpenXRFeature m_feature = null; + private readonly string m_featureName; + private readonly bool? m_skipRestart = null; + private readonly bool? m_skipQuitApp = null; + + public OpenXRRuntimeRestartHandler(OpenXRFeature feature, bool? skipRestart = null, bool? skipQuitApp = null) + { + m_feature = feature; + m_featureName = feature.GetType().Name; + m_skipRestart = skipRestart; + m_skipQuitApp = skipQuitApp; + + Debug.Log($"[OpenXRRuntimeRestartHandler] is created for {m_featureName}."); + + OpenXRRuntime.wantsToRestart += OpenXRRuntime_wantsToRestart; + OpenXRRuntime.wantsToQuit += OpenXRRuntime_wantsToQuit; + } + + public void Dispose() + { + Debug.Log($"[OpenXRRuntimeRestartHandler] is disposed for {m_featureName}"); + OpenXRRuntime.wantsToQuit -= OpenXRRuntime_wantsToQuit; + OpenXRRuntime.wantsToRestart -= OpenXRRuntime_wantsToRestart; + } + + private bool OpenXRRuntime_wantsToQuit() + { + if (m_feature.IsValidAndEnabled() && m_skipQuitApp == true) + { + Debug.Log($"[OpenXRRuntimeRestartHandler] {m_featureName} attempts to skip quitting the app after XR session is finished."); + return false; // skip quitting application after XR session is finished. + } + else + { + return true; // yield the decision to other wantsToQuit event handlers. + } + } + + private bool OpenXRRuntime_wantsToRestart() + { + if (m_feature.IsValidAndEnabled() && m_skipRestart == true) + { + Debug.Log($"[OpenXRRuntimeRestartHandler] {m_featureName} attempts to skip restarting XR session."); + return false; // skip restarting XR session. + } + else + { + return true; // yield the decision to other wantsToRestart event handlers. + } + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/OpenXRRuntimeRestartHandler.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/OpenXRRuntimeRestartHandler.cs.meta new file mode 100644 index 0000000..2310e36 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/OpenXRRuntimeRestartHandler.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: a8a5ed93f1d1b344c850eedc9b4a0920 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PlaneSubsystem.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PlaneSubsystem.cs new file mode 100644 index 0000000..a7ef780 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PlaneSubsystem.cs @@ -0,0 +1,204 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using System.Runtime.InteropServices; +using Unity.Collections; +using Unity.Collections.LowLevel.Unsafe; +using UnityEngine; +using UnityEngine.XR.ARSubsystems; + +namespace Microsoft.MixedReality.OpenXR +{ + internal enum XrSceneObjectTypeMSFT + { + XR_SCENE_OBJECT_TYPE_UNCATEGORIZED_MSFT = -1, + XR_SCENE_OBJECT_TYPE_BACKGROUND_MSFT = 1, + XR_SCENE_OBJECT_TYPE_WALL_MSFT = 2, + XR_SCENE_OBJECT_TYPE_FLOOR_MSFT = 3, + XR_SCENE_OBJECT_TYPE_CEILING_MSFT = 4, + XR_SCENE_OBJECT_TYPE_PLATFORM_MSFT = 5, + XR_SCENE_OBJECT_TYPE_INFERRED_MSFT = 6, + XR_SCENE_OBJECT_TYPE_MAX_ENUM_MSFT = 0x7FFFFFFF + } + + [StructLayout(LayoutKind.Sequential, Pack = 8)] + internal struct NativePlane + { + public Guid id; + public Vector3 position; + public Quaternion rotation; + public TrackingState trackingState; + public Vector2 size; + public XrSceneObjectTypeMSFT type; + } + + internal class PlaneSubsystem : XRPlaneSubsystem + { + public const string Id = "OpenXR Planefinding"; + + private class OpenXRProvider : Provider + { + private PlaneDetectionMode m_planeDetectionMode = PlaneDetectionMode.Vertical & PlaneDetectionMode.Horizontal; + + public OpenXRProvider() + { + } + public override void Start() + { + NativeLib.StartPlaneSubsystem(); + } + public override void Stop() + { + NativeLib.StopPlaneSubsystem(); + } + public override void Destroy() + { + NativeLib.DestroyPlaneSubsystem(); + } + + public override PlaneDetectionMode currentPlaneDetectionMode { get => m_planeDetectionMode; } + public override PlaneDetectionMode requestedPlaneDetectionMode + { + get => m_planeDetectionMode; + set + { + m_planeDetectionMode = value; + NativeLib.SetPlaneDetectionMode(m_planeDetectionMode); + } + } + + public unsafe override TrackableChanges GetChanges(BoundedPlane defaultPlane, Allocator allocator) + { + uint numAddedPlanes = 0; + uint numUpdatedPlanes = 0; + uint numRemovedPlanes = 0; + NativeLib.GetNumPlaneChanges(FrameTime.OnUpdate, ref numAddedPlanes, ref numUpdatedPlanes, ref numRemovedPlanes); + + using (var addedNativePlanes = new NativeArray((int)numAddedPlanes, allocator, NativeArrayOptions.UninitializedMemory)) + using (var updatedNativePlanes = new NativeArray((int)numUpdatedPlanes, allocator, NativeArrayOptions.UninitializedMemory)) + using (var removedNativePlanes = new NativeArray((int)numRemovedPlanes, allocator, NativeArrayOptions.UninitializedMemory)) + { + if (numAddedPlanes + numUpdatedPlanes + numRemovedPlanes > 0) + { + NativeLib.GetPlaneChanges( + (uint)(numAddedPlanes * sizeof(NativePlane)), + NativeArrayUnsafeUtility.GetUnsafePtr(addedNativePlanes), + (uint)(numUpdatedPlanes * sizeof(NativePlane)), + NativeArrayUnsafeUtility.GetUnsafePtr(updatedNativePlanes), + (uint)(numRemovedPlanes * sizeof(Guid)), + NativeArrayUnsafeUtility.GetUnsafePtr(removedNativePlanes)); + } + + // Added Planes + var addedPlanes = Array.Empty(); + if (numAddedPlanes > 0) + { + addedPlanes = new BoundedPlane[numAddedPlanes]; + for (int i = 0; i < numAddedPlanes; ++i) + addedPlanes[i] = ToBoundedPlane(addedNativePlanes[i], defaultPlane); + } + + // Updated Planes + var updatedPlanes = Array.Empty(); + if (numUpdatedPlanes > 0) + { + updatedPlanes = new BoundedPlane[numUpdatedPlanes]; + for (int i = 0; i < numUpdatedPlanes; ++i) + updatedPlanes[i] = ToBoundedPlane(updatedNativePlanes[i], defaultPlane); + } + + // Removed Planes + var removedPlanes = Array.Empty(); + if (numRemovedPlanes > 0) + { + removedPlanes = new TrackableId[numRemovedPlanes]; + for (int i = 0; i < numRemovedPlanes; ++i) + removedPlanes[i] = FeatureUtils.ToTrackableId(removedNativePlanes[i]); + } + + return TrackableChanges.CopyFrom( + new NativeArray(addedPlanes, allocator), + new NativeArray(updatedPlanes, allocator), + new NativeArray(removedPlanes, allocator), + allocator); + } + } + + private PlaneClassification ToPlaneClassification(XrSceneObjectTypeMSFT type) + { + switch (type) + { + case XrSceneObjectTypeMSFT.XR_SCENE_OBJECT_TYPE_WALL_MSFT: + return PlaneClassification.Wall; + + case XrSceneObjectTypeMSFT.XR_SCENE_OBJECT_TYPE_FLOOR_MSFT: + return PlaneClassification.Floor; + + case XrSceneObjectTypeMSFT.XR_SCENE_OBJECT_TYPE_CEILING_MSFT: + return PlaneClassification.Ceiling; + + case XrSceneObjectTypeMSFT.XR_SCENE_OBJECT_TYPE_PLATFORM_MSFT: + return PlaneClassification.Table; + + case XrSceneObjectTypeMSFT.XR_SCENE_OBJECT_TYPE_UNCATEGORIZED_MSFT: + case XrSceneObjectTypeMSFT.XR_SCENE_OBJECT_TYPE_BACKGROUND_MSFT: + case XrSceneObjectTypeMSFT.XR_SCENE_OBJECT_TYPE_INFERRED_MSFT: + case XrSceneObjectTypeMSFT.XR_SCENE_OBJECT_TYPE_MAX_ENUM_MSFT: + default: + return PlaneClassification.None; + } + } + + private BoundedPlane ToBoundedPlane(NativePlane nativePlane, BoundedPlane defaultPlane) + { + return new BoundedPlane( + FeatureUtils.ToTrackableId(nativePlane.id), + TrackableId.invalidId, + new Pose(nativePlane.position, nativePlane.rotation), + Vector2.zero, + nativePlane.size, + PlaneAlignment.HorizontalUp, + nativePlane.trackingState, + defaultPlane.nativePtr, + ToPlaneClassification(nativePlane.type)); // TODO: Replace the nativePtr + } + } + + [RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.SubsystemRegistration)] + static void RegisterDescriptor() + { + XRPlaneSubsystemDescriptor.Create(new XRPlaneSubsystemDescriptor.Cinfo + { + id = Id, + providerType = typeof(PlaneSubsystem.OpenXRProvider), + subsystemTypeOverride = typeof(PlaneSubsystem), + supportsArbitraryPlaneDetection = true, + supportsBoundaryVertices = false, + supportsClassification = true, + supportsHorizontalPlaneDetection = true, + supportsVerticalPlaneDetection = true, + }); + } + }; + + internal class PlaneSubsystemController : SubsystemController + { + private static List s_PlaneDescriptors = new List(); + + public PlaneSubsystemController(IOpenXRContext context) : base(context) + { + } + + public override void OnSubsystemCreate(ISubsystemPlugin plugin) + { + plugin.CreateSubsystem(s_PlaneDescriptors, PlaneSubsystem.Id); + } + + public override void OnSubsystemDestroy(ISubsystemPlugin plugin) + { + plugin.DestroySubsystem(); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PlaneSubsystem.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PlaneSubsystem.cs.meta new file mode 100644 index 0000000..8f3f394 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PlaneSubsystem.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 2a16a17d75d3b234487c7fa33d1b53b2 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PluginEnvironment.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PluginEnvironment.cs new file mode 100644 index 0000000..08173b4 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PluginEnvironment.cs @@ -0,0 +1,60 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEngine; +using UnityEngine.XR.OpenXR; + +namespace Microsoft.MixedReality.OpenXR +{ + //Must match PluginEnvironment in PluginEnvironment.h + enum PluginEnvironment + { + unityVersion = 1 << 0, + openXRPluginVersion = 1 << 1, + mrOpenXRPluginVersion = 1 << 2, + graphicsAPI = 1 << 3, + sessionCreationResult = 1 << 4, + appName = 1 << 5, + appVersion = 1 << 6, + appMode = 1 << 7, + openXRRuntimeName = 1 << 8, + openXRRuntimeVersion = 1 << 9, + apiVersion = 1 << 10 + }; + + + internal class PluginEnvironmentSubsystem + { + private static bool m_initialized = false; + + internal static void InitializePlugin() + { + if (!m_initialized) + { + m_initialized = true; + NativeLib.SetPluginEnvironment(PluginEnvironment.unityVersion, Application.unityVersion); + NativeLib.SetPluginEnvironment(PluginEnvironment.openXRPluginVersion, OpenXRRuntime.pluginVersion); + NativeLib.SetPluginEnvironment(PluginEnvironment.mrOpenXRPluginVersion, typeof(OpenXRContext).Assembly.GetName().Version.ToString()); + NativeLib.InitializePlugin(); + } + } + + internal static void OnSessionCreated() + { + string appMode = "undefined"; + +#if UNITY_EDITOR + appMode = "PlayMode"; +#else + appMode = "AppMode"; +#endif + NativeLib.SetPluginEnvironment(PluginEnvironment.appName, Application.productName); + NativeLib.SetPluginEnvironment(PluginEnvironment.appVersion, Application.version); + NativeLib.SetPluginEnvironment(PluginEnvironment.appMode, appMode); + NativeLib.SetPluginEnvironment(PluginEnvironment.openXRRuntimeName, OpenXRRuntime.name); + NativeLib.SetPluginEnvironment(PluginEnvironment.openXRRuntimeVersion, OpenXRRuntime.version); + NativeLib.SetPluginEnvironment(PluginEnvironment.apiVersion, OpenXRRuntime.apiVersion); + } + } + +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PluginEnvironment.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PluginEnvironment.cs.meta new file mode 100644 index 0000000..9f0d029 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/PluginEnvironment.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: fa6caf8b98994bc4dbb2b4870889fa48 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/RaycastSubsystem.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/RaycastSubsystem.cs new file mode 100644 index 0000000..c52d0d4 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/RaycastSubsystem.cs @@ -0,0 +1,151 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System.Collections.Generic; +using Unity.Collections; +using Unity.XR.CoreUtils; +using UnityEngine; +using UnityEngine.XR.ARFoundation; +using UnityEngine.XR.ARSubsystems; + +namespace Microsoft.MixedReality.OpenXR +{ + internal class RaycastSubsystem : XRRaycastSubsystem + { + public const string Id = "OpenXR Raycasting"; + + private class OpenXRProvider : Provider + { + ARPlaneManager m_arPlaneManager = null; +#if USE_ARFOUNDATION_5_OR_NEWER + private XROrigin m_xrOrigin = null; +#else + private ARSessionOrigin m_arSessionOrigin = null; +#endif + + public override bool TryAddRaycast(Vector2 screenPoint, float estimatedDistance, out XRRaycast raycast) + { + Debug.LogError("Persistent raycasts are not supported; use single raycasts instead. More information about single and persistent raycasts is available at https://docs.unity3d.com/Packages/com.unity.xr.arfoundation@4.1/manual/raycast-manager.html#single-raycasts."); + raycast = default; + return false; + } + + public override bool TryAddRaycast(Ray ray, float estimatedDistance, out XRRaycast raycast) + { + Debug.LogError("Persistent raycasts are not supported; use single raycasts instead. More information about single and persistent raycasts is available at https://docs.unity3d.com/Packages/com.unity.xr.arfoundation@4.1/manual/raycast-manager.html#single-raycasts."); + raycast = default; + return false; + } + + public override void RemoveRaycast(TrackableId trackableId) + { + Debug.LogError("Persistent raycasts are not supported; use single raycasts instead. More information about single and persistent raycasts is available at https://docs.unity3d.com/Packages/com.unity.xr.arfoundation@4.1/manual/raycast-manager.html#single-raycasts."); + } + + public override TrackableChanges GetChanges(XRRaycast defaultRaycast, Allocator allocator) + { + // Check if the ARPlaneManager is null each frame. + // If the app Raycasts before this GetChanges, we won't forward the request to the ARPlaneManager, + // but if the raycast request is that early on the first frame, planes likely aren't available yet. + if (m_arPlaneManager == null) + { + // Find the active XROrigin (ARSessionOrigin for ARFoundation < 5.0), and its ARPlaneManager. +#if USE_ARFOUNDATION_5_OR_NEWER + if (m_xrOrigin == null || !m_xrOrigin.gameObject.activeInHierarchy) + { + m_xrOrigin = FindObjectUtility.FindFirstObjectByType(false); + if (m_xrOrigin != null) + { + m_arPlaneManager = m_xrOrigin.GetComponent(); + } + } +#else + if (m_arSessionOrigin == null || !m_arSessionOrigin.gameObject.activeInHierarchy) + { + m_arSessionOrigin = FindObjectUtility.FindFirstObjectByType(false); + if (m_arSessionOrigin != null) + { + m_arPlaneManager = m_arSessionOrigin.GetComponent(); + } + } +#endif + } + + return base.GetChanges(defaultRaycast, allocator); + } + + public override void Stop() + { + m_arPlaneManager = null; // Reset the cached ARPlaneManager. +#if USE_ARFOUNDATION_5_OR_NEWER + m_xrOrigin = null; // Reset the cached XROrigin. +#else + m_arSessionOrigin = null; // Reset the cached ARSessionOrigin. +#endif + } + public override void Destroy() + { + m_arPlaneManager = null; // Reset the cached ARPlaneManager. +#if USE_ARFOUNDATION_5_OR_NEWER + m_xrOrigin = null; // Reset the cached XROrigin. +#else + m_arSessionOrigin = null; // Reset the cached ARSessionOrigin. +#endif + } + + public override NativeArray Raycast( + XRRaycastHit defaultRaycastHit, + Ray ray, + TrackableType trackableTypeMask, + Allocator allocator) + { + // If we don't have a reference to the ARPlaneManager, we don't raycast. + // The ARPlaneManager Raycast is typically only called when it's enabled. + if (m_arPlaneManager == null || !m_arPlaneManager.enabled) + return new NativeArray(0, allocator); + + // Use the ARPlaneManager's raycast - this is the only raycast we currently need to support. + return m_arPlaneManager.Raycast(ray, trackableTypeMask, allocator); + } + } + + [RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.SubsystemRegistration)] + static void RegisterDescriptor() + { + XRRaycastSubsystemDescriptor.RegisterDescriptor(new XRRaycastSubsystemDescriptor.Cinfo + { + id = Id, + providerType = typeof(RaycastSubsystem.OpenXRProvider), + subsystemTypeOverride = typeof(RaycastSubsystem), + supportedTrackableTypes = TrackableType.Planes, + supportsTrackedRaycasts = false, + + // If this is not supported, ARFoundation will convert the ray to world space, then call the world-based raycast + supportsViewportBasedRaycast = false, + + // If this is not supported, ARFoundation will fallback to the ARPlaneManager's raycast. + // On editor remoting, this fallback redirect is broken, so we redirect in Raycast above. + supportsWorldBasedRaycast = true + }); + } + }; + + internal class RaycastSubsystemController : SubsystemController + { + private static List s_RaycastDescriptors = new List(); + + public RaycastSubsystemController(IOpenXRContext context) : base(context) + { + } + + public override void OnSubsystemCreate(ISubsystemPlugin plugin) + { + plugin.CreateSubsystem(s_RaycastDescriptors, RaycastSubsystem.Id); + } + + public override void OnSubsystemDestroy(ISubsystemPlugin plugin) + { + plugin.DestroySubsystem(); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/RaycastSubsystem.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/RaycastSubsystem.cs.meta new file mode 100644 index 0000000..639fb07 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/RaycastSubsystem.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 1871f5ac5ada8d34cac90551d4e52f7a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SelectKeywordRecognizerProvider.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SelectKeywordRecognizerProvider.cs new file mode 100644 index 0000000..5927fe7 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SelectKeywordRecognizerProvider.cs @@ -0,0 +1,215 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR || UNITY_STANDALONE || WINDOWS_UWP +using System; +using System.Reflection; +using UnityEngine; +using UnityEngine.Windows.Speech; +using static UnityEngine.Windows.Speech.PhraseRecognizer; +using System.Collections.Generic; +#if WINDOWS_UWP +using Windows.System.UserProfile; +using Windows.UI.Input.Spatial; +#endif // WINDOWS_UWP + +namespace Microsoft.MixedReality.OpenXR +{ + internal class SelectKeywordRecognizerProvider + { + internal SelectKeywordRecognizerProvider() + { + if (!IsSupported) + { + if (!IsPlatformSupported) + { + throw new NotSupportedException($"{nameof(SelectKeywordRecognizer)} is only supported when running the application on HoloLens 2. " + + $"Please use {nameof(IsSupported)} to check for support before calling the constructor."); + } + if (!IsSystemLanguageSupported) + { + throw new NotSupportedException($"{nameof(SelectKeywordRecognizer)} is not supported by the current system language. " + + $"Please use {nameof(IsSupported)} to check for support before calling the constructor."); + } + if (!IsUnityVersionSupported) + { + throw new NotSupportedException($"{nameof(SelectKeywordRecognizer)} is not supported by the current Unity version. " + + $"This is not expected and please file a bug with the Unity version number {Application.unityVersion} for us to investigate."); + } + } + } + + internal static bool IsSupported => IsPlatformSupported && IsUnityVersionSupported && IsSystemLanguageSupported; + + internal bool IsRunning { get; private set; } = false; + +#pragma warning disable CS0067 // Turn the "never used" warning off, as this is needed by SelectKeywordRecognizer + internal event PhraseRecognizedDelegate OnPhraseRecognized; +#pragma warning restore CS0067 + + internal void Start() + { +#if WINDOWS_UWP + if (IsRunning) + { + Debug.LogWarning($"{nameof(SelectKeywordRecognizer)} is already running when Start() is called."); + return; + } + SpatialInteractionManager.SourcePressed += SpatialInteractionManager_SourcePressed; + IsRunning = true; +#endif // WINDOWS_UWP + } + + internal void Stop() + { +#if WINDOWS_UWP + if (IsRunning) + { + SpatialInteractionManager.SourcePressed -= SpatialInteractionManager_SourcePressed; + IsRunning = false; + } + else + { + Debug.LogWarning($"{nameof(SelectKeywordRecognizer)} is not running when Stop() is called."); + return; + } +#endif // WINDOWS_UWP + } + + internal void Dispose() + { +#if WINDOWS_UWP + if (IsRunning) + { + Stop(); + } + m_spatialInteractionManager = null; +#endif // WINDOWS_UWP + } + + private static bool IsPlatformSupported + { + get + { + if (!m_isPlatformSupported.HasValue) + { + m_isPlatformSupported = CheckPlatformSupport(); + } + return m_isPlatformSupported.Value; + } + } + + private static bool IsUnityVersionSupported => m_recogEventArgsConstructorInfo != null; + + private static bool IsSystemLanguageSupported => m_localizedSelectKeyword != null; + + private static bool CheckPlatformSupport() + { +#if WINDOWS_UWP + return NativeLib.IsSelectKeywordFiltered(); +#else + return false; +#endif // WINDOWS_UWP + } + + private static PhraseRecognizedEventArgs GeneratePhraseRecognizedEventArgs() + { + return (PhraseRecognizedEventArgs)m_recogEventArgsConstructorInfo.Invoke( + new object[] { m_localizedSelectKeyword, ConfidenceLevel.High, null, DateTime.Now, new TimeSpan(m_selectKeywordDurationInTicks) }); + } + + private static ConstructorInfo GetPhraseRecognizedEventArgsConstructorInfo() + { + // Find the internal constructor using m_recogEventArgsConstructorArgTypes, the array storing the argument types of the constructor + return typeof(PhraseRecognizedEventArgs).GetConstructor( + bindingAttr: BindingFlags.NonPublic | BindingFlags.Instance, types: m_recogEventArgsConstructorArgTypes, binder: null, modifiers: null); + } + + private static string GetLocalizedKeyword() + { + if (m_currentSystemLanguage == null) + { + return null; + } + + // Due to the way system language is retrieved, m_currentSystemLanguage does not have a consistent format. + // For example, for Japanese it's "ja", for Chinese Simplified it's "zh-Hans-CN", for italian it's "it-IT". + // For almost all speech-supported language, all available flavors of the languages are speech-supported. + // The only exception is Chinese, where "zh-Hans-CN" is speech-supported but "zh-Hant-TW" is not. This if check handles this special case. + if (m_currentSystemLanguage == "zh-Hant-TW") + { + return null; + } + + // Only check for the first two letters as m_currentSystemLanguage has inconsistent formats for different locales + if (m_localizedSelectKeywordLookup.TryGetValue(m_currentSystemLanguage.Substring(0, 2), out string localizedKeyword)) + { + return localizedKeyword; + } + return null; + } + +#if WINDOWS_UWP + private void SpatialInteractionManager_SourcePressed(SpatialInteractionManager sender, SpatialInteractionSourceEventArgs args) + { + if (args.State.Source.Kind == SpatialInteractionSourceKind.Voice) + { + UnityEngine.WSA.Application.InvokeOnAppThread(() => + { + PhraseRecognizedEventArgs eventArgs = GeneratePhraseRecognizedEventArgs(); + OnPhraseRecognized.Invoke(eventArgs); + }, false); + } + } + + private static SpatialInteractionManager m_spatialInteractionManager = null; + + private static SpatialInteractionManager SpatialInteractionManager + { + get + { + if (m_spatialInteractionManager == null) + { + UnityEngine.WSA.Application.InvokeOnUIThread(() => + { + m_spatialInteractionManager = SpatialInteractionManager.GetForCurrentView(); + }, true); + } + + return m_spatialInteractionManager; + } + } +#endif // WINDOWS_UWP + + // The typical duration of pronouncing the "select" keyword in ticks (100 nanoseconds). 1 second. + private const long m_selectKeywordDurationInTicks = 10000000; + + private static readonly Type[] m_recogEventArgsConstructorArgTypes = new Type[] { + typeof(string), typeof(ConfidenceLevel), typeof(SemanticMeaning[]), typeof(DateTime), typeof(TimeSpan) }; + + private static readonly Dictionary m_localizedSelectKeywordLookup = new Dictionary + { + {"en", "select"}, + {"ja", "選択"}, + {"es", "seleccionar"}, + {"zh", "选择"}, + {"de", "auswählen"}, + {"fr", "sélectionner"}, + {"it", "seleziona"}, + }; + + private static readonly string m_currentSystemLanguage = +#if WINDOWS_UWP + GlobalizationPreferences.Languages[0]; +#else + null; +#endif // WINDOWS_UWP + + private static readonly string m_localizedSelectKeyword = GetLocalizedKeyword(); + + private static readonly ConstructorInfo m_recogEventArgsConstructorInfo = GetPhraseRecognizedEventArgsConstructorInfo(); + + private static bool? m_isPlatformSupported = null; + } +} +#endif // UNITY_EDITOR || UNITY_STANDALONE || WINDOWS_UWP \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SelectKeywordRecognizerProvider.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SelectKeywordRecognizerProvider.cs.meta new file mode 100644 index 0000000..b6e3aa9 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SelectKeywordRecognizerProvider.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 05d9e6ffae8f8cc43abe27ae4c42cb5e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SessionSubsystem.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SessionSubsystem.cs new file mode 100644 index 0000000..8d7d3b9 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SessionSubsystem.cs @@ -0,0 +1,172 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using Unity.Collections; +using UnityEngine; +using UnityEngine.Scripting; +using UnityEngine.XR.ARSubsystems; + +namespace Microsoft.MixedReality.OpenXR +{ + [Preserve] + internal class SessionSubsystem : XRSessionSubsystem + { + public const string Id = "OpenXR Session"; + + private class OpenXRProvider : Provider + { + private readonly Guid m_sessionGuid; + + private readonly Feature m_allSupportedFeatures = Feature.AnyTrackingMode | Feature.PlaneTracking | Feature.Raycast | Feature.Meshing; + // The requested features, excluding the requested tracking mode + private readonly Feature m_requestedBaseFeatures = Feature.PlaneTracking | Feature.Raycast | Feature.Meshing; + + private Feature m_requestedTrackingMode = Feature.PositionAndRotation; + private NativeSpaceLocationFlags m_trackingStateFlags; + + public OpenXRProvider() + { + m_sessionGuid = Guid.NewGuid(); + } + public override void Start() { } + public override void Stop() { } + public override void Destroy() { } + + public override Feature currentTrackingMode + { + get + { + if (!m_trackingStateFlags.HasFlag(NativeSpaceLocationFlags.OrientationValid) || + !m_trackingStateFlags.HasFlag(NativeSpaceLocationFlags.PositionValid)) + return Feature.None; + + if (!m_trackingStateFlags.HasFlag(NativeSpaceLocationFlags.PositionTracked)) + return Feature.RotationOnly; + + return Feature.PositionAndRotation; + } + } + + public override Feature requestedTrackingMode + { + get => m_requestedTrackingMode; + set + { + if (m_requestedTrackingMode != Feature.PositionAndRotation && m_requestedTrackingMode != Feature.RotationOnly) + { + Debug.Log("Session supported requested tracking modes are PositionAndRotation and RotationOnly."); + return; + } + m_requestedTrackingMode = value; + } + } + + public override TrackingState trackingState + { + get + { + if (!m_trackingStateFlags.HasFlag(NativeSpaceLocationFlags.PositionValid) || + !m_trackingStateFlags.HasFlag(NativeSpaceLocationFlags.OrientationValid)) + return TrackingState.None; + + if (m_trackingStateFlags.HasFlag(NativeSpaceLocationFlags.All)) + return TrackingState.Tracking; + + return TrackingState.Limited; + } + } + + public override NotTrackingReason notTrackingReason + { + get + { + if (!m_trackingStateFlags.HasFlag(NativeSpaceLocationFlags.PositionValid) || + !m_trackingStateFlags.HasFlag(NativeSpaceLocationFlags.OrientationValid)) + return NotTrackingReason.Initializing; + + if (m_trackingStateFlags.HasFlag(NativeSpaceLocationFlags.All)) + return NotTrackingReason.None; + + return NotTrackingReason.Relocalizing; + } + } + + public override bool matchFrameRateEnabled { get => false; } + public override bool matchFrameRateRequested { get => false; } + public override int frameRate { get => 0; } // Framerate is not supported unless matchFrameRateEnabled = true + + public override Feature requestedFeatures { get => m_requestedBaseFeatures | m_requestedTrackingMode; } + public override IntPtr nativePtr { get => IntPtr.Zero; } + public override Guid sessionId { get => m_sessionGuid; } + + /// + /// Get the session's availability, such as whether the platform supports XR. + /// SessionAvailability.None: "Default value. The availability is unknown." + /// SessionAvailability.Supported: "The current device is AR capable (but might require a software update)." + /// SessionAvailability.Installed: "The required AR software is installed on the device." + /// + public override Promise GetAvailabilityAsync() + { + if (OpenXRContext.Current.SystemId != 0) + { + return Promise.CreateResolvedPromise(SessionAvailability.Supported | SessionAvailability.Installed); + } + + return Promise.CreateResolvedPromise(SessionAvailability.None); + } + + public override NativeArray GetConfigurationDescriptors(Allocator allocator) + { + // Sessions may have multiple 'modes' of operation, each with a different set of capabilities. + // Our session only has one such mode, so this array will always be one ConfigurationDescriptor long. + var nativeArray = new NativeArray(1, allocator, NativeArrayOptions.UninitializedMemory); + nativeArray[0] = new ConfigurationDescriptor(IntPtr.Zero, m_allSupportedFeatures, 0); + return nativeArray; + } + + public override void OnApplicationPause() { } + public override void OnApplicationResume() { } + + // Only one configuration is supported, so the configuration settings can be inferred in the standard update. + public override void Update(XRSessionUpdateParams updateParams, Configuration configuration) => Update(updateParams); + public override void Update(XRSessionUpdateParams updateParams) + { + m_trackingStateFlags = NativeLib.GetPrimaryViewTrackingState(); + } + } + + [RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.SubsystemRegistration)] + static void RegisterDescriptor() + { + XRSessionSubsystemDescriptor.RegisterDescriptor(new XRSessionSubsystemDescriptor.Cinfo + { + id = Id, + providerType = typeof(SessionSubsystem.OpenXRProvider), + subsystemTypeOverride = typeof(SessionSubsystem), + supportsInstall = false, + supportsMatchFrameRate = false + }); + } + }; + + internal class SessionSubsystemController : SubsystemController + { + private static List s_SessionDescriptors = new List(); + + public SessionSubsystemController(IOpenXRContext context) : base(context) + { + } + + public override void OnSubsystemCreate(ISubsystemPlugin plugin) + { + plugin.CreateSubsystem(s_SessionDescriptors, SessionSubsystem.Id); + } + + public override void OnSubsystemDestroy(ISubsystemPlugin plugin) + { + plugin.DestroySubsystem(); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SessionSubsystem.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SessionSubsystem.cs.meta new file mode 100644 index 0000000..70182f9 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SessionSubsystem.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: cf07d17abfabf30459ae006cf82f30be +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SubsystemController.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SubsystemController.cs new file mode 100644 index 0000000..66bedb9 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SubsystemController.cs @@ -0,0 +1,101 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + internal delegate void OpenXRContextEvent(IOpenXRContext sender, EventArgs args); + internal delegate void OpenXRContextEvent(IOpenXRContext sender, TEventArgs args) where TEventArgs : EventArgs; + + internal interface IOpenXRContext + { + ulong Instance { get; } + ulong SystemId { get; } + ulong Session { get; } + bool IsSessionRunning { get; } + XrSessionState SessionState { get; } + ulong SceneOriginSpace { get; } + + event OpenXRContextEvent InstanceCreated; // after instance is created + event OpenXRContextEvent InstanceDestroying; // before instance is destroyed + event OpenXRContextEvent SessionCreated; // after session is created + event OpenXRContextEvent SessionDestroying; // before session is destroyed + event OpenXRContextEvent SessionBegun; // after session is begun + event OpenXRContextEvent SessionEnding; // before session is ended + } + + internal interface ISubsystemPlugin + { + void CreateSubsystem(List descriptors, string id) + where TDescriptor : ISubsystemDescriptor + where TSubsystem : ISubsystem; + void StartSubsystem() where T : class, ISubsystem; + void StopSubsystem() where T : class, ISubsystem; + void DestroySubsystem() where T : class, ISubsystem; + } + + internal abstract class SubsystemController + { + protected readonly IOpenXRContext Context; + + public SubsystemController(IOpenXRContext context) + { + Context = context; + } + + public virtual void OnSubsystemCreate(ISubsystemPlugin plugin) + { + } + public virtual void OnSubsystemStart(ISubsystemPlugin plugin) + { + } + public virtual void OnSubsystemStop(ISubsystemPlugin plugin) + { + } + public virtual void OnSubsystemDestroy(ISubsystemPlugin plugin) + { + } + + internal static SubsystemController CreateFromInternalType(string fullName, IOpenXRContext context) + { + Type type = FindInternalPackageType(fullName); + + if (type != null) + { + // Throw exceptions if the type is not configured correctly + var ctor = type.GetConstructor(new Type[] { typeof(IOpenXRContext) }); + if (ctor == null) + { + throw new Exception($"Type {type.FullName} does not have a constructor that takes an IOpenXRContext"); + } + + return (SubsystemController)ctor.Invoke(new object[] { context }); + } + + return null; + } + + private static Type FindInternalPackageType(string fullName) + { + Type subsystemControllerType = typeof(T); + foreach (var assembly in AppDomain.CurrentDomain.GetAssemblies()) + { + if (assembly.FullName.StartsWith("Microsoft.MixedReality.OpenXR.Internal")) + { + foreach (var type in assembly.GetTypes()) + { + if (subsystemControllerType.IsAssignableFrom(type) && + type.FullName == fullName) + { + return type; + } + } + } + } + return null; + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SubsystemController.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SubsystemController.cs.meta new file mode 100644 index 0000000..e0e077a --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/SubsystemController.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 63b9ed13fdaa07c4da707bb2879a12e3 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/TrackingMapSubsystem.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/TrackingMapSubsystem.cs new file mode 100644 index 0000000..55947b1 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/TrackingMapSubsystem.cs @@ -0,0 +1,68 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Runtime.CompilerServices; +using System.Runtime.InteropServices; +using UnityEngine; +using UnityEngine.XR.OpenXR; + +namespace Microsoft.MixedReality.OpenXR +{ + internal class TrackingMapSubsystem : Disposable + { + private static MixedRealityFeaturePlugin Feature => OpenXRFeaturePlugin.Feature; + + internal static TrackingMapSubsystem TryCreateTrackingMapSubsystem() + { + if (!Feature.IsValidAndEnabled()) + { + Debug.LogWarning($"{MixedRealityFeaturePlugin.featureName} is not enabled."); + return null; + } + + return new TrackingMapSubsystem(); + } + + private TrackingMapSubsystem() + { + NativeLib.StartMapTrackingSubsystem(); + } + + internal bool SupportsApplicationExclusiveMaps() + { + return NativeLib.MapTrackingManagerSupportsApplicationExclusiveMaps(); + } + + internal TrackingMapType ActiveTrackingMapType + { + get => (TrackingMapType)NativeLib.MapTrackingManagerGetActiveTrackingMapType(); + } + + internal Guid ActivateApplicationExclusiveMap(Guid? existingMapId) + { + Guid mapId = existingMapId.HasValue ? existingMapId.Value : Guid.Empty; + if (NativeLib.MapTrackingManagerActivateExclusiveMap(ref mapId)) + { + return mapId; + } + else + { + // Known error cases: + // 1. Application-exclusive maps are not supported + // 2. Map transition already in progress + throw new InvalidOperationException(); + } + } + + internal void ActivateSharedMapAsync() + { + if (!NativeLib.MapTrackingManagerActivateSharedMap()) + { + // Known error cases: + // 1. Map transition already in progress + throw new InvalidOperationException(); + } + } + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/TrackingMapSubsystem.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/TrackingMapSubsystem.cs.meta new file mode 100644 index 0000000..05f90ca --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/TrackingMapSubsystem.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 8505e2b2a3d9478fb40851540efcc333 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ViewConfigurationSettings.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ViewConfigurationSettings.cs new file mode 100644 index 0000000..0ae6594 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ViewConfigurationSettings.cs @@ -0,0 +1,131 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using System.Runtime.InteropServices; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + // Used to report reprojection settings for a view configuration to the native layer + [StructLayout(LayoutKind.Sequential, Pack = 8)] + internal struct NativeReprojectionSettings + { + public ReprojectionMode reprojectionMode; + + public Vector3 reprojectionPlaneOverridePosition; + public byte reprojectionPlaneOverridePositionHasValue; + + public Vector3 reprojectionPlaneOverrideNormal; + public byte reprojectionPlaneOverrideNormalHasValue; + + public Vector3 reprojectionPlaneOverrideVelocity; + public byte reprojectionPlaneOverrideVelocityHasValue; + + internal NativeReprojectionSettings(ReprojectionSettings settings) : this() + { + reprojectionMode = settings.ReprojectionMode; + + if (settings.ReprojectionPlaneOverridePosition.HasValue) + { + reprojectionPlaneOverridePosition = settings.ReprojectionPlaneOverridePosition.Value; + reprojectionPlaneOverridePositionHasValue = 1; + } + + if (settings.ReprojectionPlaneOverrideNormal.HasValue) + { + reprojectionPlaneOverrideNormal = settings.ReprojectionPlaneOverrideNormal.Value; + reprojectionPlaneOverrideNormalHasValue = 1; + } + + if (settings.ReprojectionPlaneOverrideVelocity.HasValue) + { + reprojectionPlaneOverrideVelocity = settings.ReprojectionPlaneOverrideVelocity.Value; + reprojectionPlaneOverrideVelocityHasValue = 1; + } + } + } + + // Used to provide view configuration information from the native layer + internal class OpenXRViewConfiguration + { + private ViewConfigurationType m_viewConfigurationType; + private ReprojectionMode[] m_supportedReprojectionModes; + + public ViewConfigurationType ViewConfigurationType { get => m_viewConfigurationType; } + + internal bool HasTrackingFlags(NativeSpaceLocationFlags nativeSpaceLocationFlags) + { + NativeSpaceLocationFlags flags = NativeLib.GetViewTrackingFlags(m_viewConfigurationType); + return (flags & nativeSpaceLocationFlags) == nativeSpaceLocationFlags; + } + + public bool IsActive { get => NativeLib.GetViewConfigurationIsActive(m_viewConfigurationType); } + + public bool IsPrimary { get => NativeLib.GetViewConfigurationIsPrimary(m_viewConfigurationType); } + + public IReadOnlyList SupportedReprojectionModes { get => m_supportedReprojectionModes; } + + public float StereoSeparationAdjustment { get => NativeLib.GetStereoSeparationAdjustment(); } + + public OpenXRViewConfiguration(ViewConfigurationType viewConfigurationType) + { + m_viewConfigurationType = viewConfigurationType; + + uint numSupportedModes = NativeLib.GetSupportedReprojectionModesCount(m_viewConfigurationType); + m_supportedReprojectionModes = new ReprojectionMode[numSupportedModes]; + NativeLib.GetSupportedReprojectionModes(m_viewConfigurationType, m_supportedReprojectionModes, numSupportedModes); + } + + public void SetReprojectionSettings(ReprojectionSettings reprojectionSettings) + { + NativeLib.SetReprojectionSettings(m_viewConfigurationType, new NativeReprojectionSettings(reprojectionSettings)); + } + + public void SetStereoSeparationAdjustment(float stereoSeparationAdjustment) + { + NativeLib.SetStereoSeparationAdjustment(stereoSeparationAdjustment); + } + } + + internal class OpenXRViewConfigurationSettings : SubsystemController + { + private List m_enabledViewConfigurations = new List(); + private ViewConfiguration m_primaryViewConfiguration = null; + + public IReadOnlyList EnabledViewConfigurations => m_enabledViewConfigurations; + public ViewConfiguration PrimaryViewConfiguration => m_primaryViewConfiguration; + + public OpenXRViewConfigurationSettings(IOpenXRContext context) : base(context) + { + context.SessionBegun += Context_SessionBegun; + context.SessionEnding += Context_SessionEnding; + } + + private void Context_SessionBegun(IOpenXRContext sender, EventArgs args) + { + // Enabled view configurations are changed when session begin/end + uint viewConfigurationTypesCount = NativeLib.GetEnabledViewConfigurationTypesCount(); + ViewConfigurationType[] viewConfigurationTypes = new ViewConfigurationType[viewConfigurationTypesCount]; + NativeLib.GetEnabledViewConfigurationTypes(viewConfigurationTypes, viewConfigurationTypesCount); + + foreach (ViewConfigurationType viewConfigurationType in viewConfigurationTypes) + { + OpenXRViewConfiguration openxrViewConfiguration = new OpenXRViewConfiguration(viewConfigurationType); + ViewConfiguration viewConfiguration = new ViewConfiguration(openxrViewConfiguration); + m_enabledViewConfigurations.Add(viewConfiguration); + if (openxrViewConfiguration.IsPrimary) + { + m_primaryViewConfiguration = viewConfiguration; + } + } + } + + private void Context_SessionEnding(IOpenXRContext sender, EventArgs args) + { + m_enabledViewConfigurations.Clear(); + m_primaryViewConfiguration = null; + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ViewConfigurationSettings.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ViewConfigurationSettings.cs.meta new file mode 100644 index 0000000..5af7978 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/ViewConfigurationSettings.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 476890cd2491ad94980bd970901f111f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/XrSessionState.cs b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/XrSessionState.cs new file mode 100644 index 0000000..d0b88b9 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/XrSessionState.cs @@ -0,0 +1,65 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using UnityEngine.XR.OpenXR; + +namespace Microsoft.MixedReality.OpenXR +{ + /// + /// Represents the xr session state in its lifecycle. + /// Reference https://www.khronos.org/registry/OpenXR/specs/1.0/html/xrspec.html#session-lifecycle for more details on session state machine in OpenXR. + /// + internal enum XrSessionState : int + { + /// + /// Indicates an unknown state of session, typically means the session is not created yet. + /// + Unknown = 0, + + /// + /// Indicates that the runtime considers the session is idle. + /// Applications in this state should minimize resource consumption. + /// + Idle = 1, + + /// + /// Indicates that the runtime desires the application to prepare rendering resources, begin its session and synchronize its frame loop with the runtime. + /// Unity engine will handle the necessary preparation and begin a session. + /// + Ready = 2, + + /// + /// Indicates that the application has synchronized its frame loop with the runtime, but its frames are not visible to the user. + /// + Synchronized = 3, + + /// + /// Indicates that the application has synchronized its frame loop with the runtime, and the session's frames will be visible to the user, + /// but the session is not eligible to receive XR input. + /// + Visible = 4, + + /// + /// indicates that the application has synchronized its frame loop with the runtime, the session's frames will be visible to the user, + /// and the session is eligible to receive XR input. + /// + Focused = 5, + + /// + /// Indicates that the runtime has determined that the application should halt its rendering loop. + /// Unity engine will handle the stopping of a running session. + /// + Stopping = 6, + + /// + /// Indicates the runtime is no longer able to operate with the current session, for example due to the loss of a display hardware connection. + /// + LossPending = 7, + + /// + /// Indicates the runtime wishes the application to terminate its XR experience, typically due to a user request via a runtime user interface. + /// + Exiting = 8, + }; +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/Subsystems/XrSessionState.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/XrSessionState.cs.meta new file mode 100644 index 0000000..e7d96dc --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Subsystems/XrSessionState.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 234b40136eff31048b3ce1f0f45f59c4 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/UnitySubsystemsManifest.json b/com.microsoft.mixedreality.openxr/Runtime/UnitySubsystemsManifest.json new file mode 100644 index 0000000..85ec316 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/UnitySubsystemsManifest.json @@ -0,0 +1,15 @@ +{ + "name": "OpenXR Extension", + "version": "1.11.1", + "libraryName": "MicrosoftOpenXRPlugin", + "inputs": [ + { + "id": "OpenXR Input Extension" + } + ], + "meshings":[ + { + "id": "OpenXR Mesh Extension" + } + ] +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/UnitySubsystemsManifest.json.meta b/com.microsoft.mixedreality.openxr/Runtime/UnitySubsystemsManifest.json.meta new file mode 100644 index 0000000..8b4ca32 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/UnitySubsystemsManifest.json.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: 52d6bedd3d439c94695fba295f6ca1d9 +TextScriptImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities.meta b/com.microsoft.mixedreality.openxr/Runtime/Utilities.meta new file mode 100644 index 0000000..8c64569 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 7d6b6142fdfb53e4d8969c3b05182c06 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/BuildProcessorHelpers.cs b/com.microsoft.mixedreality.openxr/Runtime/Utilities/BuildProcessorHelpers.cs new file mode 100644 index 0000000..bcc28a5 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/BuildProcessorHelpers.cs @@ -0,0 +1,190 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR +using System; +using System.Collections.Generic; +using System.IO; +using System.Text; +using System.Xml; +using UnityEditor; +using UnityEngine.XR.Management; +using UnityEngine.XR.OpenXR; +using UnityEngine.XR.OpenXR.Features; + +namespace Microsoft.MixedReality.OpenXR.Editor +{ + internal static class BuildProcessorHelpers + { + internal static bool IsFeatureEnabled() where T : OpenXRFeature + { + T feature = GetOpenXRFeature(); + return feature != null && feature.enabled; + } + + // Get the feature from the OpenXRSettings for the current or provided build target group. + internal static T GetOpenXRFeature(BuildTargetGroup? buildTargetGroup = null, bool returnNullWhenLoaderDisabled = true) where T : OpenXRFeature + { + foreach (OpenXRFeature feature in GetOpenXRFeatures(buildTargetGroup, returnNullWhenLoaderDisabled)) + { + if (feature is T) + { + return feature as T; + } + } + + return null; + } + + internal static OpenXRFeature[] GetOpenXRFeatures(BuildTargetGroup? buildTargetGroup = null, bool returnNullWhenLoaderDisabled = true) + { + BuildTargetGroup providedOrDefaultTargetGroup = buildTargetGroup ?? BuildPipeline.GetBuildTargetGroup(EditorUserBuildSettings.activeBuildTarget); + + if (returnNullWhenLoaderDisabled && !IsLoaderEnabledForTarget(providedOrDefaultTargetGroup)) + { + return Array.Empty(); + } + + EditorBuildSettings.TryGetConfigObject(Constants.k_SettingsKey, out UnityEngine.Object obj); + OpenXRSettings openXRSettings = null; + if (obj is IPackageSettings packageSettings) + { + openXRSettings = packageSettings.GetSettingsForBuildTargetGroup(providedOrDefaultTargetGroup); + } + + return openXRSettings != null ? openXRSettings.GetFeatures() : Array.Empty(); + } + + internal static bool IsLoaderEnabledForTarget(BuildTargetGroup buildTargetGroup) + { + XRManagerSettings settings = XRSettingsHelpers.GetOrCreateXRManagerSettings(buildTargetGroup); + if (settings == null) + { + return false; + } + + IReadOnlyList loaders = settings.activeLoaders; + for (int i = 0; i < loaders.Count; i++) + { + if (loaders[i] is OpenXRLoaderBase) + { + return true; + } + } + + return false; + } + + internal class AndroidXmlDocument : XmlDocument + { + private readonly string m_Path; + protected XmlNamespaceManager nsMgr; + public const string AndroidXmlNamespace = "http://schemas.android.com/apk/res/android"; + + public AndroidXmlDocument(string path) + { + m_Path = path; + using (var reader = new XmlTextReader(m_Path)) + { + reader.Read(); + Load(reader); + } + + nsMgr = new XmlNamespaceManager(NameTable); + nsMgr.AddNamespace("android", AndroidXmlNamespace); + } + + public string Save() + { + return SaveAs(m_Path); + } + + public string SaveAs(string path) + { + using (var writer = new XmlTextWriter(path, new UTF8Encoding(false))) + { + writer.Formatting = Formatting.Indented; + Save(writer); + } + + return path; + } + } + + internal class AndroidManifest : AndroidXmlDocument + { + internal readonly XmlElement RootElement; + internal readonly XmlElement ApplicationElement; + internal readonly XmlElement IntentFilterElement; + + internal AndroidManifest(string path) : base(path) + { + RootElement = SelectSingleNode("/manifest") as XmlElement; + ApplicationElement = SelectSingleNode("/manifest/application") as XmlElement; + IntentFilterElement = SelectSingleNode("/manifest/application/activity/intent-filter") as XmlElement; + } + + private static string _manifestFilePath; + + internal static string GetManifestPath(string basePath) + { + if (!string.IsNullOrEmpty(_manifestFilePath)) return _manifestFilePath; + + var pathBuilder = new StringBuilder(basePath); + pathBuilder.Append(Path.DirectorySeparatorChar).Append("src"); + pathBuilder.Append(Path.DirectorySeparatorChar).Append("main"); + pathBuilder.Append(Path.DirectorySeparatorChar).Append("AndroidManifest.xml"); + _manifestFilePath = pathBuilder.ToString(); + + return _manifestFilePath; + } + + internal XmlNode GetOrCreateChild(XmlNode node, string name) + { + foreach (XmlNode child in node.ChildNodes) + { + if (child.Name == name) + { + return child; + } + } + return node.AppendChild(CreateElement(name)); + } + + internal XmlAttribute CreateAndroidAttribute(string key, string value) + { + XmlAttribute attr = CreateAttribute("android", key, AndroidXmlNamespace); + attr.Value = value; + return attr; + } + + internal static bool HasAttribute(XmlNode node, string name, string value) + { + foreach (XmlAttribute attribute in node.Attributes) + { + if (attribute.Name == name && attribute.Value == value) + { + return true; + } + } + return false; + } + + // return false if attribute is not found. + internal static bool SetAttribute(XmlNode node, string name, string value) + { + foreach (XmlAttribute attribute in node.Attributes) + { + if (attribute.Name == name) + { + attribute.Value = value; + return true; + } + } + return false; + } + } + } +} + +#endif \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/BuildProcessorHelpers.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Utilities/BuildProcessorHelpers.cs.meta new file mode 100644 index 0000000..3c1be11 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/BuildProcessorHelpers.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 977e31ae048f2f849aefab855623ba5e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/Disposable.cs b/com.microsoft.mixedreality.openxr/Runtime/Utilities/Disposable.cs new file mode 100644 index 0000000..01fe6c8 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/Disposable.cs @@ -0,0 +1,50 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; + +namespace Microsoft.MixedReality.OpenXR +{ + internal abstract class Disposable : IDisposable + { + protected bool disposedValue { get; private set; } + + protected virtual void DisposeManagedResources() + { + // Dispose managed state (managed objects) + } + + protected virtual void DisposeNativeResources() + { + // Free unmanaged resources (unmanaged objects) + // Set large fields to null + } + + private void Dispose(bool disposing) + { + if (!disposedValue) + { + if (disposing) + { + DisposeManagedResources(); + } + + DisposeNativeResources(); + disposedValue = true; + } + } + + ~Disposable() + { + // Do not change this code. Put cleanup code in 'DisposeManagedResources or DisposeNativeResources' methods + Dispose(disposing: false); + } + + public void Dispose() + { + // Do not change this code. Put cleanup code in 'DisposeManagedResources or DisposeNativeResources' methods + Dispose(disposing: true); + System.GC.SuppressFinalize(this); + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/Disposable.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Utilities/Disposable.cs.meta new file mode 100644 index 0000000..644b449 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/Disposable.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 83a1ea41e10af5f4db8b9bf90d5ea80e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/DocURLAttribute.cs b/com.microsoft.mixedreality.openxr/Runtime/Utilities/DocURLAttribute.cs new file mode 100644 index 0000000..0d8696c --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/DocURLAttribute.cs @@ -0,0 +1,19 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + [AttributeUsage(AttributeTargets.Field)] + internal class DocURLAttribute : PropertyAttribute + { + public string Url { get; } + + public DocURLAttribute(string url) + { + Url = url; + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/DocURLAttribute.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Utilities/DocURLAttribute.cs.meta new file mode 100644 index 0000000..a10792f --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/DocURLAttribute.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 5cf62f8781f52464490753f91e197d6f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/EditorDrawerVisibleToBuildTargetAttribute.cs b/com.microsoft.mixedreality.openxr/Runtime/Utilities/EditorDrawerVisibleToBuildTargetAttribute.cs new file mode 100644 index 0000000..eef8a6f --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/EditorDrawerVisibleToBuildTargetAttribute.cs @@ -0,0 +1,24 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR + +using System; +using UnityEditor; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + [AttributeUsage(AttributeTargets.Field)] + internal class EditorDrawerVisibleToBuildTargetAttribute : PropertyAttribute + { + public BuildTargetGroup[] BuildTargetGroups { get; } + + public EditorDrawerVisibleToBuildTargetAttribute(params BuildTargetGroup[] buildTargetGroups) + { + BuildTargetGroups = buildTargetGroups; + } + } +} + +#endif diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/EditorDrawerVisibleToBuildTargetAttribute.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Utilities/EditorDrawerVisibleToBuildTargetAttribute.cs.meta new file mode 100644 index 0000000..6c50fd9 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/EditorDrawerVisibleToBuildTargetAttribute.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 1c4f8f61c8d11b642b0fc7349bc491be +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/FeatureUtils.cs b/com.microsoft.mixedreality.openxr/Runtime/Utilities/FeatureUtils.cs new file mode 100644 index 0000000..7332114 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/FeatureUtils.cs @@ -0,0 +1,32 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using UnityEngine.XR.ARSubsystems; +using UnityEngine.XR.OpenXR.Features; + +namespace Microsoft.MixedReality.OpenXR +{ + internal static class FeatureUtils + { + // Note:Guid.Empty is converted into TrackableId.invalidId + internal static TrackableId ToTrackableId(Guid guid) + { + byte[] bytes = guid.ToByteArray(); + ulong subId1 = BitConverter.ToUInt64(bytes, 0); + ulong subId2 = BitConverter.ToUInt64(bytes, 8); + return new TrackableId(subId1, subId2); + } + + // Note:TrackableId.invalidId is converted into Guid.Empty + internal static Guid ToGuid(TrackableId id) + { + byte[] bytes = new byte[16]; + Array.Copy(BitConverter.GetBytes(id.subId1), 0, bytes, 0, 8); + Array.Copy(BitConverter.GetBytes(id.subId2), 0, bytes, 8, 8); + return new Guid(bytes); + } + + internal static bool IsValidAndEnabled(this OpenXRFeature plugin) => plugin != null && plugin.enabled; + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/FeatureUtils.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Utilities/FeatureUtils.cs.meta new file mode 100644 index 0000000..5fa34d1 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/FeatureUtils.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: a92ec9fecf231c544ba18b8e3fc4446c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/FindObjectUtility.cs b/com.microsoft.mixedreality.openxr/Runtime/Utilities/FindObjectUtility.cs new file mode 100644 index 0000000..c724a90 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/FindObjectUtility.cs @@ -0,0 +1,64 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using UnityEngine; +using System; + +namespace Microsoft.MixedReality.OpenXR +{ + // A static utility used to avoid deprecated Find Object functions in favor of replacements introduced in Unity >= 2021.3.18. + internal static class FindObjectUtility + { + + // Returns the first object matching the specified type. + // If Unity >= 2021.3.18, calls FindFirstObjectByType. Otherwise calls FindObjectOfType. + // includeInactive - If true, inactive objects will be included in the search. False by default. + internal static T FindFirstObjectByType(bool includeInactive = false) where T : Component + { +#if UNITY_2021_3_18_OR_NEWER + return UnityEngine.Object.FindFirstObjectByType(includeInactive ? FindObjectsInactive.Include : FindObjectsInactive.Exclude); +#else + return UnityEngine.Object.FindObjectOfType(includeInactive); +#endif + } + + // Returns an object matching the specified type. + // If Unity >= 2021.3.18, calls FindAnyObjectByType. Otherwise calls FindObjectOfType. + // includeInactive - If true, inactive objects will be included in the search. False by default. + internal static T FindAnyObjectByType(bool includeInactive = false) where T : Component + { +#if UNITY_2021_3_18_OR_NEWER + return UnityEngine.Object.FindAnyObjectByType(includeInactive ? FindObjectsInactive.Include : FindObjectsInactive.Exclude); +#else + return UnityEngine.Object.FindObjectOfType(includeInactive); +#endif + } + + // Returns all objects matching the specified type. + // If Unity >= 2021.3.18, calls FindObjectsByType. Otherwise calls FindObjectsOfType. + // includeInactive - If true, inactive objects will be included in the search. False by default. + // sort - If false, results will not sorted by InstanceID. True by default. + internal static T[] FindObjectsByType(bool includeInactive = false, bool sort = true) where T : Component + { +#if UNITY_2021_3_18_OR_NEWER + return UnityEngine.Object.FindObjectsByType(includeInactive ? FindObjectsInactive.Include : FindObjectsInactive.Exclude, sort ? FindObjectsSortMode.InstanceID : FindObjectsSortMode.None); +#else + return UnityEngine.Object.FindObjectsOfType(includeInactive); +#endif + } + + // Returns all objects matching the specified type. + // If Unity >= 2021.3.18, calls FindObjectsByType. Otherwise calls FindObjectsOfType. + // includeInactive - If true, inactive objects will be included in the search. False by default. + // sort - If false, results will not sorted by InstanceID. True by default. + // type - The type to search for. + internal static UnityEngine.Object[] FindObjectsByType(Type type, bool includeInactive = false, bool sort = true) + { +#if UNITY_2021_3_18_OR_NEWER + return UnityEngine.Object.FindObjectsByType(type, includeInactive ? FindObjectsInactive.Include : FindObjectsInactive.Exclude, sort ? FindObjectsSortMode.InstanceID : FindObjectsSortMode.None); +#else + return UnityEngine.Object.FindObjectsOfType(type, includeInactive); +#endif + } + } +} \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/FindObjectUtility.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Utilities/FindObjectUtility.cs.meta new file mode 100644 index 0000000..c7649ee --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/FindObjectUtility.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 387b311e2f49d074c829816d25698835 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/LabelWidthAttribute.cs b/com.microsoft.mixedreality.openxr/Runtime/Utilities/LabelWidthAttribute.cs new file mode 100644 index 0000000..9b1b587 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/LabelWidthAttribute.cs @@ -0,0 +1,19 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +using System; +using UnityEngine; + +namespace Microsoft.MixedReality.OpenXR +{ + [AttributeUsage(AttributeTargets.Field)] + internal class LabelWidthAttribute : PropertyAttribute + { + public float Width { get; } + + public LabelWidthAttribute(float width) + { + Width = width; + } + } +} diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/LabelWidthAttribute.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Utilities/LabelWidthAttribute.cs.meta new file mode 100644 index 0000000..d320879 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/LabelWidthAttribute.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 26632e2942b82cf43ac9e470d68d4c60 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/XRSettingsHelpers.cs b/com.microsoft.mixedreality.openxr/Runtime/Utilities/XRSettingsHelpers.cs new file mode 100644 index 0000000..3975f96 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/XRSettingsHelpers.cs @@ -0,0 +1,75 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License. + +#if UNITY_EDITOR + +using UnityEditor; +using UnityEditor.XR.Management; +using UnityEngine.XR.Management; + +namespace Microsoft.MixedReality.OpenXR +{ + internal static class XRSettingsHelpers + { + /// + /// Provides the XRGeneralSettings corresponding to the specified BuildTargetGroup. + /// If the XRGeneralSettings asset wasn't previously created, this ensures it's created. + /// + public static XRGeneralSettings GetOrCreateXRGeneralSettings(BuildTargetGroup targetGroup) + { + XRGeneralSettings settings = XRGeneralSettingsPerBuildTarget.XRGeneralSettingsForBuildTarget(targetGroup); + + if (settings == null) + { + XRGeneralSettingsPerBuildTarget generalSettings = GetXRGeneralSettingsPerBuildTarget(); + + if (generalSettings != null && !generalSettings.HasSettingsForBuildTarget(targetGroup)) + { + generalSettings.CreateDefaultSettingsForBuildTarget(targetGroup); + } + + settings = XRGeneralSettingsPerBuildTarget.XRGeneralSettingsForBuildTarget(targetGroup); + } + + return settings; + } + + /// + /// Provides the XRManagerSettings corresponding to the specified BuildTargetGroup. + /// If the XRManagerSettings asset wasn't previously created, this ensures it's created. + /// + public static XRManagerSettings GetOrCreateXRManagerSettings(BuildTargetGroup targetGroup) + { + XRGeneralSettings settings = GetOrCreateXRGeneralSettings(targetGroup); + + if (settings != null && settings.AssignedSettings == null) + { + XRGeneralSettingsPerBuildTarget generalSettings = GetXRGeneralSettingsPerBuildTarget(); + + if (generalSettings != null && !generalSettings.HasManagerSettingsForBuildTarget(targetGroup)) + { + generalSettings.CreateDefaultManagerSettingsForBuildTarget(targetGroup); + } + } + + return settings != null ? settings.AssignedSettings : null; + } + + /// + /// Tries to read out the XRGeneralSettingsPerBuildTarget from XRGeneralSettingsPerBuildTarget. + /// If the config object hasn't been stored yet, the XR Plug-in Management window is opened to trigger its creation. + /// + private static XRGeneralSettingsPerBuildTarget GetXRGeneralSettingsPerBuildTarget() + { + if (!EditorBuildSettings.TryGetConfigObject(XRGeneralSettings.k_SettingsKey, out XRGeneralSettingsPerBuildTarget generalSettings)) + { + SettingsService.OpenProjectSettings("Project/XR Plug-in Management"); + EditorBuildSettings.TryGetConfigObject(XRGeneralSettings.k_SettingsKey, out generalSettings); + } + + return generalSettings; + } + } +} + +#endif // UNITY_EDITOR diff --git a/com.microsoft.mixedreality.openxr/Runtime/Utilities/XRSettingsHelpers.cs.meta b/com.microsoft.mixedreality.openxr/Runtime/Utilities/XRSettingsHelpers.cs.meta new file mode 100644 index 0000000..0b11a14 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/Utilities/XRSettingsHelpers.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 249441f81553f814c9592197e8a7f83f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/com.microsoft.mixedreality.openxr/Runtime/android.meta b/com.microsoft.mixedreality.openxr/Runtime/android.meta new file mode 100644 index 0000000..4e47e25 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/android.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: df1c6739f809b6547b4eda0b4d2632f1 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/android/arm64.meta b/com.microsoft.mixedreality.openxr/Runtime/android/arm64.meta new file mode 100644 index 0000000..4890e80 --- /dev/null +++ b/com.microsoft.mixedreality.openxr/Runtime/android/arm64.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 09b936b5c28050844880225e639eedaf +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: \ No newline at end of file diff --git a/com.microsoft.mixedreality.openxr/Runtime/android/arm64/libMicrosoftOpenXRPlugin.so b/com.microsoft.mixedreality.openxr/Runtime/android/arm64/libMicrosoftOpenXRPlugin.so new file mode 100644 index 0000000000000000000000000000000000000000..60caecdf54b52b75748f1bb60981a428c6749183 GIT binary patch literal 1255096 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 z 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